File

projects/ngui-common/src/lib/ngui-utils/src/dynamic-component.service.ts

Description

Insert a component dynamically using a service

Example

import { DynamicComponentService } from './dynamic.component.service';
import { MyDynamicComponent } from './my-1.component';

@Component({
template: ` ... <div #dymamic></div>`
})
export class MyComponent {
@ViewChild('dynamic', {read:ViewContainerRef}) vcr: ViewContainerRef;

constructor(public dcs: DynamicComponentService) {}

insertComp() {
let compRef = this.dcs.createComponent(MyDynamicComponent, this.vcr);
ths.dcs.insertComonent(cmpRef);
compRef.instance.items = [1,2,3];              // dealing with @input
compRef.instance.output$.subscribe(val => {}); // dealing with @output
}
}

Index

Properties
Methods

Constructor

constructor(factoryResolver)
Parameters :
Name Optional
factoryResolver No

Methods

createComponent
createComponent(component: any, into?: ViewContainerRef)

returns component reference The reason to seperate createCompnent and insertComponent is to allow some actions before we insert into a hostView. e.g styling, setting attributes, etc

Parameters :
Name Type Optional
component any No
into ViewContainerRef Yes
Returns : ComponentRef<any>
insertComponent
insertComponent(componentRef: ComponentRef)

insert component

Parameters :
Name Type Optional
componentRef ComponentRef<any> No
Returns : Component

Properties

factoryResolver
Type : ComponentFactoryResolver

used to create a factory from a component class

rootViewContainer
Type : ViewContainerRef

defines where a dynamic components insert into

import {
  Component,
  ComponentFactoryResolver,
  ComponentRef,
  Inject,
  Injectable,
  ViewContainerRef
} from '@angular/core';

/**
 * Insert a component dynamically using a service
 *

 ### Example
 ```ts
 import { DynamicComponentService } from './dynamic.component.service';
 import { MyDynamicComponent } from './my-1.component';

 @Component({
   template: ` ... <div #dymamic></div>`
 })
 export class MyComponent {
   @ViewChild('dynamic', {read:ViewContainerRef}) vcr: ViewContainerRef;

   constructor(public dcs: DynamicComponentService) {}

   insertComp() {
     let compRef = this.dcs.createComponent(MyDynamicComponent, this.vcr);
     ths.dcs.insertComonent(cmpRef);
     compRef.instance.items = [1,2,3];              // dealing with @input
     compRef.instance.output$.subscribe(val => {}); // dealing with @output
   }
 }
 ```
 */
@Injectable()
export class DynamicComponentService {
  /** used to create a factory from a component class */
  factoryResolver: ComponentFactoryResolver;
  /** defines where a dynamic components insert into */
  rootViewContainer: ViewContainerRef;

  constructor(@Inject(ComponentFactoryResolver) factoryResolver) {
    this.factoryResolver = factoryResolver;
  }

  /**
   * returns component reference
   * The reason to seperate `createCompnent` and `insertComponent` is
   * to allow some actions before we insert into a hostView.
   * e.g styling, setting attributes, etc
   */
  createComponent(component: any, into?: ViewContainerRef): ComponentRef<any> {
    this.rootViewContainer = into || this.rootViewContainer;
    const factory = this.factoryResolver.resolveComponentFactory(component);

    return factory.create(this.rootViewContainer.parentInjector);
  }

  /**
   * insert component
   */
  insertComponent(componentRef: ComponentRef<any>): Component {
    const compId = `ngui-dyn-${Math.floor(Math.random() * 10 ** 7) + 10 ** 6}`;
    componentRef.location.nativeElement.setAttribute('id', compId);
    componentRef.instance.id = compId;

    this.rootViewContainer.insert(componentRef.hostView);

    return componentRef.instance;
  }

}

results matching ""

    No results matching ""