Angular2 - 如何从应用程序外部调用组件功能

时间:2016-02-09 16:08:35

标签: angular

我正在使用具有回调功能的javascript对象。一旦回调被触发,我想调用Angular2组件内的一个函数。

示例 HTML文件。

    var run = new Hello('callbackfunction');

    function callbackfunction(){   
     // how to call the function **runThisFunctionFromOutside**
   }
   <script>
      System.config({
        transpiler: 'typescript', 
        typescriptOptions: { emitDecoratorMetadata: true }, 
        packages: {'js/app': {defaultExtension: 'ts'}} 
      });
      System.import('js/app/main')
            .then(null, console.error.bind(console));
    </script>

我的 App.component.ts

import {Component NgZone} from 'angular2/core';
import {GameButtonsComponent} from './buttons/game-buttons.component';
@Component({
  selector: 'my-app',
  template: ' blblb'
})
export class AppComponent {

constructor(private _ngZone: NgZone){}

ngOnInit(){
    calledFromOutside() {
        this._ngZone.run(() => {
          this.runThisFunctionFromOutside();
    });
  }
  }
runThisFunctionFromOutside(){
   console.log("run");
}

如何调用App.component.ts中的 runThisFunctionFromOutside 函数

6 个答案:

答案 0 :(得分:52)

我基本上遵循this answer,但我不希望我的“外部”代码知道任何关于NgZone的内容。这是app.component.ts:

import {Component, NgZone, OnInit, OnDestroy} from '@angular/core';

@Component({
  selector: 'my-app',
  templateUrl: 'app.component.html'
})
export class AppComponent implements OnInit, OnDestroy {
  constructor(private ngZone: NgZone) {}

  ngOnInit() {
    window.my = window.my || {};
    window.my.namespace = window.my.namespace || {};
    window.my.namespace.publicFunc = this.publicFunc.bind(this);
  }

  ngOnDestroy() {
    window.my.namespace.publicFunc = null;
  }

  publicFunc() {
    this.ngZone.run(() => this.privateFunc());
  }

  privateFunc() {
    // do private stuff
  }
}

我还必须为TypeScript添加一个定义来扩展window对象。我把它放在typings.d.ts:

interface Window { my: any; }

现在,从控制台调用该功能非常简单:

my.namespace.publicFunc()

答案 1 :(得分:51)

另见How do expose angular 2 methods publicly?

构建组件时,将其自身分配给全局变量。然后你可以从那里引用它并调用方法。 不要忘记使用zone.run(() => { ... }),以便Angular获得有关所需更改检测运行的通知。

 function callbackfunction(){   
   // window['angularComponentRef'] might not yet be set here though
   window['angularComponent'].zone.run(() => {
     runThisFunctionFromOutside(); 
   });
 }

constructor(private _ngZone: NgZone){
  window['angularComponentRef'] = {component: this, zone: _ngZone};
}

ngOnDestroy() {
  window.angularComponent = null;
}

Plunker example1

在浏览器控制台中,您必须从<topframe>切换到plunkerPreviewTarget....,因为Plunker会在iFrame中执行代码。然后运行

window['angularComponentRef'].zone.run(() => {window['angularComponentRef'].component.callFromOutside('1');})

window.angularComponentRef.zone.run(() => {window.angularComponentRef.componentFn('2');})

另一种方法

将在Angular之外调度事件并在Angular中收听它们,如Angular 2 - communication of typescript functions with external js libraries

中所述

Plunker example2(来自评论)

答案 2 :(得分:6)

以下是一个解决方案。

function callbackfunction(){   
   window.angularComponent.runThisFunctionFromOutside();
}
       <script>
          System.config({
            transpiler: 'typescript', 
            typescriptOptions: { emitDecoratorMetadata: true }, 
            packages: {'js/app': {defaultExtension: 'ts'}} 
          });
          System.import('js/app/main')
                .then(null, console.error.bind(console));
        </script>

我的App.component.ts

import {Component NgZone} from 'angular2/core';
import {GameButtonsComponent} from './buttons/game-buttons.component';
@Component({
    selector: 'my-app',
       template: ' blblb'
})
export class AppComponent {

  constructor(private _ngZone: NgZone){
  window.angularComponent = {runThisFunctionFromOutside: this.runThisFunctionFromOutside, zone: _ngZone};
}


    runThisFunctionFromOutside(){
      console.log("run");
    }
}

答案 3 :(得分:1)

不使用全局变量的另一种方法是使用传递控制对象并将其属性绑定到要公开的变量和方法。

export class MyComponentToControlFromOutside implements OnChanges {

  @Input() // object to bind to internal methods
  control: {
    openDialog,
    closeDialog
  };

  ngOnChanges() {
    if (this.control) {
      // bind control methods to internal methods
      this.control.openDialog = this.internalOpenDialog.bind(this);
      this.control.closeDialog = this.internalCloseDialog;
    }
  }

  internalOpenDialog(): Observable<boolean> {
    // ...
  }

  internalCloseDialog(result: boolean) {
    // ...
  }
}
export class MyHostComponent {
   controlObject= {};
}
<my-component-to-control [control]="controlObject"></my-component-to-control>

<a (click)="controlObject.open()">Call open method</a>

答案 4 :(得分:1)

使用回调时遇到类似情况&#39; eventClick&#39; fullCalendar库的回调是从角度区域外返回的,导致我的应用程序具有部分和不可靠的效果。我能够将区域方法和闭包引用组合到组件中,如下所示,以便引发输出事件。一旦我开始在zone.run()方法内执行事件,事件及其效果再次可以预测并通过角度变化检测来获取。希望这有助于某人。

constructor(public zone: NgZone) { // code removed for clarity
}

ngOnInit() {
    this.configureCalendar();
}

private configureCalendar() {
    // FullCalendar settings
    this.uiConfig = {
        calendar: { // code removed for clarity

        }
    };

    this.uiConfig.calendar.eventClick = this.onEventClick();

}

private onEventClick() {
    const vm = this;

    return function (event, element, view) {
        vm.zone.run(() => {
            vm.onSequenceSelected.emit(event.sequenceSource);                    
        });

        return false;

    };
}

答案 5 :(得分:0)

只需添加到@Dave Kennedy

  

现在,从控制台调用该功能非常简单:

     

my.namespace.publicFunc()

1)如果我们尝试从不同的域访问我们组件的公共方法,您将陷入CORS问题(如果服务器和客户端代码都存在于同一个问题中,则可以解决交叉源问题机)。

2)如果您使用javascript从服务器调用此方法,则必须使用window.opener.my.namespace.publicFunc()而不是window.my.namespace.publicFunc():

window.opener.my.namespace.publicFunc();