如何在组件B中呈现组件A的一部分

时间:2018-01-22 06:06:16

标签: angular ng-template ng-content

我有一个组件AB

<div class="A">
    <div class="A1"></div>
    <div class="A2"></div>
</div>

<div class="B">
    <!--want to display <div class="A1"></div> here-->
</div>

组件B不包含组件A或方式

<componentA></componentA>
<componentB></componentB>

我尝试将<div class="A1"></div>打包到ng-template,然后在componentB我正在查询,但始终是undefined

@ViewChild('templ') templ: TemplateRef<any>;

然后我意识到@ViewChild是查询组件子项,那么如何查询和呈现<ng-template>它不是子组件?

3 个答案:

答案 0 :(得分:2)

将组件A的模板渲染为组件B的内容

<ComponentB>
   <ComponentA></ComponentA>
</ComponentB>

componentB内部模板内容&#39;内容&#39; (在这种情况下为ComponentA的模板)

ComponentB.component.html

<ng-content> </ng-content>

要在ComponentA中使用ComponentB

访问呈现的ContentChild
@ContentChild(ComponentA) footer: componentA;

答案 1 :(得分:2)

请看看Alex Rickabaugh的这个awesome sideNav sample,准确演示你需要的东西。

根据您的需要,您需要在ng-container中定义ComponentA。然后,您需要使用自定义指令标记ComponentB的一部分,并且有一项连接ComponentAComponentB的服务:

<div class="A1"><div>
<div class="A2" *leftNav >
  <h3>Content</h3>
</div>

指令:

@Directive({
  selector: '[leftNav]',
})
export class LeftNavDirective implements OnInit {
  constructor(private leftNav: LeftNav, private ref: TemplateRef<any>) {}

  ngOnInit(): void {
    this.leftNav.setContents(this.ref);
  }
}

服务:

@Injectable()
export class LeftNav {
  private _state = new BehaviorSubject<TemplateRef<any>|null>(null);
  readonly contents = this._state.asObservable();

  setContents(ref: TemplateRef<any>): void {
    this._state.next(ref);
  }

  clearContents(): void {
    this._state.next(null);
  }
}

最后,在您的ComponentB中,您必须在<div class="A1"> ... <div>订阅ngAfterViewInit内容:

ngAfterViewInit(): void {
    this
      .leftNav
      .contents
      .subscribe(ref => {
        if (this._current !== null) {
          this._current.destroy();
          this._current = null;
        }
        if (ref === null) {
          return;
        }
        this._current = this.vcr.createEmbeddedView(ref);
    });
  }

P.S该主题有videoslides

答案 2 :(得分:0)

您可以在父组件中创建公共ng-template,并将其templateRef传递给组件Input绑定中的子组件。之后内部组件将填充该模板的模板绑定。

这意味着在组件内添加@Input a1: TemplateRef<any>;,并将ng-template放在组件A&amp;组件的父组件中。 B的父母(基本上他们有同一个父母)。

<ng-template #a1>
    <div class="A1">
       {{item.Description}}
    </div>
</ng-tmeplate>

组件A模板

<div class="A">
    <ng-container 
       *ngTemplateOutlet="a1;context:{item: valueFromComponentA}">
    </ng-container>
    <div class="A2"></div>
</div>

组件B temlpate

<div class="B">
    <ng-container 
       *ngTemplateOutlet="a1;context:{item: valueFromComponentB}">
    </ng-container>
</div>