| 
									
										
										
										
											2016-02-05 23:27:06 -08:00
										 |  |  |  | include ../_util-fns | 
					
						
							| 
									
										
										
										
											2015-10-16 20:39:30 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  |   We’ve all used a form to log in, submit a help request, place an order, book a flight, | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   schedule a meeting, and perform countless other data entry tasks. | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   Forms are the mainstay of business applications. | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   表单是商业应用的支柱,我们用它来执行登录、求助、下单、预订机票、安排会议,以及不计其数的其它数据录入任务。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   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. | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   任何经验丰富的 Web 开发人员都能使用适当的标签拼凑出 HTML 表单。 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   但是,要想做出具有贴心的数据输入体验的表单, | 
					
						
							|  |  |  |  |   引导用户明晰、高效地完成表单背后的工作流程,挑战就大多了。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   *That* takes design skills that are, to be frank, well out of scope for this guide. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   坦白地讲,*这当中*所需要的设计技能超出了本章的范围。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   It also takes framework support for | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   **two-way data binding, change tracking, validation, and error handling** | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   ... which we shall cover in this guide on Angular forms. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   **双向数据绑定、变更跟踪、有效性验证和错误处理**等功能离不开框架的支持。 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   本章将介绍 Angular 表单相关的内容。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We will build a simple form from scratch, one step at a time. Along the way we'll learn how to: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   下面,从零开始,一步一步构建出一个简单的表单。在这个过程中,我们将学会如何: | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   - Build an Angular form with a component and template | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |     使用组件和模板构建 Angular 表单 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |        | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   - Use `ngModel` to create two-way data bindings for reading and writing input control values | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |     用`ngModel`创建双向数据绑定,以读取和写入输入控件的值 | 
					
						
							|  |  |  |  |        | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   - Track state changes and the validity of form controls | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |     跟踪状态的变化,并验证表单控件 | 
					
						
							|  |  |  |  |        | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   - Provide visual feedback using special CSS classes that track the state of the controls | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |     使用特殊的CSS类来跟踪控件的状态并给出视觉反馈 | 
					
						
							|  |  |  |  |        | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   - Display validation errors to users and enable/disable form controls | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |     向用户显示验证错误提示,以及启用/禁用表单控件 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |        | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   - Share information across HTML elements using template reference variables | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |     使用模板引用变量在 HTML 元素之间共享信息 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-22 15:41:13 -07:00
										 |  |  |  |   Run the <live-example></live-example>. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-23 14:02:54 +01:00
										 |  |  |  |   运行<live-example>在线例子</live-example> | 
					
						
							| 
									
										
										
										
											2016-06-19 11:50:27 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-16 20:39:30 -07:00
										 |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Template-driven forms | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-24 22:53:59 +08:00
										 |  |  |  |   ## 模板驱动的表单 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   Many of us will build forms by writing templates in the Angular [template syntax](./template-syntax.html) with | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   the form-specific directives and techniques described in this guide. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   通常,使用 Angular [模板语法](./template-syntax.html)编写模板,结合本章所描述的表单专用指令和技术来构建表单。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | .l-sub-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |     That's not the only way to create a form but it's the way we'll cover in this guide. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |     这不是创建表单的唯一方式,本章中只介绍模板驱动的表单。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We can build almost any form we need with an Angular template — login forms, contact forms, pretty much any business form. | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   We can lay out the controls creatively, bind them to data, specify validation rules and display validation errors, | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   conditionally enable or disable specific controls, trigger built-in visual feedback, and much more. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   利用 Angular 模板,可以构建几乎所有表单 — 登录表单、联系人表单…… 以及任何的商务表单。 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   可以创造性的摆放各种控件、把它们绑定到数据、指定校验规则、显示校验错误、有条件的禁用或 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   启用特定的控件、触发内置的视觉反馈等等,不胜枚举。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   It will be pretty easy because Angular handles many of the repetitive, boilerplate tasks we'd | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   otherwise wrestle with ourselves. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   它用起来很简单,这是因为 Angular 处理了大多数重复、单调的任务,这让我们可以不必亲自操刀、身陷其中。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We'll discuss and learn to build a template-driven form that looks like this: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-24 22:53:59 +08:00
										 |  |  |  |   我们将讨论和学习构建如下的“模板驱动”表单: | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2015-12-12 22:18:44 +00:00
										 |  |  |  |   img(src="/resources/images/devguide/forms/hero-form-1.png" width="400px" alt="Clean Form") | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Here at the *Hero Employment Agency* we use this form to maintain personal information about heroes. | 
					
						
							|  |  |  |  |   Every hero needs a job. It's our company mission to match the right hero with the right crisis! | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   这里是*英雄职业介绍所*,使用这个表单来维护候选英雄们的个人信息。每个英雄都需要一份工作。 | 
					
						
							|  |  |  |  |   公司的任务就是让适当的英雄去解决它/她所擅长应对的危机! | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   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. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   表单中的三个字段,其中两个是必填的。必填的字段在左侧有个绿色的竖条,方便用户分辨哪些是必填项。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   If we delete the hero name, the form displays a validation error in an attention-grabbing style: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   如果删除了英雄的名字,表单就会用醒目的样式把验证错误显示出来。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2016-04-24 22:53:59 +08:00
										 |  |  |  |   img(src="/resources/images/devguide/forms/hero-form-2.png" width="400px" alt="无效!名字是必填项") | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Note that the submit button is disabled, and the "required" bar to the left of the input control changed from green to red. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-24 22:53:59 +08:00
										 |  |  |  |   注意,提交按钮被禁用了,而且输入控件左侧的“必填”条从绿色变为了红色。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | .l-sub-section | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   :marked | 
					
						
							|  |  |  |  |     We'll customize the colors and location of the "required" bar with standard CSS. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |     稍后,会使用标准 CSS 来定制“必填”条的颜色和位置。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We'll build this form in small steps: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   我们将一点点构建出此表单: | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Create the `Hero` model class. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |      创建`Hero`模型类 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Create the component that controls the form. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      创建控制此表单的组件。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Create a template with the initial form layout. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      创建具有初始表单布局的模板。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Bind data properties to each form control using the `ngModel` two-way data binding syntax. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      使用`ngModel`双向数据绑定语法把数据属性绑定到每个表单输入控件。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Add a `name` attribute to each form input control. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      往每个表单输入控件上添加`name`属性 (attribute)。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Add custom CSS to provide visual feedback. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      添加自定义 CSS 来提供视觉反馈。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Show and hide validation error messages. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      显示和隐藏有效性验证的错误信息。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Handle form submission with **ngSubmit**. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      使用 **ngSubmit** 处理表单提交。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Disable the form’s submit button until the form is valid. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      禁用此表单的提交按钮,直到表单变为有效。 | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  |   ## Setup | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:52:43 +01:00
										 |  |  |  |   ## 搭建 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   Follow the [setup](setup.html) instructions for creating a new project | 
					
						
							|  |  |  |  |   named <span ngio-ex>angular-forms</span>. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-27 10:38:15 +00:00
										 |  |  |  |   按照[搭建本地开发环境](setup.html)的说明,创建一个名为<span ngio-ex>angular-forms</span>的新项目。 | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Create the Hero model class | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   ## 创建 Hero 模型类 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   As users enter form data, we'll capture their changes and update an instance of a model. | 
					
						
							|  |  |  |  |   We can't lay out the form until we know what the model looks like. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   当用户输入表单数据时,需要捕获它们的变化,并更新到模型的实例中。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   除非知道模型里有什么,否则无法设计表单的布局。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   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`). | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   最简单的模型是个“属性包”,用来存放应用中一件事物的事实。 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   这里使用三个必备字段 (`id`、`name`、`power`),和一个可选字段 (`alterEgo`,译注:中文含义是第二人格,例如 X 战警中的 Jean / 黑凤凰)。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   In the `!{_appDir}` directory, create the following file with the given content: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   在应用文件夹中创建下列文件: | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExample('src/app/hero.ts') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   It's an anemic model with few requirements and no behavior. Perfect for our demo. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   这是一个少量需求和零行为的贫血模型。对演示来说很完美。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   The TypeScript compiler generates a public field for each `public` constructor parameter and | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   assigns the parameter’s value to that field automatically when we create new heroes. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   TypeScript 编译器为每个`public`构造函数参数生成一个公共字段,在创建新的英雄实例时,自动把参数值赋给这些公共字段。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   The `alterEgo` is optional, so the constructor lets us omit it; note the (?) in `alterEgo?`. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   `alterEgo`是可选的,调用构造函数时可省略,注意`alterEgo?`中的问号 (?)。 | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   We can create a new hero like this: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   可以这样创建新英雄: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.ts', 'SkyDog', '') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Create a form component | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   ## 创建表单组件 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   An Angular form has two parts: an HTML-based _template_ and a component _class_ | 
					
						
							|  |  |  |  |   to handle data and user interactions programmatically. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   Angular 表单分为两部分:基于 HTML 的*模板*和组件*类*,用来程序处理数据和用户交互。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   We begin with the class because it states, in brief, what the hero editor can do. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   先从组件类开始,是因为它可以简要说明英雄编辑器能做什么。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Create the following file with the given content: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   创建下列文件: | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.ts', 'v1') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   There’s nothing special about this component, nothing form-specific, | 
					
						
							|  |  |  |  |   nothing to distinguish it from any component we've written before. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   这个组件没有什么特别的地方,没有表单相关的东西,与之前写过的组件没什么不同。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Understanding this component requires only the Angular concepts covered in previous guides. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   只需要前面章节中学过的概念,就可以完全理解这个组件: | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. The code imports the Angular core library, and the `Hero` model we just created. | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      这段代码导入了Angular核心库以及我们刚刚创建的`Hero`模型。 | 
					
						
							| 
									
										
										
										
											2016-08-16 10:38:04 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |   1. The `@Component` selector value of "hero-form" means we can drop this form in a parent template with a `<hero-form>` tag. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |      `@Component`选择器"hero-form"表示可以用`<hero-form>`标签把这个表单放进父模板。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 18:51:54 -07:00
										 |  |  |  |   1. The `moduleId: module.id` property sets the base for module-relative loading of the `templateUrl`. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |      `moduleId: module.id`属性设置了基地址,用于从相对模块路径加载`templateUrl`。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. The `templateUrl` property points to a separate file for the template HTML. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      `templateUrl`属性指向一个独立的 HTML 模板文件。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. We defined dummy data for `model` and `powers`, as befits a demo. | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   Down the road, we can inject a data service to get and save real data | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   or perhaps expose these properties as | 
					
						
							|  |  |  |  |   [inputs and outputs](./template-syntax.html#inputs-outputs) for binding to a | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   parent component. None of this concerns us now and these future changes won't affect our form. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      为`model`和`powers`定义了供演示用的假数据。 将来,可以注入服务来获取和保存真实数据, 或者暴露这些属性为[输入与输出属性](./template-syntax.html#inputs-outputs),绑定到父组件上。 先不关心这些,因为这些未来的变化不会影响到表单。 | 
					
						
							|  |  |  |  |       | 
					
						
							|  |  |  |  |   1. We threw in a `diagnostic` property  to return a JSON representation of our model. | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   It'll help us see what we're doing during our development; we've left ourselves a cleanup note to discard it later. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      在最后增加`diagnostic`属性,它返回这个模型的 JSON 形式。在开发过程中,它用于调试,最后清理时会丢弃它。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   ### Why the separate template file? | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   ### 为何分离模板文件? | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   Why don't we write the template inline in the component file as we often do elsewhere? | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   为什么不与我们在其他地方常常做的那样,以内联的方式把模板写在组件文件中呢? | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   There is no “right” answer for all occasions. We like inline templates when they are short. | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   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. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   没有什么答案在所有场合都总是“正确”的。当模板足够短的时候,内联形式更招人喜欢。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   但大多数的表单模板都不短。通常,TypeScript 和 JavaScript 文件不是写(读)大型 HTML 的好地方, | 
					
						
							|  |  |  |  |   而且没有几个编辑器能对混写的 HTML 和代码提供足够的帮助。 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   我们还是喜欢内容清晰、目标明确的短文件,像这个一样。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   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`. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   就算是在仅仅显示少数表单项目时,表单模板一般都比较庞大。所以通常最好的方式是将 HTML 模板放到单独的文件中。 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   一会儿将编写这个模板文件。在这之前,先退一步,再看看`app.module.ts`和`app.component.ts`,让它们使用新的`HeroFormComponent`。 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | .l-main-section | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Revise *app.module.ts* | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   ## 修改 *app.module.ts* | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   `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`. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   `app.module.ts`定义了应用的根模块。其中标识即将用到的外部模块,以及声明属于本模块中的组件,例如`HeroFormComponent`。 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   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. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   因为模板驱动的表单位于它们自己的模块,所以在使用表单之前,需要将`FormsModule`添加到应用模块的`imports`数组中。 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   Replace the contents of the "QuickStart" version with the following: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-12 07:57:51 +08:00
										 |  |  |  |   把“快速起步”版的文件替换为如下内容: | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExample('forms/ts/src/app/app.module.ts', null, 'src/app/app.module.ts') | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							|  |  |  |  | .l-sub-section | 
					
						
							|  |  |  |  |   :marked | 
					
						
							|  |  |  |  |     There are three changes: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-12 07:57:51 +08:00
										 |  |  |  |     有三处更改: | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     1. We import `FormsModule` and our new `HeroFormComponent`. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |        导入`FormsModule`和新组件`HeroFormComponent`。 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     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`. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |        把`FormsModule`添加到`ngModule`装饰器的`imports`列表中,这样应用就能访问模板驱动表单的所有特性,包括`ngModel`。 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     1. We add the `HeroFormComponent` to the list of `declarations` defined in the `ngModule` decorator. This makes | 
					
						
							|  |  |  |  |     the `HeroFormComponent` component visible throughout this module. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |        把`HeroFormComponent`添加到`ngModule`装饰器的`declarations`列表中,使`HeroFormComponent`组件在整个模块中可见。 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | .alert.is-important | 
					
						
							|  |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |     If a component, directive, or pipe belongs to a module in the `imports` array, _DON'T_ re-declare it in the `declarations` array. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     If you wrote it and it should belong to this module, _DO_ declare it in the `declarations` array. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |     如果组件、指令或管道出现在模块的`imports`数组中,_不要_把它声明在`declarations`数组中。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     如果它是你自己写的,并且属于当前模块,_就要_把它声明在`declarations`数组中。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Revise *app.component.ts* | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   ## 修改 *app.component.ts* | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   `AppComponent` is the application's root component. It will host our new `HeroFormComponent`. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   `AppComponent`是应用的根组件,`HeroFormComponent`将被放在其中。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  |   Replace the contents of the "QuickStart" version with the following: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   把"快速起步"的版本内容替换成下列代码:+makeExample('src/app/app.component.ts') | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  | .l-sub-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |     There are only two changes. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |     仅有的两处修改。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-16 15:50:11 +03:00
										 |  |  |  |     1. The `template` is simply the new element tag identified by the component's `selector` property. | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  |     This will display the hero form when the application component is loaded. | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |     We've also dropped the `name` field from the class body. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  |        `template`中只有新元素标签,即组件的`selector`属性。当应用组件被加载时,将显示这个英雄表单。 | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Create an initial HTML form template | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   ## 创建初始 HTML 表单模板 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Create the new template file with the following contents: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   用下列内容新建模板文件: | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExample('src/app/hero-form.component.html', 'start') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   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. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   这只是一段普通的旧式 HTML 5 代码。这里有两个`Hero`字段,`name`和`alterEgo`,供用户输入。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   The *Name* `<input>` control has the HTML5 `required` attribute; | 
					
						
							|  |  |  |  |   the *Alter Ego* `<input>` control does not because `alterEgo` is optional. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   *Name* `<input>`控件具有 HTML5 的`required`属性;但 *Alter Ego* `<input>`控件没有,因为`alterEgo`字段是可选的。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  |   We've got a *Submit* button at the bottom with some classes on it for styling. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   底部有个 *Submit* 按钮,具有一些 CSS 样式类。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   **We are not using Angular yet**. There are no bindings, no extra directives, just layout. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   **还没有真正用到Angular**。没有绑定,没有额外的指令,只有布局。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-22 15:41:13 -07:00
										 |  |  |  |   The `container`, `form-group`, `form-control`, and `btn` classes | 
					
						
							| 
									
										
										
										
											2016-04-18 12:33:27 -07:00
										 |  |  |  |   come from [Twitter Bootstrap](http://getbootstrap.com/css/). Purely cosmetic. | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   We're using Bootstrap to give the form a little style! | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   `container`、`form-group`、`form-control`和`btn`类来自 [Twitter Bootstrap](http://getbootstrap.com/css/)。纯粹是装饰。 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   我们使用 Bootstrap 来美化表单。嘿,一点样式都没有的表单算个啥! | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | .callout.is-important | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   header Angular forms do not require a style library | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   header Angular 表单不需要任何样式库 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   :marked | 
					
						
							|  |  |  |  |     Angular makes no use of the `container`, `form-group`, `form-control`, and `btn` classes or | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |     the styles of any external library. Angular apps can use any CSS library, or none at all. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     Angular 不需要`container`、`form-group`、`form-control`和`btn`类, | 
					
						
							|  |  |  |  |     或者外部库的任何样式。Angular 应用可以使用任何 CSS 库…… ,或者啥都不用。 | 
					
						
							| 
									
										
										
										
											2015-10-19 09:30:15 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Let's add the stylesheet. Open `index.html` and add the following link to the `<head>`: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   我们来添加样式表。打开`index.html`,并把下列链接添加到`<head>`中: | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/index.html', 'bootstrap') | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Add powers with _*ngFor_ | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   ## 用 ***ngFor*** 添加超能力 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Our hero must choose one super power from a fixed list of Agency-approved powers. | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   We maintain that list internally (in `HeroFormComponent`). | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   我们的英雄必须从认证过的固定列表中选择一项超能力。 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   这个列表位于`HeroFormComponent`中。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   We'll add a `select` to our | 
					
						
							| 
									
										
										
										
											2016-04-30 07:01:16 -07:00
										 |  |  |  |   form and bind the options to the `powers` list using `ngFor`, | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   a technique seen previously in the [Displaying Data](./displaying-data.html) guide. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   在表单中添加`select`,用`ngFor`把`powers`列表绑定到列表选项。 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   我们在之前的[显示数据](./displaying-data.html)一章中见过`ngFor`。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Add the following HTML *immediately below* the *Alter Ego* group: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   在 *Alter Ego* 的紧下方添加如下 HTML: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.html (powers)') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   This code repeats the `<option>` tag for each power in the list of powers. | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   The `pow` template input variable is a different power in each iteration; | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   we display its name using the interpolation syntax. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   列表中的每一项超能力都会渲染成`<option>`标签。 | 
					
						
							|  |  |  |  |   模板输入变量`p`在每个迭代指向不同的超能力,使用双花括号插值表达式语法来显示它的名称。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 13:39:19 -08:00
										 |  |  |  | .l-main-section#ngModel | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Two-way data binding with _ngModel_ | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   ## 使用 *ngModel* 进行双向数据绑定 | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   Running the app right now would be disappointing. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |   如果立即运行此应用,你将会失望。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2016-04-24 22:53:59 +08:00
										 |  |  |  |   img(src="/resources/images/devguide/forms/hero-form-3.png" width="400px" alt="没有数据绑定的早期表单") | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   We don't see hero data because we are not binding to the `Hero` yet. | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   We know how to do that from earlier guides. | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   [Displaying Data](./displaying-data.html) taught us property binding. | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   [User Input](./user-input.html) showed us how to listen for DOM events with an | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   event binding and how to update a component property with the displayed value. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   因为还没有绑定到某个英雄,所以看不到任何数据。 | 
					
						
							|  |  |  |  |   解决方案见前面的章节。 | 
					
						
							|  |  |  |  |   [显示数据](./displaying-data.html)介绍了属性绑定。 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   [用户输入](./user-input.html)介绍了如何通过事件绑定来监听 DOM 事件,以及如何用显示值更新组件的属性。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   Now we need to display, listen, and extract at the same time. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   现在,需要同时进行显示、监听和提取。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We could use the techniques we already know, but | 
					
						
							|  |  |  |  |   instead we'll introduce something new: the `[(ngModel)]` syntax, which | 
					
						
							|  |  |  |  |   makes binding the form to the model super easy. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |   虽然可以在表单中再次使用这些技术。 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   但是,这里将介绍个新东西,`[(ngModel)]`语法,使表单绑定到模型的工作变得超级简单。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Find the `<input>` tag for *Name* and update it like this: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   找到 *Name* 对应的`<input>`标签,并且像这样修改它: | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.html (excerpt)', 'ngModelName-1') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | .l-sub-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |     We added a diagnostic interpolation after the input tag | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |     so we can see what we're doing. | 
					
						
							| 
									
										
										
										
											2016-04-07 17:56:35 +02:00
										 |  |  |  |     We left ourselves a note to throw it away when we're done. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     在 input 标签后添加用于诊断的插值表达式,以看清正在发生什么事。 | 
					
						
							|  |  |  |  |     给自己留个备注,提醒我们完成后移除它。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-12-11 11:50:57 -08:00
										 |  |  |  |   Focus on the binding syntax: `[(ngModel)]="..."`. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:52:43 +01:00
										 |  |  |  |   聚焦到绑定语法`[(ngModel)]="..."`上。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-03 03:49:45 +02:00
										 |  |  |  |   If we run the app right now and started typing in the *Name* input box, | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   adding and deleting characters, we'd see them appearing and disappearing | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   from the interpolated text. | 
					
						
							|  |  |  |  |   At some point it might look like this. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   如果现在运行这个应用,开始在*姓名*输入框中键入,添加和删除字符,将看到它们从插值结果中显示和消失。 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:52:43 +01:00
										 |  |  |  |   某一瞬间,它看起来可能是这样: | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2016-04-24 22:53:59 +08:00
										 |  |  |  |   img(src="/resources/images/devguide/forms/ng-model-in-action.png" width="400px" alt="操作中的ngModel") | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   The diagnostic is evidence that values really are flowing from the input box to the model and | 
					
						
							|  |  |  |  |   back again. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  | .l-sub-section | 
					
						
							|  |  |  |  |   :marked | 
					
						
							|  |  |  |  |     That's **two-way data binding**! | 
					
						
							|  |  |  |  |     For more information about `[(ngModel)]` and two-way data bindings, see | 
					
						
							|  |  |  |  |     the [Template Syntax](template-syntax.html#ngModel) page. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |     诊断信息可以证明,数据确实从输入框流动到模型,再反向流动回来。**这就是双向数据绑定!** | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2016-08-22 15:41:13 -07:00
										 |  |  |  |   Notice that we also added a `name` attribute to our `<input>` tag and set it to "name" | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  |   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. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   注意,`<input>`标签还添加了`name`属性 (attribute),并设置为 "name",表示英雄的名字。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   使用任何唯一的值都可以,但使用具有描述性的名字会更有帮助。 | 
					
						
							| 
									
										
										
										
											2016-08-12 07:57:51 +08:00
										 |  |  |  |   当在表单中使用`[(ngModel)]`时,必须要定义`name`属性。 | 
					
						
							| 
									
										
										
										
											2016-06-19 11:50:27 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | .l-sub-section | 
					
						
							|  |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |     Internally Angular creates `FormControl` instances and  | 
					
						
							|  |  |  |  |     registers them with an `NgForm` directive that Angular attached to the `<form>` tag. | 
					
						
							|  |  |  |  |     Each `FormControl` is registered under the name we assigned to the `name` attribute. | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |     We'll talk about `NgForm` [later in this guide](#ngForm). | 
					
						
							| 
									
										
										
										
											2016-06-19 11:50:27 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     在内部,Angular 创建了一些`FormControl`,并把它们注册到`NgForm`指令,再将该指令附加到`<form>`标签。 | 
					
						
							|  |  |  |  |     注册每个`FormControl`时,使用`name`属性值作为键值。[本章后面](#ngForm)会讨论`NgForm`。 | 
					
						
							| 
									
										
										
										
											2016-10-18 10:47:45 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   Let's add similar `[(ngModel)]` bindings and `name` attributes to *Alter Ego* and *Hero Power*. | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   We'll ditch the input box binding message | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   and add a new binding (at the top) to the component's `diagnostic` property. | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   Then we can confirm that two-way data binding works *for the entire hero model*. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   为*第二人格*和*超能力*属性添加类似的`[(ngModel)]`绑定和`name`属性。 | 
					
						
							|  |  |  |  |   抛弃输入框的绑定消息,在组件顶部添加到`diagnostic`属性的新绑定。 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   这样就能确认双向数据绑定*在整个 Hero 模型上*都能正常工作了。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   After revision, the core of our form should look like this: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   修改之后,这个表单的核心看起来是这样的: | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.html (excerpt)', 'ngModel-2') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | .l-sub-section | 
					
						
							|  |  |  |  |   :marked | 
					
						
							|  |  |  |  |     - Each input element has an `id` property that is used by the `label` element's `for` attribute | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     to match the label to its input control. | 
					
						
							| 
									
										
										
										
											2016-08-16 10:38:04 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |       每个 input 元素都有`id`属性,`label`元素的`for`属性用它来匹配到对应的输入控件。 | 
					
						
							| 
									
										
										
										
											2016-08-16 10:38:04 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |     - Each input element has a `name` property that is required by Angular forms to register the control with the form. | 
					
						
							| 
									
										
										
										
											2016-08-16 10:38:04 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |       每个 input 元素都有`name`属性,Angular 表单用它注册控件。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   If we run the app  now and changed every hero model property, the form might display like this: | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   如果现在运行本应用,修改 Hero 模型的每个属性,表单看起来像这样: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   img(src="/resources/images/devguide/forms/ng-model-in-action-2.png" width="400px" alt="ngModel in action") | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   The diagnostic near the top of the form | 
					
						
							|  |  |  |  |   confirms that all of our changes are reflected in the model. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   表单顶部的诊断信息反映出所做的一切更改。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-11 11:50:57 -08:00
										 |  |  |  |   **Delete** the `{{diagnostic}}` binding at the top as it has served its purpose. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   表单顶部的`{{diagnostic}}`绑定已经完成了它的使命,**删除**它。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Track control state and validity with _ngModel_ | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   ## 通过 **ngModel** 跟踪修改状态与有效性验证 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   A form isn't just about data binding. We'd also like to know the state of the controls in our form. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   表单不仅是关于数据绑定的。我们还想知道表单中各个控件的状态。 | 
					
						
							| 
									
										
										
										
											2015-10-19 09:30:15 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Using `ngModel` in a form gives us more than just a two way data binding. It also tells | 
					
						
							|  |  |  |  |   us if the user touched the control, if the value changed, or if the value became invalid. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   在表单中使用`ngModel`可以获得比仅使用双向数据绑定更多的控制权。它还会告诉我们很多信息:用户碰过此控件吗?它的值变化了吗?数据变得无效了吗? | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  |   The *NgModel* directive doesn't just track state; it updates the control with special Angular CSS classes that reflect the state. | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We can leverage those class names to change the appearance of the control. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   *NgModel* 指令不仅仅跟踪状态。它还使用特定的 Angular CSS 类来更新控件,以反映当前状态。 | 
					
						
							|  |  |  |  |   可以利用这些 CSS 类来修改控件的外观,显示或隐藏消息。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  | table | 
					
						
							|  |  |  |  |   tr | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     th | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |       p State | 
					
						
							|  |  |  |  |       p 状态 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     th | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |       p Class if true | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |       p 为真时的 CSS 类 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     th | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |       p Class if false | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |       p 为假时的 CSS 类 | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   tr | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     td | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |       p Control has been visited | 
					
						
							|  |  |  |  |       p 控件已经被访问过 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     td | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |       p <code>ng-touched</code> | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     td | 
					
						
							|  |  |  |  |       p <code>ng-untouched</code> | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   tr | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     td | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |       p Control's value has changed | 
					
						
							|  |  |  |  |       p 控件值已经变化 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     td | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |       p <code>ng-dirty</code> | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     td | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |       p <code>ng-pristine</code> | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   tr | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     td | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |       p Control's value is valid | 
					
						
							|  |  |  |  |       p 控件值是有效的 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     td | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |       p <code>ng-valid</code> | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |     td | 
					
						
							|  |  |  |  |       p <code>ng-invalid</code> | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Let's temporarily add a [template reference variable](./template-syntax.html#ref-vars) named `spy` | 
					
						
							|  |  |  |  |   to the _Name_ `<input>` tag and use it to display the input's CSS classes. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   往姓名`<input>`标签上添加名叫 **spy** 的临时[模板引用变量](./template-syntax.html#local-vars), | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   然后用这个 spy 来显示它上面的所有 CSS 类。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.html (excerpt)', 'ngModelName-2') | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Now run the app, and look at the _Name_ input box. | 
					
						
							|  |  |  |  |   Follow the next four steps *precisely*: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |   现在,运行本应用,并让*姓名*输入框获得焦点。 | 
					
						
							|  |  |  |  |   然后严格按照下面四个步骤来做: | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Look but don't touch. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      查看输入框,但别碰它。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Click inside the name box, then click outside it. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      点击输入框,然后点击输入框外面。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Add slashes to the end of the name. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      在名字的末尾添加些斜杠。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Erase the name. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      删除名字。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   The actions and effects are as follows: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-24 22:53:59 +08:00
										 |  |  |  |   动作和它对应的效果如下: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2016-04-24 22:53:59 +08:00
										 |  |  |  |   img(src="/resources/images/devguide/forms/control-state-transitions-anim.gif"  alt="控件状态转换") | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We should see the following transitions and class names: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   我们会看到下列转换及其类名: | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   img(src="/resources/images/devguide/forms/ng-control-class-changes.png" width="500px" alt="Control state transitions") | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   The `ng-valid`/`ng-invalid` pair is the most interesting to us, because we want to send a | 
					
						
							|  |  |  |  |   strong visual signal when the values are invalid. We also want to mark required fields. | 
					
						
							|  |  |  |  |   To create such visual feedback, let's add definitions for the `ng-*` CSS classes. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   (`ng-valid` | `ng-invalid`)这一对是我们最感兴趣的。当数据变得无效时,我们希望发出强力的视觉信号, | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   还想要标记出必填字段。可以通过加入自定义 CSS 来提供视觉反馈。 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   **Delete** the `#spy` template reference variable and the `TODO` as they have served their purpose. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   **删除**模板引用变量`#spy`和`TODO`,因为它们已经完成了使命。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  | .l-main-section | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Add custom CSS for visual feedback | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   ## 添加用于视觉反馈的自定义 CSS | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We can mark required fields and invalid data at the same time with a colored bar | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  |   on the left of the input box: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   可以在输入框的左侧添加带颜色的竖条,用于标记必填字段和无效输入: | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2016-04-24 22:53:59 +08:00
										 |  |  |  |   img(src="/resources/images/devguide/forms/validity-required-indicator.png" width="400px" alt="无效表单") | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We achieve this effect by adding these class definitions to a new `forms.css` file | 
					
						
							|  |  |  |  |   that we add to our project as a sibling to `index.html`: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   在新建的`forms.css`文件中,添加两个样式来实现这一效果。把这个文件添加到项目中,与`index.html`相邻。 | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  | +makeExample('src/forms.css') | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Update the `<head>` of `index.html` to include this style sheet: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   更新`index.html`中的`<head>`,以包含这个样式表: | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/index.html', 'styles') | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Show and hide validation error messages | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   ## 显示和隐藏验证错误信息 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   We can do better. The _Name_ input box is requiredand clearing it turns the bar red. That says *something* is wrong but we | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   don't know *what* is wrong or what to do about it. | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We can leverage the control's state to reveal a helpful message. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   我们能做的更好。“Name” 输入框是必填的,清空它会让左侧的条变红。这表示*某些东西*是错的,但我们不知道错在哪里,或者如何纠正。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   可以借助`ng-invalid`类来给出有用的提示。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   Here's the way it should look when the user deletes the name: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   当用户删除姓名时,看起来应该是这样的: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2016-04-24 22:53:59 +08:00
										 |  |  |  |   img(src="/resources/images/devguide/forms/name-required-error.png" width="400px" alt="必须填写姓名") | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   To achieve this effect we extend the `<input>` tag with | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   要达到这个效果,在`<input>`标签中添加: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-28 11:18:52 -07:00
										 |  |  |  |   1. a [template reference variable](./template-syntax.html#ref-vars) | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |      [模板引用变量](./template-syntax.html#ref-vars) | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   1. the "*is required*" message in a nearby `<div>` which we'll display only if the control is invalid. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |       “is required”消息,放在邻近的`<div>`元素中,只有当控件无效时,才显示它。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Here's an example of adding an error message to the _name_ input box: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   这个例子中我们把一条错误信息添加到了_name_输入框中: | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.html (excerpt)', 'name-with-error-msg') | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2016-04-28 11:18:52 -07:00
										 |  |  |  |   We need a template reference variable to access the input box's Angular control from within the template. | 
					
						
							| 
									
										
										
										
											2016-06-19 11:50:27 -04:00
										 |  |  |  |   Here we created a variable called `name` and gave it the value "ngModel". | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   模板引用变量可以访问模板中输入框的 Angular 控件。 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   这里,创建了名叫`name`的变量,并且赋值为 "ngModel"。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-10 15:48:54 -08:00
										 |  |  |  | .l-sub-section | 
					
						
							|  |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2016-08-22 15:41:13 -07:00
										 |  |  |  |     Why "ngModel"? | 
					
						
							| 
									
										
										
										
											2016-10-01 07:11:21 -07:00
										 |  |  |  |     A directive's [exportAs](../api/core/index/Directive-decorator.html) property | 
					
						
							| 
									
										
										
										
											2016-04-30 07:01:16 -07:00
										 |  |  |  |     tells Angular how to link the reference variable to the directive. | 
					
						
							| 
									
										
										
										
											2016-06-19 11:50:27 -04:00
										 |  |  |  |     We set `name` to `ngModel` because the `ngModel` directive's `exportAs` property happens to be "ngModel". | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |     为什么是 “ngModel”? | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     指令的 [exportAs](../api/core/index/DirectiveMetadata-class.html#!#exportAs) 属性告诉 Angular 如何链接模板引用变量到指令。 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |     这里把`name`设置为`ngModel`是因为`ngModel`指令的`exportAs`属性设置成了 “ngModel”。 | 
					
						
							| 
									
										
										
										
											2015-12-11 11:50:57 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   We control visibility of the name error message by binding properties of the `name` | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   control to the message `<div>` element's `hidden` property. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   我们把`div`元素的`hidden`属性绑定到`name`控件的属性,这样就可以控制“姓名”字段错误信息的可见性了。 | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.html', 'hidden-error-msg', '') | 
					
						
							| 
									
										
										
										
											2016-05-19 13:33:02 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-01 10:52:20 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  |   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. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   上例中,当控件是有效的 (valid) 或全新的 (pristine) 时,隐藏消息。 | 
					
						
							|  |  |  |  |   “全新的”意味着从它被显示在表单中开始,用户还从未修改过它的值。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-01 10:52:20 -08:00
										 |  |  |  |   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. | 
					
						
							| 
									
										
										
										
											2016-08-22 15:41:13 -07:00
										 |  |  |  |   If we arrive in this component with a new (blank) hero or an invalid hero, | 
					
						
							| 
									
										
										
										
											2016-02-01 10:52:20 -08:00
										 |  |  |  |   we'll see the error message immediately, before we've done anything. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |   这种用户体验取决于开发人员的选择。有些人会希望任何时候都显示这条消息。 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   如果忽略了`pristine`状态,就会只在值有效时隐藏此消息。 | 
					
						
							|  |  |  |  |   如果往这个组件中传入全新(空)的英雄,或者无效的英雄,将立刻看到错误信息 —— 虽然我们还啥都没做。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Some folks find that behavior disconcerting. | 
					
						
							|  |  |  |  |   They only want to see the message when the user makes an invalid change. | 
					
						
							| 
									
										
										
										
											2016-02-01 10:52:20 -08:00
										 |  |  |  |   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. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   有些人会为这种行为感到不安。它们希望只有在用户做出无效的更改时才显示这个消息。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   如果当控件是“全新”状态时也隐藏消息,就能达到这个目的。 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   在往表单中[添加新英雄](#new-hero)时,将看到这种选择的重要性。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   The hero *Alter Ego* is optional so we can leave that be. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   英雄的*第二人格*是可选项,所以不用改它。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   Hero *Power* selection is required. | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We can add the same kind of error handling to the `<select>` if we  want, | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   but it's not imperative because the selection box already constrains the | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   power to valid values. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |   英雄的*超能力*选项是必填的。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   只要愿意,可以往`<select>`上添加相同的错误处理。 | 
					
						
							|  |  |  |  |   但没有必要,这个选择框已经限制了“超能力”只能选有效值。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   We'd like to add a new hero in this form. | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   We place a "New Hero" button at the bottom of the form and bind its click event to a `newHero` component method. | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   我们希望在这个表单中添加新的英雄。 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   在表单的底部放置“New Hero(新增英雄)”按钮,并把它的点击事件绑定到`newHero`组件。 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExample('forms/ts/src/app/hero-form.component.html', | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  | 'new-hero-button-no-reset', | 
					
						
							|  |  |  |  | 'src/app/hero-form.component.html (New Hero button)') | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExample('forms/ts/src/app/hero-form.component.ts', | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  | 'new-hero', | 
					
						
							|  |  |  |  | 'src/app/hero-form.component.ts (New Hero method)')(format=".") | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-01 10:52:20 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   Run the application again, click the *New Hero* button, and the form clears. | 
					
						
							|  |  |  |  |   The *required* bars to the left of the input box are red, indicating invalid `name` and `power` properties. | 
					
						
							| 
									
										
										
										
											2016-08-22 15:41:13 -07:00
										 |  |  |  |   That's understandable as these are required fields. | 
					
						
							| 
									
										
										
										
											2016-02-01 10:52:20 -08:00
										 |  |  |  |   The error messages are hidden because the form is pristine; we haven't changed anything yet. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   再次运行应用,点击 *New Hero* 按钮,表单被清空了。 | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |   输入框左侧的*必填项*竖条是红色的,表示`name`和`power`属性是无效的。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   这可以理解,因为有一些必填字段。 | 
					
						
							|  |  |  |  |   错误信息是隐藏的,因为表单还是全新的,还没有修改任何东西。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-01 10:52:20 -08:00
										 |  |  |  |   Enter a name and click *New Hero* again. | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   The app displays a **_Name is required_** error message! | 
					
						
							|  |  |  |  |   We don't want error messages when we create a new (empty) hero. | 
					
						
							|  |  |  |  |   Why are we getting one now? | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   输入名字,再次点击 *New Hero* 按钮。 | 
					
						
							|  |  |  |  |   这次,出现了错误信息!为什么?我们不希望显示新(空)的英雄时,出现错误信息。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   Inspecting the element in the browser tools reveals that the *name* input box is _no longer pristine_. | 
					
						
							|  |  |  |  |   The form remembers that we entered a name before clicking *New Hero*. | 
					
						
							|  |  |  |  |   Replacing the hero *did not restore the pristine state* of the form controls. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   使用浏览器工具审查这个元素就会发现,这个 *name* 输入框并不是全新的。 | 
					
						
							|  |  |  |  |   表单记得我们在点击 *New Hero* 前输入的名字。 | 
					
						
							| 
									
										
										
										
											2016-05-16 18:57:25 +08:00
										 |  |  |  |   更换了英雄*并不会重置控件的“全新”状态*。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   We have to clear all of the flags imperatively which we can do | 
					
						
							|  |  |  |  |   by calling the form's `reset()` method after calling the `newHero()` method. | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   我们必须清除所有标记,在调用`newHero()`方法后调用表单的`reset()`方法即可。 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExample('forms/ts/src/app/hero-form.component.html', | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |   'new-hero-button-form-reset', | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  |   'src/app/hero-form.component.html (Reset the form)') | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2016-11-21 18:23:28 -08:00
										 |  |  |  |   Now clicking "New Hero" both resets the form and its control flags. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   现在点击“New Hero”重设表单和它的控制标记。 | 
					
						
							| 
									
										
										
										
											2016-08-22 15:41:13 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   ## Submit the form with _ngSubmit_ | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   ## 使用 *ngSubmit* 提交该表单 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   The user should be able to submit this form after filling it in. | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   The Submit button at the bottom of the form | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   does nothing on its own, but it will | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   trigger a form submit because of its type (`type="submit"`). | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-28 22:23:34 +01:00
										 |  |  |  |   在填表完成之后,用户还应该能提交这个表单。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   “Submit(提交)”按钮位于表单的底部,它自己不做任何事,但因为有特殊的 type 值 (`type="submit"`),所以会触发表单提交。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-11 11:50:57 -08:00
										 |  |  |  |   A "form submit" is useless at the moment. | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   To make it useful, bind the form's `ngSubmit` event property | 
					
						
							|  |  |  |  |   to the hero form component's `onSubmit()` method: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   现在这样仅仅触发“表单提交”是没用的。 | 
					
						
							|  |  |  |  |   要让它有用,就要把该表单的`ngSubmit`事件属性绑定到英雄表单组件的`onSubmit()`方法上: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('forms/ts/src/app/hero-form.component.html (ngSubmit)') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   We slipped in something extra there at the end!  We defined a | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   template reference variable, **`#heroForm`**, and initialized it with the value "ngForm". | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   上面代码的最后出现一些额外的东西!定义了模板引用变量**`#heroForm`**,并初始化为 "ngForm"。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-10 15:48:54 -08:00
										 |  |  |  |   The variable `heroForm` is now a reference to the `NgForm` directive that governs the form as a whole. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   现在`heroForm`变量引用的是`NgForm`指令,它代表的是表单的整体。 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  | .l-sub-section#ngForm | 
					
						
							| 
									
										
										
										
											2016-03-10 15:48:54 -08:00
										 |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2016-11-26 21:43:16 +00:00
										 |  |  |  |     ### The _NgForm_ directive | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |     ### NgForm指令     | 
					
						
							|  |  |  |  |      | 
					
						
							|  |  |  |  |     What `NgForm` directive?  | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |     We didn't add an [NgForm](../api/forms/index/NgForm-directive.html) directive! | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |     什么`NgForm`指令?之前没有添加过 [NgForm](../api/common/index/NgForm-directive.html) 指令啊! | 
					
						
							| 
									
										
										
										
											2016-08-22 15:41:13 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-10 15:48:54 -08:00
										 |  |  |  |     Angular did. Angular creates and attaches an `NgForm` directive to the `<form>` tag automatically. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     是 Angular 干的。Angular 自动创建了`NgForm`指令,并把它附加到`<form>`标签。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  |     The `NgForm` directive supplements the `form` element with additional features. | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |     It holds the controls we created for the elements with an `ngModel` directive  | 
					
						
							|  |  |  |  |     and `name` attribute, and monitors their properties including their validity. | 
					
						
							| 
									
										
										
										
											2016-03-10 15:48:54 -08:00
										 |  |  |  |     It also has its own `valid` property which is true only *if every contained | 
					
						
							|  |  |  |  |     control* is valid. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     `NgForm`指令为`form`元素扩充了额外的特性。 | 
					
						
							|  |  |  |  |     它持有通过`ngModel`指令和`name`属性为各个元素创建的那些控件,并且监视它们的属性变化,包括有效性。 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |     它还有自己的`valid`属性,只有当*其中所有控件*都有效时,它才有效。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-10 15:48:54 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   We'll bind the form's overall validity via | 
					
						
							|  |  |  |  |   the `heroForm` variable to the button's `disabled` property | 
					
						
							|  |  |  |  |   using an event binding. Here's the code: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   我们要把表单的总体有效性通过`heroForm`变量绑定到此按钮的`disabled`属性上,代码如下: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.html', 'submit-button', '') | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   If we run the application now, we find that the button is enabled | 
					
						
							|  |  |  |  |   — although it doesn't do anything useful yet. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-03 20:21:11 +08:00
										 |  |  |  |   重新运行应用。表单打开时,状态是有效的,按钮是可用的。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Now if we delete the Name, we violate the "required" rule, which | 
					
						
							|  |  |  |  |   is duly noted in the error message. | 
					
						
							|  |  |  |  |   The Submit button is also disabled. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   现在,如果我们删除*姓名*,就会违反“必填姓名”规则,就会像以前那样显示出错误信息。同时,Submit 按钮也被禁用了。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   Not impressed?  Think about it for a moment. What would we have to do to | 
					
						
							|  |  |  |  |   wire the button's enable/disabled state to the form's validity without Angular's help? | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   没感动吗?再想一会儿。如果没有 Angular `NgForm`的帮助,又该怎么让按钮的禁用/启用状态和表单的有效性关联起来呢? | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   For us, it was as simple as: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   有了 Angular,它就是这么简单: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   1. Define a template reference variable on the (enhanced) form element. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |      定义模板引用变量,放在(强化过的)form 元素上 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   2. Refer to that variable in a button many lines away. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |      从很多行之外的按钮上引用这个变量。 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   ## Toggle two form regions (extra credit) | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   ## 切换两个表单区域(额外的奖励) | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   Submitting the form isn't terribly dramatic at the moment. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   提交表单还是不够激动人心。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | .l-sub-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |     An unsurprising observation for a demo. To be honest, | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |     jazzing it up won't teach us anything new about forms. | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |     But this is an opportunity to exercise some of our newly won | 
					
						
							|  |  |  |  |     binding skills. | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |     If you aren't interested, go ahead and skip to this guide's conclusion. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |     对演示来说,这个收场很平淡的。老实说,即使让它更出彩,也无法教给我们任何关于表单的新知识。 | 
					
						
							|  |  |  |  |     但这是练习新学到的绑定技能的好机会。 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |     如果你不感兴趣,可以跳到本章的总结部分。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   Let's do something more strikingly visual. | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   Let's hide the data entry area and display something else. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   来实现一些更炫的视觉效果吧。 | 
					
						
							|  |  |  |  |   隐藏掉数据输入框,显示一些其它东西。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   Start by wrapping the form in a `<div>` and bind | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   its `hidden` property to the `HeroFormComponent.submitted` property. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   先把表单包裹进`<div>`中,再把它的`hidden`属性绑定到`HeroFormComponent.submitted`属性。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.html (excerpt)', 'edit-div') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   The main form is visible from the start because the | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   `submitted` property is false until we submit the form, | 
					
						
							|  |  |  |  |   as this fragment from the `HeroFormComponent` shows: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 01:09:14 +08:00
										 |  |  |  |   主表单从一开始就是可见的,因为`submitted`属性是 false,直到提交了这个表单。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   来自`HeroFormComponent`的代码片段告诉了我们这一点: | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.ts', 'submitted') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-11-30 20:41:09 -08:00
										 |  |  |  |   When we click the Submit button, the `submitted` flag becomes true and the form disappears | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   as planned. | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   当点击 Submit 按钮时,`submitted`标志会变成 true,并且表单像预想中一样消失了。 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Now the app needs to show something else while the form is in the submitted state. | 
					
						
							|  |  |  |  |   Add the following HTML below the `<div>` wrapper we just wrote: | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   现在,当表单处于已提交状态时,需要显示一些别的东西。 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |   在刚刚写的`<div>`包装下方,添加下列 HTML 语句: | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  | +makeExcerpt('src/app/hero-form.component.html (excerpt)', 'submitted') | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   There's our hero again, displayed read-only with interpolation bindings. | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   This `<div>` appears only while the component is in the submitted state. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 14:29:09 +08:00
										 |  |  |  |   英雄又出现了,它通过插值表达式绑定显示为只读内容。 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   这一小段 HTML 只在组件处于已提交状态时才会显示。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   The HTML includes an _Edit_ button whose click event is bound to an expression | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  |   that clears the `submitted` flag. | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   这段HTML包含一个 “Edit(编辑)”按钮,将 click 事件绑定到表达式,用于清除`submitted`标志。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   When we click the _Edit_ button, this block disappears and the editable form reappears. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   当点它时,这个只读块消失了,可编辑的表单重新出现了。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   That's as much drama as we can muster for now. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   够炫吗?好吧,我们已经尽力了! | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   ## Conclusion | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-03 20:21:11 +08:00
										 |  |  |  |   ## 结论 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   The Angular form discussed in this guide takes advantage of the following  | 
					
						
							|  |  |  |  |   framework features to provide support for data modification, validation, and more: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   本章讨论的 Angular 表单技术利用了下列框架特性来支持数据修改、验证和更多操作: | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-20 16:10:44 -07:00
										 |  |  |  |   - An Angular HTML form template. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     Angular HTML 表单模板。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   - A form component class with a `@Component` decorator. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |     带有`@Component`装饰器的表单组件类。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-31 10:33:30 -08:00
										 |  |  |  |   - Handling form submission by binding to the `NgForm.ngSubmit` event property. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |     通过绑定到`NgForm.ngSubmit`事件属性来处理表单提交。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   - Template reference variables such as `#heroForm` and `#name`. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |     模板引用变量,例如`#heroForm`和`#name`。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   - `[(ngModel)]` syntax for two-way data binding. | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |     `[(ngModel)]`语法用来实现双向数据绑定。 | 
					
						
							|  |  |  |  |      | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   - The use of `name` attributes for validation and form element change tracking. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-26 15:04:21 +08:00
										 |  |  |  |     `name`属性的用途是有效性验证和对表单元素的变更进行追踪。 | 
					
						
							|  |  |  |  |      | 
					
						
							| 
									
										
										
										
											2016-04-30 07:01:16 -07:00
										 |  |  |  |   - The reference variable’s `valid` property on input controls to check if a control is valid and show/hide error messages. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     指向 input 控件的引用变量上的`valid`属性,可用于检查控件是否有效、是否显示/隐藏错误信息。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-11 11:50:57 -08:00
										 |  |  |  |   - Controlling the submit button's enabled state by binding to `NgForm` validity. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     通过绑定到`NgForm`的有效性状态,控制提交按钮的禁用状态。 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-11 11:50:57 -08:00
										 |  |  |  |   - Custom CSS classes that provide visual feedback to users about invalid controls. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |     定制 CSS 类来给用户提供无效控件的视觉反馈。 | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-11 11:50:57 -08:00
										 |  |  |  |   Our final project folder structure should look like this: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-26 03:16:06 +08:00
										 |  |  |  |   最终的项目目录结构看起来是这样: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-28 16:15:26 -08:00
										 |  |  |  | .filetree | 
					
						
							| 
									
										
										
										
											2016-09-20 05:24:40 +02:00
										 |  |  |  |   .file angular-forms | 
					
						
							| 
									
										
										
										
											2016-01-28 16:15:26 -08:00
										 |  |  |  |   .children | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  |     .file src | 
					
						
							| 
									
										
										
										
											2016-01-28 16:15:26 -08:00
										 |  |  |  |     .children | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  |       .file app | 
					
						
							|  |  |  |  |       .children | 
					
						
							|  |  |  |  |         .file app.component.ts | 
					
						
							|  |  |  |  |         .file app.module.ts | 
					
						
							|  |  |  |  |         .file hero.ts | 
					
						
							|  |  |  |  |         .file hero-form.component.html | 
					
						
							|  |  |  |  |         .file hero-form.component.ts | 
					
						
							| 
									
										
										
										
											2016-01-28 16:15:26 -08:00
										 |  |  |  |       .file main.ts | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  |       .file tsconfig.json | 
					
						
							|  |  |  |  |       .file index.html | 
					
						
							| 
									
										
										
										
											2017-01-31 20:36:32 -05:00
										 |  |  |  |     .file node_modules ... | 
					
						
							| 
									
										
										
										
											2016-01-28 16:15:26 -08:00
										 |  |  |  |     .file package.json | 
					
						
							| 
									
										
										
										
											2015-12-11 11:50:57 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-01-26 14:15:35 -08:00
										 |  |  |  |   Here’s the code for the final version of the application: | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-03 20:21:11 +08:00
										 |  |  |  |   这里是源码的最终版本: | 
					
						
							| 
									
										
										
										
											2015-11-10 18:31:46 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  | +makeTabs( | 
					
						
							| 
									
										
										
										
											2017-02-02 18:38:17 +00:00
										 |  |  |  |   `forms/ts/src/app/hero-form.component.ts, | 
					
						
							|  |  |  |  |    forms/ts/src/app/hero-form.component.html, | 
					
						
							|  |  |  |  |    forms/ts/src/app/hero.ts, | 
					
						
							|  |  |  |  |    forms/ts/src/app/app.module.ts, | 
					
						
							|  |  |  |  |    forms/ts/src/app/app.component.ts, | 
					
						
							|  |  |  |  |    forms/ts/src/main.ts, | 
					
						
							|  |  |  |  |    forms/ts/src/index.html, | 
					
						
							|  |  |  |  |    forms/ts/src/forms.css`, | 
					
						
							| 
									
										
										
										
											2015-12-11 11:50:57 -08:00
										 |  |  |  |   'final, final,,,,,', | 
					
						
							|  |  |  |  |   `hero-form.component.ts, | 
					
						
							| 
									
										
										
										
											2016-08-22 15:41:13 -07:00
										 |  |  |  |    hero-form.component.html, | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  |    hero.ts, | 
					
						
							| 
									
										
										
										
											2016-08-09 17:38:25 +01:00
										 |  |  |  |    app.module.ts, | 
					
						
							| 
									
										
										
										
											2015-12-11 11:50:57 -08:00
										 |  |  |  |    app.component.ts, | 
					
						
							| 
									
										
										
										
											2016-01-28 16:15:26 -08:00
										 |  |  |  |    main.ts, | 
					
						
							| 
									
										
										
										
											2015-11-07 20:54:31 -08:00
										 |  |  |  |    index.html, | 
					
						
							| 
									
										
										
										
											2016-02-27 13:48:24 -08:00
										 |  |  |  |    forms.css`) |