angular-cn/aio/content/guide/dynamic-component-loader.md

274 lines
11 KiB
Markdown
Raw Normal View History

# Dynamic Component Loader
2017-03-31 19:57:13 -04:00
Merge remote-tracking branch 'en/master' into aio # Conflicts: # aio/content/guide/ajs-quick-reference.md # aio/content/guide/animations.md # aio/content/guide/aot-compiler.md # aio/content/guide/architecture.md # aio/content/guide/attribute-directives.md # aio/content/guide/bootstrapping.md # aio/content/guide/browser-support.md # aio/content/guide/cb-index.md # aio/content/guide/change-log.md # aio/content/guide/cheatsheet.md # aio/content/guide/cli-quickstart.md # aio/content/guide/component-interaction.md # aio/content/guide/component-styles.md # aio/content/guide/dependency-injection-in-action.md # aio/content/guide/dependency-injection.md # aio/content/guide/deployment.md # aio/content/guide/displaying-data.md # aio/content/guide/dynamic-component-loader.md # aio/content/guide/dynamic-form.md # aio/content/guide/form-validation.md # aio/content/guide/forms.md # aio/content/guide/glossary.md # aio/content/guide/hierarchical-dependency-injection.md # aio/content/guide/i18n.md # aio/content/guide/index.md # aio/content/guide/learning-angular.md # aio/content/guide/lifecycle-hooks.md # aio/content/guide/ngmodule-faq.md # aio/content/guide/ngmodule.md # aio/content/guide/npm-packages.md # aio/content/guide/pipes.md # aio/content/guide/quickstart.md # aio/content/guide/reactive-forms.md # aio/content/guide/router.md # aio/content/guide/security.md # aio/content/guide/server-communication.md # aio/content/guide/set-document-title.md # aio/content/guide/setup-systemjs-anatomy.md # aio/content/guide/setup.md # aio/content/guide/structural-directives.md # aio/content/guide/styleguide.md # aio/content/guide/template-syntax.md # aio/content/guide/testing.md # aio/content/guide/ts-to-js.md # aio/content/guide/typescript-configuration.md # aio/content/guide/upgrade.md # aio/content/guide/user-input.md # aio/content/guide/visual-studio-2015.md # aio/content/guide/webpack.md # aio/content/navigation.json # aio/content/tutorial/index.md # aio/content/tutorial/toh-pt1.md # aio/content/tutorial/toh-pt2.md # aio/content/tutorial/toh-pt3.md # aio/content/tutorial/toh-pt4.md # aio/content/tutorial/toh-pt5.md # aio/content/tutorial/toh-pt6.md # aio/package.json # aio/src/styles/main.scss # aio/transforms/angular.io-package/index.js
2017-07-29 12:03:22 -04:00
# 动态组件加载器
2017-03-31 19:57:13 -04:00
<div class="alert is-warning">
译注:本页讲的是一个用于显示广告的范例,而部分广告拦截器插件,比如 Chrome 的 AdGuard可能会破坏其工作逻辑因此请在本页关闭那些插件。
</div>
Component templates are not always fixed. An application may need to load new components at runtime.
组件的模板不会永远是固定的。应用可能会需要在运行期间加载一些新的组件。
2017-03-31 19:57:13 -04:00
This cookbook shows you how to use `ComponentFactoryResolver` to add components dynamically.
这本烹饪书为你展示如何使用 `ComponentFactoryResolver` 来动态添加组件。
See the <live-example name="dynamic-component-loader"></live-example>
2017-03-31 19:57:13 -04:00
of the code in this cookbook.
2018-03-03 08:06:01 -05:00
<live-example name="dynamic-component-loader"></live-example>查看本烹饪书的源码。
2017-03-31 19:57:13 -04:00
{@a dynamic-loading}
2017-03-31 19:57:13 -04:00
## Dynamic component loading
## 动态组件加载
2017-03-31 19:57:13 -04:00
The following example shows how to build a dynamic ad banner.
下面的例子展示了如何构建动态广告条。
2017-03-31 19:57:13 -04:00
The hero agency is planning an ad campaign with several different
ads cycling through the banner. New ad components are added
frequently by several different teams. This makes it impractical
to use a template with a static component structure.
英雄管理局正在计划一个广告活动,要在广告条中显示一系列不同的广告。几个不同的小组可能会频繁加入新的广告组件。
再用只支持静态组件结构的模板显然是不现实的。
2017-03-31 19:57:13 -04:00
Instead, you need a way to load a new component without a fixed
reference to the component in the ad banner's template.
你需要一种新的组件加载方式,它不需要在广告条组件的模板中引用固定的组件。
2017-03-31 19:57:13 -04:00
Angular comes with its own API for loading components dynamically.
Angular 自带的 API 就能支持动态加载组件。
2017-03-31 19:57:13 -04:00
{@a directive}
## The anchor directive
## 指令
2017-03-31 19:57:13 -04:00
Before you can add components you have to define an anchor point
to tell Angular where to insert components.
在添加组件之前,先要定义一个锚点来告诉 Angular 要把组件插入到什么地方。
2017-03-31 19:57:13 -04:00
The ad banner uses a helper directive called `AdDirective` to
mark valid insertion points in the template.
广告条使用一个名叫 `AdDirective` 的辅助指令来在模板中标记出有效的插入点。
<code-example path="dynamic-component-loader/src/app/ad.directive.ts" title="src/app/ad.directive.ts" linenums="false">
</code-example>
2017-03-31 19:57:13 -04:00
`AdDirective` injects `ViewContainerRef` to gain access to the view
container of the element that will host the dynamically added component.
`AdDirective` 注入了 `ViewContainerRef` 来获取对容器视图的访问权,这个容器就是那些动态加入的组件的宿主。
2017-03-31 19:57:13 -04:00
In the `@Directive` decorator, notice the selector name, `ad-host`;
that's what you use to apply the directive to the element.
The next section shows you how.
`@Directive` 装饰器中,要注意选择器的名称:`ad-host`,它就是你将应用到元素上的指令。下一节会展示该如何做。
2017-03-31 19:57:13 -04:00
{@a loading-components}
2017-03-31 19:57:13 -04:00
## Loading components
## 加载组件
2017-03-31 19:57:13 -04:00
Most of the ad banner implementation is in `ad-banner.component.ts`.
To keep things simple in this example, the HTML is in the `@Component`
decorator's `template` property as a template string.
广告条的大部分实现代码都在 `ad-banner.component.ts` 中。
为了让这个例子简单点HTML 被直接放在了 `@Component` 装饰器的 `template` 属性中。
2017-03-31 19:57:13 -04:00
The `<ng-template>` element is where you apply the directive you just made.
To apply the `AdDirective`, recall the selector from `ad.directive.ts`,
`ad-host`. Apply that to `<ng-template>` without the square brackets. Now Angular knows
where to dynamically load components.
`<ng-template>` 元素就是刚才制作的指令将应用到的地方。
要应用 `AdDirective`,回忆一下来自 `ad.directive.ts` 的选择器 `ad-host`。把它应用到 `<ng-template>`(不用带方括号)。
这下Angular 就知道该把组件动态加载到哪里了。
<code-example path="dynamic-component-loader/src/app/ad-banner.component.ts" region="ad-host" title="src/app/ad-banner.component.ts (template)" linenums="false">
2017-03-31 19:57:13 -04:00
</code-example>
2017-03-31 19:57:13 -04:00
The `<ng-template>` element is a good choice for dynamic components
because it doesn't render any additional output.
`<ng-template>` 元素是动态加载组件的最佳选择,因为它不会渲染任何额外的输出。
2017-03-31 19:57:13 -04:00
{@a resolving-components}
## Resolving components
2017-08-06 01:21:34 -04:00
## 解析组件
2017-03-31 19:57:13 -04:00
Take a closer look at the methods in `ad-banner.component.ts`.
深入看看 `ad-banner.component.ts` 中的方法。
2017-03-31 19:57:13 -04:00
`AdBannerComponent` takes an array of `AdItem` objects as input,
which ultimately comes from `AdService`. `AdItem` objects specify
the type of component to load and any data to bind to the
2017-03-31 19:57:13 -04:00
component.`AdService` returns the actual ads making up the ad campaign.
`AdBannerComponent` 接收一个 `AdItem` 对象的数组作为输入,它最终来自 `AdService`
`AdItem` 对象指定要加载的组件类,以及绑定到该组件上的任意数据。
`AdService` 可以返回广告活动中的那些广告。
2017-03-31 19:57:13 -04:00
Passing an array of components to `AdBannerComponent` allows for a
dynamic list of ads without static elements in the template.
`AdBannerComponent` 传入一个组件数组可以在模板中放入一个广告的动态列表,而不用写死在模板中。
2017-03-31 19:57:13 -04:00
With its `getAds()` method, `AdBannerComponent` cycles through the array of `AdItems`
and loads a new component every 3 seconds by calling `loadComponent()`.
通过 `getAds()` 方法,`AdBannerComponent` 可以循环遍历 `AdItems` 的数组,并且每三秒调用一次 `loadComponent()` 来加载新组件。
<code-example path="dynamic-component-loader/src/app/ad-banner.component.ts" region="class" title="src/app/ad-banner.component.ts (excerpt)" linenums="false">
</code-example>
2017-03-31 19:57:13 -04:00
The `loadComponent()` method is doing a lot of the heavy lifting here.
Take it step by step. First, it picks an ad.
这里的 `loadComponent()` 方法很重要。
来一步步看看。首先,它选取了一个广告。
<div class="alert is-helpful">
2017-03-31 19:57:13 -04:00
**How _loadComponent()_ chooses an ad**
**`loadComponent()` 如何选择广告**
2017-03-31 19:57:13 -04:00
The `loadComponent()` method chooses an ad using some math.
`loadComponent()` 方法使用某种算法选择了一个广告。
2018-03-24 01:12:42 -04:00
First, it sets the `currentAdIndex` by taking whatever it
2017-03-31 19:57:13 -04:00
currently is plus one, dividing that by the length of the `AdItem` array, and
2018-03-24 01:12:42 -04:00
using the _remainder_ as the new `currentAdIndex` value. Then, it uses that
2017-03-31 19:57:13 -04:00
value to select an `adItem` from the array.
2018-03-24 01:12:42 -04:00
(译注:循环选取算法)首先,它把 `currentAdIndex` 递增一,然后用它除以 `AdItem` 数组长度的*余数*作为新的 `currentAdIndex` 的值,
最后用这个值来从数组中选取一个 `adItem`
2017-04-10 11:51:13 -04:00
</div>
After `loadComponent()` selects an ad, it uses `ComponentFactoryResolver`
to resolve a `ComponentFactory` for each specific component.
2017-03-31 19:57:13 -04:00
The `ComponentFactory` then creates an instance of each component.
`loadComponent()` 选取了一个广告之后,它使用 `ComponentFactoryResolver` 来为每个具体的组件解析出一个 `ComponentFactory`
然后 `ComponentFactory` 会为每一个组件创建一个实例。
Next, you're targeting the `viewContainerRef` that
exists on this specific instance of the component. How do you know it's
this specific instance? Because it's referring to `adHost` and `adHost` is the
2017-03-31 19:57:13 -04:00
directive you set up earlier to tell Angular where to insert dynamic components.
2018-03-22 21:43:02 -04:00
接下来,你要把 `viewContainerRef` 指向这个组件的现有实例。但你怎么才能找到这个实例呢?
很简单,因为它指向了 `adHost`,而这个 `adHost` 就是你以前设置过的指令,用来告诉 Angular 该把动态组件插入到什么位置。
As you may recall, `AdDirective` injects `ViewContainerRef` into its constructor.
This is how the directive accesses the element that you want to use to host the dynamic component.
2017-03-31 19:57:13 -04:00
回忆一下,`AdDirective` 曾在它的构造函数中注入了一个 `ViewContainerRef`
因此这个指令可以访问到这个你打算用作动态组件宿主的元素。
2017-03-31 19:57:13 -04:00
To add the component to the template, you call `createComponent()` on `ViewContainerRef`.
要把这个组件添加到模板中,你可以调用 `ViewContainerRef``createComponent()`
2017-03-31 19:57:13 -04:00
The `createComponent()` method returns a reference to the loaded component.
Use that reference to interact with the component by assigning to its properties or calling its methods.
`createComponent()` 方法返回一个引用,指向这个刚刚加载的组件。
使用这个引用就可以与该组件进行交互,比如设置它的属性或调用它的方法。
2017-03-31 19:57:13 -04:00
{@a selector-references}
#### Selector references
#### 对选择器的引用
2017-03-31 19:57:13 -04:00
Generally, the Angular compiler generates a `ComponentFactory`
for any component referenced in a template. However, there are
no selector references in the templates for
dynamically loaded components since they load at runtime.
通常Angular 编译器会为模板中所引用的每个组件都生成一个 `ComponentFactory` 类。
但是,对于动态加载的组件,模板中不会出现对它们的选择器的引用。
2017-03-31 19:57:13 -04:00
To ensure that the compiler still generates a factory,
add dynamically loaded components to the `NgModule`'s `entryComponents` array:
要想确保编译器照常生成工厂类,就要把这些动态加载的组件添加到 `NgModule``entryComponents` 数组中:
<code-example path="dynamic-component-loader/src/app/app.module.ts" region="entry-components" title="src/app/app.module.ts (entry components)" linenums="false">
</code-example>
2017-03-31 19:57:13 -04:00
{@a common-interface}
## The _AdComponent_ interface
## 公共的 `AdComponent` 接口
2017-03-31 19:57:13 -04:00
In the ad banner, all components implement a common `AdComponent` interface to
standardize the API for passing data to the components.
在广告条中,所有组件都实现了一个公共接口 `AdComponent`,它定义了一个标准化的 API来把数据传给组件。
2017-03-31 19:57:13 -04:00
Here are two sample components and the `AdComponent` interface for reference:
下面就是两个范例组件及其 `AdComponent` 接口:
<code-tabs>
<code-pane title="hero-job-ad.component.ts" path="dynamic-component-loader/src/app/hero-job-ad.component.ts">
</code-pane>
<code-pane title="hero-profile.component.ts" path="dynamic-component-loader/src/app/hero-profile.component.ts">
</code-pane>
<code-pane title="ad.component.ts" path="dynamic-component-loader/src/app/ad.component.ts">
</code-pane>
</code-tabs>
2017-03-31 19:57:13 -04:00
{@a final-ad-baner}
## Final ad banner
2017-08-06 01:21:34 -04:00
## 最终的广告栏
2018-03-03 08:06:01 -05:00
The final ad banner looks like this:
2018-03-03 08:06:01 -05:00
最终的广告栏是这样的:
docs(aio): image sweep (#16609) * fix(aio): allow code blocks to clear floated images Previously the negative margin on the code headings were causing floated images to overlay the start of a code block. Now all code block successfully clear all floated elements. * feat(aio): add a `.clear` class for clearing floating images * fix(aio): tidy up image styles The css rules for `img.right` and `img.left` allow authors easy access to floating an image on the left or right, respectively. The `.image-display` rule which was always found on a figure has been simplified so that all figures have this styling. It is very unlikely that a figure will be used outside the content area; and at this time it seems like `figure` is as good an indicator that we want this kind of styling as anything. Now that images are all tagged with width and height values, we cannot assume to modify these dimensions via CSS as it can cause the image to lose its correct proportions. Until we find a better solition we must set `height` to `auto` when the screen width is below 1300px to ensure that these images maintain their proportions as they get shrunk to fit. * docs(aio): general tidy up of image HTML in guides Previously, the guides have a lot of inline image styling and unnecessary use of the `image-display` css class. Images over 700px are problematic for guide docs, so those have been given specific widths and associated heights. * docs(aio): use correct anchor for "back to the top" link The `#toc` anchor does not work when the page is wide enough that the TOC is floating to the side. * build(aio): add `#top-of-page` to path variants for link checking Since the `#top-of-page` is outside the rendered docs the `checkAnchorLinks` processor doesn't find them as valid targets for links. Adding them as a `pathVariant` solves this problem but will still catch links to docs that do not actually exist. * fix(aio): ensure that headings clear floated images * fix(aio): do not force live-example embedded image to 100% size This made them look too big, generally. Leaving them with no size means that they will look reasonable in large viewports and switch to 100% width in narrow viewports.
2017-05-09 18:53:32 -04:00
<figure>
2018-03-03 08:06:01 -05:00
<img src="generated/images/guide/dynamic-component-loader/ads.gif" alt="Ads">
</figure>
2017-03-31 19:57:13 -04:00
2017-08-06 01:21:34 -04:00
See the <live-example name="dynamic-component-loader"></live-example>.
参见<live-example name="dynamic-component-loader"></live-example>