使用另一个模块中的组件


199

我有使用angular-cli生成的Angular 2.0.0应用程序。

当我创建一个组件并将其添加到AppModule的声明数组时,一切都很好,它可以工作。

我决定将组件分开,因此我创建了TaskModule和组件TaskCard。现在我想用TaskCard在的组成部分之一AppModule(的Board成分)。

AppModule:

import { BrowserModule } from '@angular/platform-browser';
import { NgModule } from '@angular/core';
import { FormsModule } from '@angular/forms';
import { HttpModule } from '@angular/http';

import { AppComponent } from './app.component';
import { BoardComponent } from './board/board.component';
import { LoginComponent } from './login/login.component';

import { MdButtonModule } from '@angular2-material/button';
import { MdInputModule } from '@angular2-material/input';
import { MdToolbarModule } from '@angular2-material/toolbar';

import { routing, appRoutingProviders} from './app.routing';
import { PageNotFoundComponent } from './page-not-found/page-not-found.component';

import { UserService  } from './services/user/user.service';
import { TaskModule } from './task/task.module';


@NgModule({
  declarations: [
    AppComponent,
    BoardComponent,// I want to use TaskCard in this component
    LoginComponent,
    PageNotFoundComponent
  ],
  imports: [
    BrowserModule,
    FormsModule,
    HttpModule,
    MdButtonModule,
    MdInputModule,
    MdToolbarModule,
    routing,
    TaskModule // TaskCard is in this module
  ],
  providers: [UserService],
  bootstrap: [AppComponent]
})
export class AppModule { }

TaskModule:

import { NgModule } from '@angular/core';
import { TaskCardComponent } from './task-card/task-card.component';

import { MdCardModule } from '@angular2-material/card';

@NgModule({
  declarations: [TaskCardComponent],
  imports: [MdCardModule],
  providers: []
})
export class TaskModule{}

整个项目位于https://github.com/evgdim/angular2(看板文件夹)上

我想念什么?我有什么做用TaskCardComponentBoardComponent

Answers:


389

这里的主要规则是:

在组件模板的编译期间适用的选择器由声明该组件的模块以及该模块的导入的导出的可传递关闭确定。

因此,尝试将其导出:

@NgModule({
  declarations: [TaskCardComponent],
  imports: [MdCardModule],
  exports: [TaskCardComponent] <== this line
})
export class TaskModule{}

我应该出口什么?

导出其他模块中的组件应该能够在其模板中引用的声明类。这些是您的公开课。如果不导出类,则该类将保持私有状态,仅对该模块中声明的其他组件可见。

创建新模块(无论是否懒惰),任何新模块并在其中声明任何内容的那一刻,该新模块都处于干净状态(如Ward Bell在 https://devchat.tv/adv-in-angular/119中所述) -aia-避免常见的陷阱(在angular2中

Angular 为每个s 创建传递模块@NgModule

此模块收集从另一个模块导入的指令(如果导入模块的可传递模块已导出指令)或在当前模块中声明

当属于模块的angular编译模板时,将X使用X.transitiveModule.directives中收集的那些指令。

compiledTemplate = new CompiledTemplate(
    false, compMeta.type, compMeta, ngModule, ngModule.transitiveModule.directives);

https://github.com/angular/angular/blob/4.2.x/packages/compiler/src/jit/compiler.ts#L250-L251

在此处输入图片说明

根据上图这样

  • YComponent不能ZComponent在其模板中使用,因为不包含的directives数组,因为尚未导入其传递模块包含在数组中。Transitive module YZComponentYModuleZModuleZComponentexportedDirectives

  • XComponent模板内,我们可以使用ZComponent因为Transitive module X有指令数组,其中包含ZComponent因为 XModule导入模块(YModule)和导出模块(ZModule)的输出指令ZComponent

  • AppComponent模板内我们无法使用,XComponent因为AppModule进口XModuleXModule没有出口XComponent

也可以看看


13
如何在导入模块的路由定义中使用此“ TaskCardComponent”?
jackOfAll

17
真是个好答案。您创建图纸了吗?如果是这样,我无语。并不是每个人都在做出这样的努力。谢谢
Royi Namir

4

@yuruzi,如果没有参考stackoverflow.com/questions/47246638/… ,我将无法直接通过dom节点... plnkr.co/edit/DnnjFBa3HLzFKNIdE4q5?p=preview
Karty

@yurzui ...我不知道YComponent如何导出ZModule,因为我看到它是一个单独的文件(y.module.ts),并且没有任何导入以便导出其他模块(即z (.module.ts)[请告诉我,如果它是基本问题]
OmGanesh,

42

您必须export从您的NgModule

@NgModule({
  declarations: [TaskCardComponent],
  exports: [TaskCardComponent],
  imports: [MdCardModule],
  providers: []
})
export class TaskModule{}

2
在TaskModule导入AppModule中之前,此方法有效。延迟加载TaskModule时失败。
阿伦(Arun)

40

(角度2-角度7)

组件只能在单个模块中声明。为了使用另一个模块中的组件,您需要执行两个简单的任务:

  1. 将组件导出到另一个模块中
  2. 将另一个模块导入当前模块

第一个模块:

有一个组件(我们称它为“ ImportantCopmonent”),我们想在第二个模块的页面中重复使用。

@NgModule({
declarations: [
    FirstPage,
    ImportantCopmonent // <-- Enable using the component html tag in current module
],
imports: [
  IonicPageModule.forChild(NotImportantPage),
  TranslateModule.forChild(),
],
exports: [
    FirstPage,
    ImportantCopmonent // <--- Enable using the component in other modules
  ]
})
export class FirstPageModule { }

第二模块:

通过导入FirstPageModule,重用“ ImportantCopmonent”

@NgModule({
declarations: [
    SecondPage,
    Example2ndComponent,
    Example3rdComponent
],
imports: [
  IonicPageModule.forChild(SecondPage),
  TranslateModule.forChild(),
  FirstPageModule // <--- this Imports the source module, with its exports
], 
exports: [
    SecondPage,
]
})
export class SecondPageModule { }

2

请注意,为了创建一个所谓的“功能模块”,您需要在其中导入CommonModule。因此,您的模块初始化代码将如下所示:

import { NgModule } from '@angular/core';
import { CommonModule } from '@angular/common';

import { TaskCardComponent } from './task-card/task-card.component';
import { MdCardModule } from '@angular2-material/card';

@NgModule({
  imports: [
    CommonModule,
    MdCardModule 
  ],
  declarations: [
    TaskCardComponent
  ],
  exports: [
    TaskCardComponent
  ]
})
export class TaskModule { }

此处提供更多信息:https : //angular.io/guide/ngmodule#create-the-feature-module



0

一种很棒的方法是从加载模块NgModuleFactory,您可以通过调用以下命令将模块加载到另一个模块中:

constructor(private loader: NgModuleFactoryLoader, private injector: Injector) {}

loadModule(path: string) {
    this.loader.load(path).then((moduleFactory: NgModuleFactory<any>) => {
        const entryComponent = (<any>moduleFactory.moduleType).entry;
        const moduleRef = moduleFactory.create(this.injector);
        const compFactory = moduleRef.componentFactoryResolver.resolveComponentFactory(entryComponent);
        this.lazyOutlet.createComponent(compFactory);
    });
}

我从这里得到的。


现在已弃用NgModuleFactoryLoader,那么执行此操作的最佳替代方法是什么?
Muzaffar Mahmood

-2

解决了如何在其他模块中使用在模块中声明的组件。

基于Royi Namir的解释(非常感谢)。在使用延迟加载时,缺少在任何其他模块中重用模块中声明的组件的部分。

第一个:将组件导出到包含它的模块中:

@NgModule({
  declarations: [TaskCardComponent],
  imports: [MdCardModule],
  exports: [TaskCardComponent] <== this line
})
export class TaskModule{}

第二个:在要使用TaskCardComponent的模块中:

import { NgModule } from '@angular/core';
import { CommonModule } from '@angular/common';
import { MdCardModule } from '@angular2-material/card';

@NgModule({
  imports: [
   CommonModule,
   MdCardModule
   ],
  providers: [],
  exports:[ MdCardModule ] <== this line
})
export class TaskModule{}

这样,第二个模块导入第一个模块,该模块导入和导出组件。

当我们在第二个模块中导入模块时,我们需要再次导出它。现在,我们可以在第二个模块中使用第一个组件。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.