Продолжение урока по передаче данных между компонентами в Angular

Продолжение урока по передаче данных между компонентами в Angular

Angular, как один из самых популярных фреймворков для разработки веб-приложений, предоставляет разработчикам множество инструментов для эффективного управления данными и их передачи между компонентами. В этой статье будут рассмотрены продвинутые техники и концепции, связанные с обменом информацией в Angular-приложениях.

Повторение основ: @Input и @Output

Прежде чем погрузиться в более сложные концепции, стоит вспомнить базовые механизмы передачи данных в Angular:

  • @Input — декоратор для передачи данных от родительского компонента к дочернему
  • @Output — декоратор для отправки событий от дочернего компонента к родительскому

Эти механизмы являются фундаментом для построения взаимодействия между компонентами, но для создания сложных приложений требуются более продвинутые техники.

Использование сервисов для обмена данными

Сервисы в Angular предоставляют мощный инструмент для организации обмена данными между компонентами, особенно когда речь идет о не связанных напрямую элементах приложения.

Создание общего сервиса

Для начала необходимо создать сервис, который будет служить посредником для обмена данными:

 import { Injectable } from '@angular/core'; import { BehaviorSubject } from 'rxjs'; @Injectable({ providedIn: 'root' }) export class DataSharingService { private messageSource = new BehaviorSubject("Initial message"); currentMessage = this.messageSource.asObservable(); constructor() { } changeMessage(message: string) { this.messageSource.next(message); } } 

В этом примере используется BehaviorSubject из библиотеки RxJS, который позволяет хранить текущее значение и отправлять его новым подписчикам.

Использование сервиса в компонентах

Теперь любой компонент может использовать этот сервис для отправки или получения данных:

 import { Component, OnInit } from '@angular/core'; import { DataSharingService } from './data-sharing.service'; @Component({ selector: 'app-sender', template: `  ` }) export class SenderComponent { constructor(private dataSharingService: DataSharingService) { } newMessage() { this.dataSharingService.changeMessage("Hello from Sender Component") } } @Component({ selector: 'app-receiver', template: ` 

{{message}}

` }) export class ReceiverComponent implements OnInit { message: string; constructor(private dataSharingService: DataSharingService) { } ngOnInit() { this.dataSharingService.currentMessage.subscribe(message => this.message = message) } }

Этот подход особенно полезен, когда необходимо организовать обмен данными между компонентами, которые не имеют прямой связи родитель-потомок.

Использование NgRx для управления состоянием приложения

Для крупных приложений с сложной логикой и множеством взаимодействующих компонентов часто используется библиотека NgRx, реализующая концепцию Redux в Angular.

Основные концепции NgRx

  • Store — центральное хранилище состояния приложения
  • Actions — описывают изменения в состоянии
  • Reducers — чистые функции, которые принимают текущее состояние и action, и возвращают новое состояние
  • Effects — обрабатывают побочные эффекты, такие как запросы к API
  • Selectors — извлекают определенные части состояния

Пример использования NgRx

Рассмотрим пример реализации простого счетчика с использованием NgRx:

 // actions.ts import { createAction } from '@ngrx/store'; export const increment = createAction('[Counter] Increment'); export const decrement = createAction('[Counter] Decrement'); export const reset = createAction('[Counter] Reset'); // reducers.ts import { createReducer, on } from '@ngrx/store'; import * as CounterActions from './counter.actions'; export const initialState = 0; export const counterReducer = createReducer( initialState, on(CounterActions.increment, state => state + 1), on(CounterActions.decrement, state => state - 1), on(CounterActions.reset, state => 0) ); // app.component.ts import { Component } from '@angular/core'; import { Store } from '@ngrx/store'; import { Observable } from 'rxjs'; import * as CounterActions from './counter.actions'; @Component({ selector: 'app-root', template: `    

Current Count: {{ count$ | async }}

` }) export class AppComponent { count$: Observable; constructor(private store: Store<{ count: number }>) { this.count$ = store.select('count'); } increment() { this.store.dispatch(CounterActions.increment()); } decrement() { this.store.dispatch(CounterActions.decrement()); } reset() { this.store.dispatch(CounterActions.reset()); } }

Этот подход обеспечивает предсказуемое управление состоянием и упрощает отладку приложения.

Использование RxJS для реактивного программирования

RxJS (Reactive Extensions for JavaScript) является мощной библиотекой для работы с асинхронными операциями и потоками данных. Angular тесно интегрирован с RxJS, что позволяет эффективно управлять потоками данных в приложении.

Основные концепции RxJS

  • Observable — представляет поток данных или событий
  • Observer — подписывается на Observable и реагирует на его события
  • Operators — функции для преобразования, фильтрации и комбинирования Observables
  • Subject — особый тип Observable, который может как излучать события, так и подписываться на них

Пример использования RxJS в Angular

Рассмотрим пример реализации поиска с автодополнением:

 import { Component, OnInit } from '@angular/core'; import { FormControl } from '@angular/forms'; import { Observable } from 'rxjs'; import { debounceTime, distinctUntilChanged, switchMap } from 'rxjs/operators'; import { SearchService } from './search.service'; @Component({ selector: 'app-search', template: `  
  • {{ result }}
` }) export class SearchComponent implements OnInit { searchControl = new FormControl(); results$: Observable; constructor(private searchService: SearchService) {} ngOnInit() { this.results$ = this.searchControl.valueChanges.pipe( debounceTime(300), distinctUntilChanged(), switchMap(term => this.searchService.search(term)) ); } }

В этом примере используются операторы RxJS для создания эффективного механизма поиска:

  • debounceTime — задерживает запросы на указанное время, чтобы избежать лишних запросов при быстром вводе
  • distinctUntilChanged — предотвращает повторные запросы с тем же значением
  • switchMap — отменяет предыдущий запрос, если поступил новый, и переключается на него

Передача данных через роутинг

Angular Router предоставляет возможность передавать данные между компонентами через параметры маршрута или состояние навигации.

Передача параметров через URL

Это удобный способ передачи простых данных, которые могут быть отображены в URL:

 // app-routing.module.ts const routes: Routes = [ { path: 'user/:id', component: UserComponent } ]; // sender.component.ts import { Router } from '@angular/router'; @Component({...}) export class SenderComponent { constructor(private router: Router) {} navigateToUser(userId: string) { this.router.navigate(['/user', userId]); } } // user.component.ts import { ActivatedRoute } from '@angular/router'; @Component({...}) export class UserComponent implements OnInit { userId: string; constructor(private route: ActivatedRoute) {} ngOnInit() { this.route.params.subscribe(params => { this.userId = params['id']; }); } } 

Передача данных через состояние навигации

Для передачи более сложных объектов, которые не должны отображаться в URL, можно использовать состояние навигации:

 // sender.component.ts import { Router } from '@angular/router'; @Component({...}) export class SenderComponent { constructor(private router: Router) {} navigateWithState() { const navigationExtras: NavigationExtras = { state: { example: 'This is a complex object' } }; this.router.navigate(['/receiver'], navigationExtras); } } // receiver.component.ts import { Component } from '@angular/core'; import { Router } from '@angular/router'; @Component({...}) export class ReceiverComponent { state: any; constructor(private router: Router) { const navigation = this.router.getCurrentNavigation(); this.state = navigation?.extras.state; } } 

Использование ViewChild и ContentChild

Декораторы ViewChild и ContentChild позволяют получить доступ к дочерним элементам и компонентам непосредственно из кода родительского компонента.

ViewChild

ViewChild используется для доступа к элементам, которые находятся в шаблоне компонента:

 import { Component, ViewChild, AfterViewInit, ElementRef } from '@angular/core'; @Component({ selector: 'app-parent', template: `   ` }) export class ParentComponent implements AfterViewInit { @ViewChild('inputElement') inputElement: ElementRef; @ViewChild('childComponent') childComponent: ChildComponent; ngAfterViewInit() { console.log(this.inputElement.nativeElement.value); this.childComponent.someMethod(); } } 

ContentChild

ContentChild используется для доступа к элементам, которые проецируются в компонент через ng-content:

 // parent.component.ts @Component({ selector: 'app-parent', template: `  

This is projected content

` }) export class ParentComponent {} // child.component.ts import { Component, ContentChild, AfterContentInit, ElementRef } from '@angular/core'; @Component({ selector: 'app-child', template: `` }) export class ChildComponent implements AfterContentInit { @ContentChild('projectedContent') projectedContent: ElementRef; ngAfterContentInit() { console.log(this.projectedContent.nativeElement.textContent); } }

Использование HostBinding и HostListener

Эти декораторы позволяют взаимодействовать с DOM-элементом, в котором размещен компонент.

HostBinding

HostBinding позволяет установить свойства элемента-хоста компонента:

 import { Component, HostBinding } from '@angular/core'; @Component({ selector: 'app-example', template: `

This is an example component

` }) export class ExampleComponent { @HostBinding('class.active') isActive = false; @HostBinding('style.color') color = 'red'; toggleActive() { this.isActive = !this.isActive; } }

HostListener

HostListener позволяет реагировать на события, происходящие на элементе-хосте:

 import { Component, HostListener } from '@angular/core'; @Component({ selector: 'app-example', template: `

Click anywhere on this component

` }) export class ExampleComponent { @HostListener('click', ['$event']) onClick(event: MouseEvent) { console.log('Component clicked', event); } @HostListener('window:resize, ['$event']) onResize(event: Event) { console.log('Window resized', event); } }

Использование NgTemplateOutlet и NgComponentOutlet

Эти директивы позволяют динамически вставлять шаблоны и компоненты в представление.

NgTemplateOutlet

NgTemplateOutlet используется для динамической вставки шаблонов:

 import { Component } from '@angular/core'; @Component({ selector: 'app-example', template: `   

Hello, {{ name }}!

` }) export class ExampleComponent { }

NgComponentOutlet

NgComponentOutlet позволяет динамически создавать компоненты:

 import { Component } from '@angular/core'; @Component({ selector: 'app-dynamic', template: `

I am a dynamic component

` }) export class DynamicComponent { } @Component({ selector: 'app-example', template: ` ` }) export class ExampleComponent { componentToLoad = DynamicComponent; }

Использование Change Detection Strategy

Стратегия обнаружения изменений в Angular определяет, как фреймворк проверяет компоненты на наличие изменений.

Default стратегия

По умолчанию Angular использует стратегию ChangeDetectionStrategy.Default, которая проверяет все компоненты при каждом событии.

OnPush стратегия

Стратегия ChangeDetectionStrategy.OnPush позволяет оптимизировать производительность, проверяя компонент только при изменении его входных данных:

 import { Component, ChangeDetectionStrategy, Input } from '@angular/core'; @Component({ selector: 'app-child', template: `

{{ data }}

`, changeDetection: ChangeDetectionStrategy.OnPush }) export class ChildComponent { @Input() data: string; } @Component({ selector: 'app-parent', template: `` }) export class ParentComponent { parentData = 'Initial data'; updateData() { this.parentData = 'Updated data'; } }

Использование зон в Angular

Зоны в Angular — это механизм, который помогает отслеживать асинхронные операции и автоматически запускать обнаружение изменений.

NgZone

NgZone — это сервис, который позволяет выполнять код внутри или вне Angular зоны:

 import { Component, NgZone } from '@angular/core'; @Component({ selector: 'app-example', template: `

{{ message }}

` }) export class ExampleComponent { message = 'Initial message'; constructor(private ngZone: NgZone) { } runOutsideAngular() { this.ngZone.runOutsideAngular(() => { // Этот код выполняется вне Angular зоны setTimeout(() => { this.ngZone.run(() => { // Этот код выполняется внутри Angular зоны this.message = 'Updated message'; }); }, 1000); }); } }

Использование Dependency Injection в Angular

Dependency Injection (DI) — это паттерн проектирования, который широко используется в Angular для обеспечения слабой связанности компонентов и сервисов.

Провайдеры

Провайдеры определяют, как создавать и предоставлять зависимости:

 import { Injectable } from '@angular/core'; @Injectable({ providedIn: 'root' }) export class LoggerService { log(message: string) { console.log(message); } } @Component({ selector: 'app-example', template: `

Check console for logs

`, providers: [LoggerService] }) export class ExampleComponent { constructor(private logger: LoggerService) { this.logger.log('ExampleComponent created'); } }

Иерархия инжекторов

Angular использует иерархическую систему инжекторов, которая соответствует иерархии компонентов:

 import { Component, Injectable } from '@angular/core'; @Injectable() class ParentService { } @Injectable() class ChildService { } @Component({ selector: 'app-parent', template: ` 

Parent component

`, providers: [ParentService] }) class ParentComponent { constructor(private parentService: ParentService) { } } @Component({ selector: 'app-child', template: `

Child component

`, providers: [ChildService] }) class ChildComponent { constructor( private parentService: ParentService, private childService: ChildService ) { } }

Использование Interceptors для HTTP-запросов

Interceptors позволяют перехватывать и модифицировать HTTP-запросы и ответы.

Создание Interceptor

 import { Injectable } from '@angular/core'; import { HttpInterceptor, HttpRequest, HttpHandler, HttpEvent } from '@angular/common/http'; import { Observable } from 'rxjs'; @Injectable() export class AuthInterceptor implements HttpInterceptor { intercept(req: HttpRequest, next: HttpHandler): Observable> { const authToken = localStorage.getItem('authToken'); const authReq = req.clone({ headers: req.headers.set('Authorization', `Bearer ${authToken}`) }); return next.handle(authReq); } } 

Регистрация Interceptor

 import { NgModule } from '@angular/core'; import { HTTP_INTERCEPTORS } from '@angular/common/http'; import { AuthInterceptor } from './auth.interceptor'; @NgModule({ providers: [ { provide: HTTP_INTERCEPTORS, useClass: AuthInterceptor, multi: true } ] }) export class AppModule { } 

Использование Directives для расширения функциональности элементов

Директивы позволяют изменять поведение и внешний вид элементов DOM.

Атрибутивные директивы

 import { Directive, ElementRef, Input, OnInit } from '@angular/core'; @Directive({ selector: '[appHighlight]' }) export class HighlightDirective implements OnInit { @Input('appHighlight') highlightColor: string; constructor(private el: ElementRef) { } ngOnInit() { this.el.nativeElement.style.backgroundColor = this.highlightColor || 'yellow'; } } 

Структурные директивы

 import { Directive, Input, TemplateRef, ViewContainerRef } from '@angular/core'; @Directive({ selector: '[appUnless]' }) export class UnlessDirective { private hasView = false; constructor( private templateRef: TemplateRef, private viewContainer: ViewContainerRef ) { } @Input() set appUnless(condition: boolean) { if (!condition && !this.hasView) { this.viewContainer.createEmbeddedView(this.templateRef); this.hasView = true; } else if (condition && this.hasView) { this.viewContainer.clear(); this.hasView = false; } } } 

Использование Angular Animations

Angular предоставляет мощный модуль анимаций для создания плавных переходов и эффектов.

Определение анимации

 import { trigger, state, style, animate, transition } from '@angular/animations'; @Component({ selector: 'app-example', template: ` 

Animation example

`, animations: [ trigger('openClose', [ state('open', style({ height: '200px', opacity: 1, backgroundColor: 'yellow' })), state('closed', style({ height: '100px', opacity: 0.5, backgroundColor: 'green' })), transition('open => closed', [ animate('1s') ]), transition('closed => open', [ animate('0.5s') ]), ]), ], }) export class ExampleComponent { isOpen = true; toggle() { this.isOpen = !this.isOpen; } }

Оптимизация производительности Angular приложений

Оптимизация является ключевым аспектом разработки Angular приложений, особенно при работе с большими объемами данных или сложными интерфейсами.

Lazy Loading

Lazy Loading позволяет загружать модули по требованию, что уменьшает начальное время загрузки приложения:

 // app-routing.module.ts const routes: Routes = [ { path: 'admin', loadChildren: () => import('./admin/admin.module').then(m => m.AdminModule) } ]; // admin-routing.module.ts const routes: Routes = [ { path: '', component: AdminDashboardComponent } ]; @NgModule({ imports: [RouterModule.forChild(routes)], exports: [RouterModule] }) export class AdminRoutingModule { } 

TrackBy функция

Использование TrackBy в ngFor может значительно улучшить производительность при работе с большими списками:

 @Component({ selector: 'app-example', template: ` 
  • {{ item.name }}
` }) export class ExampleComponent { items = [ { id: 1, name: 'Item 1' }, { id: 2, name: 'Item 2' }, { id: 3, name: 'Item 3' } ]; trackByFn(index, item) { return item.id; } }

Виртуальная прокрутка

Для эффективной работы с очень большими списками можно использовать виртуальную прокрутку:

 import { Component } from '@angular/core'; import { ScrollingModule } from '@angular/cdk/scrolling'; @Component({ selector: 'app-example', template: `  
{{ item }}
`, styles: [` .example-viewport { height: 200px; width: 200px; border: 1px solid black; } .example-item { height: 50px; } `] }) export class ExampleComponent { items = Array.from({length: 100000}).map((_, i) => `Item #${i}`); }

Работа с формами в Angular

Angular предоставляет два подхода к работе с формами: Template-driven и Reactive forms.

Template-driven формы

Этот подход основан на директивах в шаблоне и подходит для простых форм:

 import { Component } from '@angular/core'; @Component({ selector: 'app-example', template: ` 
` }) export class ExampleComponent { onSubmit(form: NgForm) { if (form.valid) { console.log(form.value); } } }

Reactive формы

Reactive формы предоставляют более гибкий подход и лучше подходят для сложных форм:

 import { Component, OnInit } from '@angular/core'; import { FormBuilder, FormGroup, Validators } from '@angular/forms'; @Component({ selector: 'app-example', template: ` 
` }) export class ExampleComponent implements OnInit { form: FormGroup; constructor(private fb: FormBuilder) { } ngOnInit() { this.form = this.fb.group({ firstName: ['', Validators.required], lastName: ['', Validators.required] }); } onSubmit() { if (this.form.valid) { console.log(this.form.value); } } }

Работа с асинхронными операциями в Angular

Управление асинхронными операциями является важной частью разработки веб-приложений. Angular предоставляет несколько инструментов для работы с асинхронностью.

Использование Promises

Promises — это объекты, представляющие результат асинхронной операции:

 import { Component } from '@angular/core'; @Component({ selector: 'app-example', template: `

{{ data }}

` }) export class ExampleComponent { data: string; constructor() { this.fetchData().then(result => this.data = result); } fetchData(): Promise { return new Promise((resolve) => { setTimeout(() => resolve('Data fetched'), 2000); }); } }

Использование RxJS Observables

Observables предоставляют более мощные инструменты для работы с асинхронными потоками данных:

 import { Component } from '@angular/core'; import { Observable, of } from 'rxjs'; import { delay } from 'rxjs/operators'; @Component({ selector: 'app-example', template: `

{{ data | async }}

` }) export class ExampleComponent { data: Observable; constructor() { this.data = this.fetchData(); } fetchData(): Observable { return of('Data fetched').pipe(delay(2000)); } }

Работа с HTTP в Angular

Angular предоставляет HttpClient для выполнения HTTP-запросов.

Выполнение GET-запроса

 import { Component, OnInit } from '@angular/core'; import { HttpClient } from '@angular/common/http'; import { Observable } from 'rxjs'; @Component({ selector: 'app-example', template: ` 
  • {{ user.name }}
` }) export class ExampleComponent implements OnInit { users: Observable; constructor(private http: HttpClient) { } ngOnInit() { this.users = this.http.get('https://api.example.com/users'); } }

Выполнение POST-запроса

 import { Component } from '@angular/core'; import { HttpClient } from '@angular/common/http'; @Component({ selector: 'app-example', template: ` 
` }) export class ExampleComponent { user = { name: '' }; constructor(private http: HttpClient) { } onSubmit() { this.http.post('https://api.example.com/users', this.user) .subscribe( response => console.log('User created', response), error => console.error('Error creating user', error) ); } }

Тестирование в Angular

Тестирование является важной частью разработки Angular приложений. Angular предоставляет инструменты для модульного и интеграционного тестирования.

Модульное тестирование компонента

 import { ComponentFixture, TestBed } from '@angular/core/testing'; import { ExampleComponent } from './example.component'; describe('ExampleComponent', () => { let component: ExampleComponent; let fixture: ComponentFixture; beforeEach(async () => { await TestBed.configureTestingModule({ declarations: [ ExampleComponent ] }) .compileComponents(); }); beforeEach(() => { fixture = TestBed.createComponent(ExampleComponent); component = fixture.componentInstance; fixture.detectChanges(); }); it('should create', () => { expect(component).toBeTruthy(); }); it('should have a title', () => { expect(component.title).toEqual('Example Component'); }); }); 

Тестирование сервиса

 import { TestBed } from '@angular/core/testing'; import { HttpClientTestingModule, HttpTestingController } from '@angular/common/http/testing'; import { UserService } from './user.service'; describe('UserService', () => { let service: UserService; let httpMock: HttpTestingController; beforeEach(() => { TestBed.configureTestingModule({ imports: [ HttpClientTestingModule ], providers: [ UserService ] }); service = TestBed.inject(UserService); httpMock = TestBed.inject(HttpTestingController); }); afterEach(() => { httpMock.verify(); }); it('should retrieve users', () => { const dummyUsers = [ { id: 1, name: 'John' }, { id: 2, name: 'Jane' } ]; service.getUsers().subscribe(users => { expect(users.length).toBe(2); expect(users).toEqual(dummyUsers); }); const req = httpMock.expectOne('api/users'); expect(req.request.method).toBe('GET'); req.flush(dummyUsers); }); }); 

Безопасность в Angular приложениях

Безопасность является критически важным аспектом разработки веб-приложений. Angular предоставляет несколько встроенных механизмов для обеспечения безопасности.

Защита от XSS-атак

Angular автоматически экранирует значения в шаблонах, чтобы предотвратить XSS-атаки:

 @Component({ selector: 'app-example', template: `
{{ userInput }}
` }) export class ExampleComponent { userInput = ''; }

В этом случае Angular отобразит текст как есть, не выполняя скрипт.

CSRF-защита

Angular автоматически добавляет XSRF-TOKEN к HTTP-запросам для защиты от CSRF-атак. Для этого необходимо настроить сервер для отправки XSRF-TOKEN в cookie.

Безопасные ссылки

Angular предоставляет механизм безопасных ссылок для предотвращения открытия вредоносных ссылок:

 @Component({ selector: 'app-example', template: `Safe Link` }) export class ExampleComponent { constructor(private sanitizer: DomSanitizer) {} get safeUrl() { return this.sanitizer.bypassSecurityTrustUrl('javascript:alert("XSS")'); } } 

Интернационализация в Angular

Интернационализация (i18n) позволяет адаптировать приложение для различных языков и регионов.

Использование i18n-атрибутов

 @Component({ selector: 'app-example', template: ` 

Welcome to our app!

Please select your language

` }) export class ExampleComponent { }

Создание файлов переводов

Для каждого языка создается файл перевода, например, messages.fr.xlf:

      Welcome to our app! Bienvenue dans notre application!     

Оптимизация Angular приложения для продакшена

Перед развертыванием Angular приложения в продакшен среду, необходимо выполнить ряд оптимизаций.

Сборка приложения для продакшена

 ng build --prod 

Эта команда выполняет следующие оптимизации:

  • Минификация кода
  • Удаление неиспользуемого кода (tree-shaking)
  • Сжатие ресурсов
  • Оптимизация загрузки модулей

Использование Service Workers

Service Workers позволяют создавать приложения, работающие офлайн:

 ng add @angular/pwa 

Эта команда добавляет необходимые файлы и конфигурации для использования Service Worker в приложении.

Заключение

В этой статье были рассмотрены продвинутые техники передачи данных между компонентами в Angular, а также множество других важных аспектов разработки Angular приложений. От базовых механизмов, таких как @Input и @Output, до сложных концепций, таких как NgRx и RxJS, Angular предоставляет разработчикам мощные инструменты для создания сложных и масштабируемых веб-приложений.

Важно помнить, что эффективное использование этих техник требует глубокого понимания архитектуры Angular и принципов реактивного программирования. Разработчикам рекомендуется постоянно практиковаться и изучать новые возможности фреймворка, чтобы создавать более эффективные и производительные приложения.

По мере развития Angular и веб-технологий в целом, появляются новые инструменты и подходы к разработке. Поэтому критически важно следить за обновлениями фреймворка и лучшими практиками в сообществе Angular разработчиков.

В конечном счете, выбор конкретных техник и подходов зависит от требований проекта, его масштаба и специфики. Грамотное применение рассмотренных в этой статье концепций позволит создавать гибкие, масштабируемые и легко поддерживаемые Angular приложения.

Читайте также  LinkedIn отмечает увеличение спроса на специалистов по маркетингу
Советы по созданию сайтов