include ../_util-fns :marked ## Setup to develop locally ## 为本地开发搭建环境 Real application development takes place in a local development environment like your machine. 我们通常在本地开发环境中(例如你的机器)进行真实的应用程序开发。 Follow the [setup](../guide/setup.html) instructions for creating a new project named after which the file structure should look like this: 根据[开发环境](../guide/setup.html)中的说明创建一个名为的新项目, 该项目的文件结构应该是这样的: .filetree .file angular-tour-of-heroes .children .file src .children .file app .children .file app.component.ts .file app.module.ts .file main.ts .file index.html .file styles.css .file systemjs.config.js .file tsconfig.json .file node_modules ... .file package.json :marked When we're done with this first episode, the app runs like this . 在我们完成本章时,得到的应用和这个在线例子一样。 ## Keep the app transpiling and running ## 保持应用不断转译和运行 We want to start the TypeScript compiler, have it watch for changes, and start our server. Do this by entering the following command in the terminal window. 开发过程中需要启动 TypeScript 编译器,让它监视文件变更,并且启动开发服务器。 在命令行窗口中输入以下命令: code-example(language="sh" class="code-shell"). npm start :marked This command runs the compiler in watch mode, starts the server, launches the app in a browser, and keeps the app running while we continue to build the Tour of Heroes. 这个命令会在监视模式下运行编译器,启动开发服务器,在浏览器中启动我们的应用, 并在后续构建《英雄指南》过程中,应用能持续运行。 .l-main-section :marked ## Show our Hero ## 显示我们的英雄 We want to display Hero data in our app 我们要在应用中显示英雄数据。 Update the `AppComponent` so it has two properties:   a `title` property for the application name and a `hero` property for a hero named "Windstorm". 更新`AppComponent`并添加两个属性:`title`属性表示应用的名字,`hero`属性表示一个名叫 “Windstorm” 的英雄。 +makeExample('toh-1/ts-snippets/app.component.snippets.pt1.ts', 'app-component-1', 'src/app/app.component.ts (AppComponent class)')(format=".") :marked Now update the template in the `@Component` decoration with data bindings to these new properties. 下面,更新`@Component`装饰器中指定的模板,为这些新属性建立数据绑定。 +makeExample('toh-1/ts-snippets/app.component.snippets.pt1.ts', 'show-hero') :marked The browser should refresh and display our title and hero. 保存后,浏览器应自动刷新,显示标题和英雄。 The double curly braces tell our app to read the `title` and `hero` properties from the component and render them. This is the "interpolation" form of one-way data binding. 这里的双大括号会告诉应用:从组件中读取`title`和`hero`属性,并且渲染它们。 这就是单向数据绑定的“插值表达式”形式。 .l-sub-section :marked Learn more about interpolation in the [Displaying Data chapter](../guide/displaying-data.html). 要了解插值表达式的更多知识,见[显示数据](../guide/displaying-data.html)。 :marked ### Hero object ### Hero 对象 At the moment, our hero is just a name. Our hero needs more properties. Let's convert the `hero` from a literal string to a class. 此时此刻,我们的英雄只是一个名字。显然,它/她应该有更多属性。 让我们把`hero`从一个字符串字面量换成一个类。 Create a `Hero` class with `id` and `name` properties. For now put this near the top of the `app.component.ts` file, just below the import statement. 创建一个`Hero`类,它具有`id`和`name`属性。 现在,把下列代码放在`app.component.ts`的顶部,仅次于 import 语句。 +makeExample('toh-1/ts/src/app/app.component.ts', 'hero-class-1', 'src/app/app.component.ts (Hero class)')(format=".") :marked Now that we have a `Hero` class, let’s refactor our component’s `hero` property to be of type `Hero`. Then initialize it with an id of `1` and the name, "Windstorm". 现在,有了一个`Hero`类,我们把组件`hero`属性的类型换成`Hero`。 然后以`1`为 id、以 “Windstorm” 为名字,初始化它。 +makeExample('toh-1/ts/src/app/app.component.ts', 'hero-property-1', 'src/app/app.component.ts (hero property)')(format=".") :marked Because we changed the hero from a string to an object, we update the binding in the template to refer to the hero’s `name` property. 我们把`hero`从一个字符串换成了对象,所以也得更新模板中的绑定表达式,来引用`hero`的`name`属性。 +makeExample('toh-1/ts-snippets/app.component.snippets.pt1.ts', 'show-hero-2') :marked The browser refreshes and continues to display our hero’s name. 浏览器自动刷新,并继续显示这位英雄的名字。 ### Adding more HTML ### 添加更多的 HTML Displaying a name is good, but we want to see all of our hero’s properties. We’ll add a `
` for our hero’s `id` property and another `
` for our hero’s `name`. 能显示名字虽然不错,但我们还想看到这位英雄的所有属性。 我们将添加一个`
`来显示英雄的`id`属性,用另一个`
`来显示英雄的`name`属性。 +makeExample('toh-1/ts-snippets/app.component.snippets.pt1.ts', 'show-hero-properties') :marked Uh oh, our template string is getting long. We better take care of that to avoid the risk of making a typo in the template. 啊哦!我们的模板字符串已经太长了。我们最好小心点,免得在模板中出现拼写错误。 ### Multi-line template strings ### 多行模板字符串 We could make a more readable template with string concatenation but that gets ugly fast, it is harder to read, and it is easy to make a spelling error. Instead, let’s take advantage of the template strings feature in ES2015 and TypeScript to maintain our sanity. 我们可以通过字符串加法来制作更可读的模板,但这样仍然太难看了,难于阅读,容易拼错。 这样不行!我们要借助 ES2015 和 TypeScript 提供的模板字符串来保持清爽。 Change the quotes around the template to back-ticks and put the `

`, `

` and `
` elements on their own lines. 把模板的双引号改成反引号,并且让`

`,`

`和`
`标签各占一行。 +makeExample('toh-1/ts-snippets/app.component.snippets.pt1.ts', 'multi-line-strings', 'src/app/app.component.ts (AppComponent\'s template)') .callout.is-important header A back-tick is not a single quote header 反引号不是单引号 :marked **Be careful!** A back-tick (`) looks a lot like a single quote ('). It's actually a completely different character. Back-ticks can do more than demarcate a string. Here we use them in a limited way to spread the template over multiple lines. Everything between the back-ticks at the beginning and end of the template is part of a single template string. **小心!**反引号 (`) 虽然看起来很像单引号 ('),但它们是截然不同的字符。 反引号能做的可不仅仅是标记字符串的边界。 在这里,我们只用它来把我们的模板变成多行的,而没有涉及更多用途。 所有被反引号引起来的部分,都是一个单一模板字符串的一部分。 .l-main-section :marked ## Editing Our Hero ## 编辑我们的英雄 We want to be able to edit the hero name in a textbox. 我们想在一个文本框中编辑英雄的名字。 Refactor the hero name `