10

我的要求是构建一个具有 2 个或更多 html 模板的组件,其中每个 html 模板具有至少 20 个控件,并根据少数条件加载该特定模板。

注意:我选择了 3 个不同的模板,因为控件根据模板类型而有所不同,其中作为输入的单个 ts 文件和驱动获取和保存模板中值的逻辑保持不变。因此,我决定将 3 个模板和一个 ts 文件作为一个组件。

//sub.component.ts
@Component({
     selector: 'sub-component',
     template: `
               <div [ngSwitch]="templateType">
                    <ng-template *ngSwitchCase="'1'"> ${require('./sub1.component.html')} </template>
                    <ng-template *ngSwitchCase="'2'"> ${require('./sub2.component.html')} </template>
                    <ng-template *ngSwitchCase="'3'"> ${require('./sub3.component.html')} </template>
                    <ng-template ngSwitchDefault> ${require('./sub1.component.html')} </template>
</div>
    `
})

我已经尝试过上述替代方法,因为它看起来是实现该行为的简单解决方案,但编译失败并找不到要求。在 AngularJS 中,我们有 ng-Include 来填充任何模板,但看起来 ng-template 不支持加载外部 html 内容。

请不要将此标记为重复,因为出现了许多与此类似的查询,但大多数解决方案已被弃用或不适用于 Angular 4。请提供替代方案,而不是附加不同的链接。

4

2 回答 2

13

我知道这个问题很老,但希望这对尝试类似事情的人有所帮助。

不幸的是,最简单的方法是让每个模板成为自己的组件。否则,您必须注入和清理 HTML。由于注入未经清理的 HTML 的安全风险,他们删除了 ng-include 和类似功能。如果您不必在模块中专门导入和声明所有这些附加组件,它就不会是一个 PITA,但是唉...

您可以创建一个简单的指令来获取 templateRefs,然后在页面上查询具有这些指令的元素,从中获取模板 ref,并将它们插入到其他地方。这至少可以让您将所有模板保存在单独的文件中。我通常将 3 或 4 个模板放在一个单独的组件中,并将它们包含在要使用 . 我将描述如何做到这一点。

获取模板引用的指令

import { Directive, TemplateRef, Input } from '@angular/core';

@Directive({
 selector: 'get-template',
})
export class GetTemplateDirective {
  @Input() name: string;
  constructor(public template: TemplateRef<any>) {  }
}

然后对于模板,创建一个包含所有模板的超级简单组件

@Component({
  selector: 'sub-component-templates',
  template: `

<ng-template get-template [name]="tpl1">
  Put Whatever here, including other components if you please
</ng-template> 

<ng-template get-template [name]="tpl2">
  Different template here
</ng-template> 

 ... and so on and so on...
`
})
export class Templates { }

将所有相关的新组件导入到您的模块中,然后将它们包含在将呈现模板的主组件中

我通常用 ng-content 来做,所以它在父组件中清楚地表明该组件正在为其模板引用另一个组件。

例如,在父..

<sub-component>
    <sub-component-templates></sub-component-templates>
</sub-component>

然后在子组件中

 import { Component, ViewChild, ContentChildren, QueryList } from '@angular/core';
import { GetTemplateDirective } from 'wherever';

@Component({
selector: 'sub-component',
template: `

<ng-content></ng-content>

 <div #templateRenderer></div>
`
})
export class SubComponent {

@ViewChild('templateRenderer',{read:ViewContainerRef}) anchor: ViewContainerRef;
@ContentChildren(GetTemplateDirective) templates: QueryList<GetTemplateDirective>;

ngAfterContentInit()  {

  ... at this stage, you will have access to all the included templates with that directive on them. You can perform your logic to choose which one you want. Once you have selected the proper one, you can embed it like so ... 
 let desiredTemplateName = 'whatever';

     for (let t of this.templates.toArray()) {
       if(t.name === desiredTemplateName) {
          this.anchor.createEmbeddedView(t.template);
          break;        
       } 
     }  
   }

}

你可以看到这对于你想要做的事情来说是非常复杂的。将它们创建为单独的组件会更容易,并使用 ngSwitchCase 来选择合适的组件。我上面描述的方法的优点是它可以让您将模板保存在任何您真正想要的地方,并且您可以在同一个外部组件中包含 100 个(实际上不超过带有模板的最小装饰组件)如果您想要,或者通过服务移动它们,或其他任何东西。

有关如何使用编译器的工作示例,请参见此处 - https://plnkr.co/edit/fdP9Oc?p=info

还是挺复杂的……

如果您将模板存储为类的属性,您可以稍后根据需要更改它。只需添加一个模板引用导入...

 import { Component, ViewChild, ContentChildren, QueryList, TemplateRef } from '@angular/core';

然后创建一个属性

 template: TemplateRef<any>;

然后稍后您可以使用查询列表中的一个将其切换出来,并使用视图容器的方法再次创建嵌入式视图。

Angular 2 / 4 使某些事情变得更容易......并使某些事情变得更加困难。但我想在这种情况下,它是以安全的名义。

于 2017-08-27T07:03:37.850 回答
0

实际上,当我们说话时,我正试图弄清楚这一点,当我遇到上面的答案时,它给了我一个我即将尝试的想法。希望我能取得一些成功,并能够用更具体的东西来更新这个答案。

我一直在做我的问卷调查,我的问题是带有文字答案的多项选择题,带有“1-10 级”机制的多项选择题,然后是一些需要文字回答的问题。

我认为制作一个component并将每个模板包装在一个ngIf连接到数据将传递到的类中的变量的条件中可以触发模板。

所以数据看起来像这样

Questions:[{
    question:'blahblahblah',
    answers: [..array of answers..],
    template: 'a'
    },
    {
    question: 'yaddayaddayadda',
    answers: [],
    template: 'b'
    },
    {
    etc.
    }
    ]

然后在组件类中你可以有这样的东西

@Component ({
    selector: 'question-component',
    templateUrl: './template.html'
})

export class QuestionComponent {

    @input() data: yourDataType;

    constructor() {} 

}

然后在这个组件的模板中有类似的东西

<div *ngIf="data.template === a">
    <!-- code for template with binding and everything -->
</div>

<div *ngIf="data.template === b">
    <!-- code -->
</div>

<!-- etc. etc. etc. -->

然后在主组件的模板中,您可以执行类似的操作

<div *ngFor = "question of Questions">
    <question-component [data]="question"></question-component>
</div>

这一切都是我脑海中的假设,所以我可能会遗漏一些东西,但我觉得在此期间开始探索这里是值得的。我要看看我是否可以让它满足我的需求。

于 2017-08-27T06:13:19.293 回答