从Angular2中的子组件切换父组件中的属性(类似于AngularJS中的$ emit)

时间:2016-03-01 09:47:00

标签: typescript angular eventemitter

我目前正在教自己Angular2,我有一个在AngularJS中很容易纠正的实际问题,但目前我正在通过示例查找Angular2的解决方案。无论如何,我有一个名为App的顶级组件,在该组件中我有一个应用css类的属性,这里是我的HTML用作组件模板 - 你可以看到我希望在哪里应用css使用ngClass的课程。 showMenu值只是一个布尔值。

<div class="navigation" role="navigation">
    <ul>
        <li><a [routerLink]="['Home]">Home</a></li>
        <li><a [routerLink]="['Mail']">Mail</a></li>
        <li><a [routerLink]="['Friends']">Friends</a></li>
        <li><a [routerLink]="['Games']">Games</a></li>
    </ul>
</div>
<!-- below is where I wish to toggle the class -->
<div class="site-wrap" [ngClass]="{'slide-right': showMenu}">
    <div class="container-fluid">
        <div class="row nav">
            <top-navigation></top-navigation>
        </div>
    </div>

    <div class="container-fluid content-gradient">
        <div class="row">
            <div class="col-md-10">
                <router-outlet></router-outlet>
            </div>
            <div class="col-md-2">
                <contacts-list></contacts-list>
            </div>
        </div>
    </div>

    <div class="container-fluid footer">
        <footer-navigation></footer-navigation>
    </div>
</div>

以下是App的组件代码(这里看不多......)

export class App {

    showMenu: boolean = false;
    constructor() {
    }
    // I need to listen for the toggle somehow?
    toggleMenu():void {
        this.showMenu = !this.showMenu;
    }

}

现在你可以看到我有一个带有选择器top-navigation的子组件。此组件TopNavigation在按钮上有一个方法,我希望在父级/顶级组件中切换showMenu的值。这是我目前的代码(只是小提取)

export class TopNavigation {

          constructor() {
            // Things happening here
          }

          toggleMenu():void {
            // I want to toggle the parent property by somekind of $emit
          }
    }

这是组件的模板HTML(不是全部)。

<div class="col-xs-2 col-sm-1  hidden-md hidden-lg hamburger-menu">
    <button class="btn btn-default" (click)="toggleMenu()">&#9776;</button>
</div>

在AngularJS中,我会在$scope.$emit函数上有一个toggleMenu,在父级中我会有一个$scope.$on来监听这个事件。我目前正在阅读Angular2中的EventEmitters,但是如果有人有快速解释/示例,我会非常感激,特别是因为我认为这是Angular开发人员的常见/实际日常任务。

2 个答案:

答案 0 :(得分:6)

我可以利用定义EventEmitter属性的共享服务。您的App组件将在其上进行订阅,以便在TopNavigation组件触发相应事件时收到通知。

  • 共享服务

    @Injectable()
    export class MenuService {
      showMenuEvent: EventEmitter = new EventEmitter();
    }
    

    不要忘记在boostrap函数中定义相应的提供程序,以便能够为整个应用程序共享相同的服务实例:`bootstrap(App,[MenuService]);

  • App组件

    @Component({ ... })
    export class App {
      showMenu: boolean = false;
      constructor(menuService: MenuService) {
        menuService.showMenuEvent.subscribe(
          (showMenu) => {
            this.showMenu = !this.showMenu;
          }
       );
     }
    

    }

  • TopNavigation组件:

    export class TopNavigation {
      constructor(private menuService: MenuService) {
        // Things happening here
      }
    
      toggleMenu():void {
        this.showMenu = this.showMenu;
        this.menuService.showMenuEvent.emit(this.showMenu);
      }
    }
    

有关详细信息,请参阅此问题:

答案 1 :(得分:4)

您可以使用双向绑定。

<top-navigation [(toggleMenu)]="showMenu"></top-navigation>
也许单向也行,我不知道你的具体要求

<top-navigation (toggleMenuChange)="showMenu=$event"></top-navigation>
export class TopNavigation {
      @Output() toggleMenuChange: EventEmitter = new EventEmitter();

      // for two-way binding
      @Input() toggleMenu:boolean = false;

      constructor() {
        // Things happening here
      }

      toggleMenu():void {
        this.toggleMenu = !this.toggleMenu;
        this.toggleMenu.emit(this.toggleMenu);
      }
}