当前位置: 首页 > article >正文

Angular 15 独立组件详解

前言

在 Angular 15 中,独立 APIs 正式稳定,开发人员可以使用独立组件开发各种组件、指令、管道和构建应用程序。独立 APIs 减少了在开发时对 ngModule 的依赖,提升了开发体验。
目前,组件库 ngx-tethys 紧跟 Angular 脚步,现在已支持独立组件。

什么是独立组件

独立组件提供了一种简化的方式来构建 Angular 应用程序。独立组件、指令和管道旨在通过减少对 NgModule 的需求来简化创作体验。现有应用程序可以选择性地以增量方式采用新的独立风格,而无需任何重大更改。

如何创建独立组件

NgModule 创建组件

回顾下在独立 APIs 之前,创建组件、指令、管道需要完成哪些:
创建组件、指令、管道
创建 ngModule ,在此 ngModule 中需要:
通过 declarations 声明新创建的组件、指令和管道
通过 imports 导入在该模块中使用到的其他组件、指令、管道所属的模块
通过 exports 导出可供其他模块使用的组件、指令和管道
通过 providers 设置一些供组件使用的服务

比如创建一个按钮组件:
创建组件

@Component({
    selector: 'button',
    templateUrl: './button.component.html',
    changeDetection: ChangeDetectionStrategy.OnPush
})
export class ButtonComponent {}

定义 ButtonModule

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

import { ButtonComponent } from './button.component';

@NgModule({
    declarations: [ButtonComponent,],
    imports: [CommonModule],
    exports: [ButtonComponent,],
    providers: []
})
export class ButtonModule {}

每次新增、修改组件时,都需要在 ngModule 中进行更改,过程比较繁琐。

独立 APIs 创建组件

使用独立 APIs 创建组件、指令、管道:
创建组件、指令、管道时,在元数据中配置 standalone: true
在元数据中 imports 使用的其他独立组件、模块

比如创建按钮组件:其中 NgIf 、 NgClass 为 Angular 提供的独立组件

@Component({
    selector: 'button',
    templateUrl: './button.component.html',
    changeDetection: ChangeDetectionStrategy.OnPush,
    standalone: true,
    imports: [NgIf, NgClass]
})
export class ButtonComponent {
}

回顾之前使用 NgModule 创建组件时,在模块定义中还会导出可用于其他模块/组件使用的组件列表,如果使用模块中多个组件,直接导入模块就行。使用独立组件如何设置一系列组件供其他模块/组件使用呢?

方式一:
定义一个 ngModule 作为多个独立组件的集合。此处需要注意的是:独立组件不可以再次在 ngModule 中声明。

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

import { ThyButtonComponent } from './button.component';


@NgModule({
    imports: [CommonModule, ThyButtonComponent],
    exports: [ButtonComponent]
})
export class ButtonModule {}

方式二:
定义一个常量数组,将多个独立组件导出。此处需要注意的是: 常量需要 as const ,为 Angular 编译器提供了正确编译所需的额外信息。

export const BUTTON_COMPONENTS = [ThyButtonComponent] as const;

指令组合

如何使用
在指令支持独立指令后,独立指令还可以直接用于其他组件、指令,复用独立指令的逻辑。在指令或组件上使用 hostDirectives:[] ,将独立指令应用于组件。
如下示例, ThyFlex 为独立指令,指令有输入参数 thyDirection :

@Directive({
    selector: '[flex]',
    standalone: true,
    host: {
        class: 'd-flex'
    }
})
export class Flex implements OnInit, OnChanges {
    @Input() direction: FlexDirection;
    
    constructor() {}
}

组件 flex :

@Component({
    selector: 'flex',
    template: `<ng-content></ng-content>`,
    standalone: true,
    changeDetection: ChangeDetectionStrategy.OnPush,
    hostDirectives: [
        {
            directive: Flex,
            inputs: ['thyDirection: direction']
        }
    ],
    imports: [Flex]
})
export class FlexComponent {}

需要注意的是: 默认情况下,指令Flex 的输入、输出参数不会作为组件ThyFlexComponent 的公开 API,需要如上显示的定义。
使用:

<flex thyDirection="xxx"></flex>

执行顺序
使用组合指令后,组件和宿主指令执行顺序:
宿主指令和直接在模板中使用的组件、指令会经历相同的生命周期。但是,宿主指令总是会在应用它们的组件或指令之前执行它们的构造函数、生命周期钩子和绑定。因此顺序为:
指令 Flex 实例化
组件 FlexComponent 实例化
指令 Flex 接收输入,执行 ngOninit
组件 FlexComponent 接收输入,执行 ngOninit
指令 Flex 应用宿主绑定
组件 FlexComponent 应用宿主绑定

依赖注入

使用了 hostDirectives 的组件或指令可以注入这些宿主指令的实例。

@Component({
    selector: 'flex',
    template: `<ng-content></ng-content>`,
    standalone: true,
    changeDetection: ChangeDetectionStrategy.OnPush,
    hostDirectives: [
        {
            directive: Flex,
            inputs: ['direction']
        }
    ],
    imports: [Flex]
})
export class FlexComponent implements OnInit {
    private flexDirective = inject(Flex);

    constructor() {
        console.log('FlexComponent constructor');
    }

    ngOnInit(): void {
        console.log('FlexComponent ngOnInit');
    }
}

当把宿主指令应用于组件时,组件和宿主指令都可以定义提供者。如果带有 hostDirectives 的组件或指令以及这些宿主指令都提供相同的注入令牌,则带有 hostDirectives 的类定义的提供者会优先于宿主指令定义的提供者。

优势:
增强使用宿主指令的组件的功能;
增强代码的复用性

缺点:
过度使用宿主指令会影响应用程序的内存使用。

应用中使用

NgModule 启动应用

通过根模块启动应用,通过 bootstrapModule 启动,程序入口文件 main.ts 如下:

import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';

platformBrowserDynamic().bootstrapModule(AppModule)
  .catch(err => console.error(err));

AppModule 中,除了上边提到的 ngModule 中配置,还会通过 providers 设置全局的依赖,通过 bootstrap: [ AppComponent ] 设置根组件。启动后,将根组件插入到 index.html 页面中,进而构建组件树( bootstrap 中支持配置多个组件,一般情况跟组件只会设置一个组件)。

@NgModule({
  declarations: [
    AppComponent
  ],
  imports: [
    BrowserModule,
    AppRoutingModule
  ],
  providers: [],
  bootstrap: [AppComponent]
})
export class AppModule { }

独立应用启动

通过使用独立组件作为应用程序的根组件,可以在没有任何 ​ NgModule ​的情况下引导 Angular 应用程序。使用 ​ bootstrapApplication ​启动应用,配置根组件(根组件为独立组件):
main.ts:

bootstrapApplication(AppComponent).catch(err => console.error(err));

AppComponent

@Component({
  selector: 'app-root',
  standalone: true,
  imports: [CommonModule, RouterOutlet, TopBarComponent],
  templateUrl: './app.component.html',
  styleUrls: ['./app.component.scss']
})
export class AppComponent {
  title = 'demo-standalone';
}

回顾使用 ngModule 启动应用时,除了指定启动模块、启动组件,还需要配置全局服务和路由。在使用独立 APIs 启动应用如何配置呢?
依赖
可以在启动应用程序时,配置全局依赖。

bootstrapApplication(AppComponent, {
  providers: [
    ProjectStore
  ]
})
  .catch((err) => console.error(err));

如果第三方库仅支持 ngModule 模式配置依赖,可以使用 importProvidersFrom 配置。

import {LibraryModule} from 'ngmodule-based-library';

bootstrapApplication(PhotoAppComponent, {
  providers: [
    importProvidersFrom(
      RouterModule.forRoot(routes)
    )
  ]
});

此外,Angular 其他模块或者第三方库也支持了以 provide-* 的方法配置依赖。

provideRouter() // 可用于配置路由
provideHttpClient() // 可用于配置 HttpClient 服务
provideZoneChangeDetection() // 可用于配置 ngZone
provideAnimations() // 需要动画时,使用该方法
provideNoopAnimations()
provideServiceWorker()
provideServerRendering()
provideClientHydration()

在之前 NgModule 应用中,会根据在 @ngModule.providers 或者 @({providedIn: “…”}) 中配置依赖,根据配置创建模块注入器,在独立应用中,没有模块的概念,Angular 会创建环境注入器 environment injectors 。
一下场景会创建环境注入器:
@NgModule.providers ,通过 NgModule 引导的应用程序时
@({provideIn: “…”})
bootstrapApplication 独立应用启动时配置的 providers
Route 配置的 providers

除了在以上使用中 Angular 创建环境注入器,Angular 还支持 createEnvironmentInjector 创建。
路由

Angular 为支持独立应用,提供了 provideRouter 方法用于配置独立应用的路由。

export const routes: Routes = [ 
    { path: 'products', component: ProductListComponent }
];
bootstrapApplication(AppComponent, {
  providers: [
    provideRouter(routes),
    // importProvidersFrom(
    //   RouterModule.forRoot(routes)
    // ),
  ]
})
  .catch((err) => console.error(err));

惰性加载

之前,要惰性加载模块,Angular 提供了 loadChildren 方法,加载一组路由,使用如下:

const routes: Routes = [
  {
    path: 'items',
    loadChildren: () => import('./items/items.module').then(m => m.ItemsModule)
  }
];

在独立应用中,同样支持通过 loadChildren 惰性加载一组路由

// In the main application:
export const ROUTES: Route[] = [
  {path: 'admin', loadChildren: () => import('./admin/routes').then(mod => mod.ADMIN_ROUTES)},
  // ...
];
// In admin/routes.ts:
export const ADMIN_ROUTES: Route[] = [
  {path: 'home', component: AdminHomeComponent},
  {path: 'users', component: AdminUsersComponent},
  // ...
];

此外还支持 loadComponent 加载独立组件。

export const ROUTES: Route[] = [
  {path: 'admin', loadComponent: () => import('./admin/panel.component').then(mod => mod.AdminPanelComponent)},
];

简化使用:路由器会理解并使用 default 导出来的路由或者组件。

// In the main application:
export const ROUTES: Route[] = [
  {path: 'admin', loadChildren: () => import('./admin/routes')},
  // ...
];
// In admin/routes.ts:
export default [
  {path: 'home', component: AdminHomeComponent},
  {path: 'users', component: AdminUsersComponent},
  // ...
] as Route[];

工具

原有升级应用或类库升级使用独立组件
Angular (版本大于 15.2.0) 提供了完善的 Schematic 帮助原有应用升级转换为独立组件模式。

ng generate @angular/core:standalone

按照下面列出的顺序运行迁移:
运行 ng g @angular/core:standalone 并选择 “Convert all components, directives and pipes to standalone”
运行 ng g @angular/core:standalone 并选择 “Remove unnecessary NgModule classes”
运行 ng g @angular/core:standalone 并选择 “Bootstrap the project using standalone APIs”
运行任何静态分析(lint)和格式检查,修复任何故障,并提交结果

生成独立启动应用

ng new demo-standalone --standalone

生成独立组件

ng g c button --standalone

http://www.kler.cn/a/370821.html

相关文章:

  • 【无法下载github文件】虚拟机下ubuntu无法拉取github文件
  • 前端【2】html添加样式、CSS选择器
  • 80_Redis内存策略
  • 2023-2024 学年 广东省职业院校技能大赛(高职组)“信息安全管理与评估”赛题一
  • 【机器学习:二十二、机器学习项目开发的技巧】
  • 读书笔记~管理修炼-风险性决策:学会缩小风险阈值
  • Linux shell编程学习笔记87:blkid命令——获取块设备信息
  • 触觉智能Purple Pi OH鸿蒙开发板成功适配OpenHarmony5.0 Release,开启新征程!
  • 自动驾驶-传感器简述
  • D52【python 接口自动化学习】- python基础之模块与标准库
  • android 12 应用安装白名单
  • C++ 整型大数运算(大整数运算)项目
  • # Docker:技术架构的演进之路
  • Vue学习记录之二十一 Vue3中3种编程风格介绍
  • Vue.js/ElementUI-el-upload 与Spring Boot实现文件上传
  • 【Hadoop】hadoop的路径分不清?HDFS路径与本地文件系统路径的区别
  • 【计算机网络 - 基础问题】每日 3 题(五十四)
  • 使用ONNX Runtime对模型进行推理
  • python基于深度学习的音乐推荐方法研究系统
  • 一般公司流程图详情版
  • OSPF特殊区域及其他特性
  • centos下面的jdk17的安装配置
  • C#中的委托、匿名方法、Lambda、Action和Func
  • C++进阶-->多态(Polymorphism)
  • 大模型AI在教育领域有哪些创业机会?
  • 【云原生】云原生后端详解:架构与实践