Menguasai Pengurusan Keadaan Sudut menggunakan NgRx

WBOY
Lepaskan: 2024-09-10 16:30:36
asal
534 orang telah melayarinya

Pengurusan negeri dalam Angular memastikan data dikongsi secara konsisten dan cekap merentas semua bahagian aplikasi. Daripada setiap komponen mengurus datanya sendiri, stor pusat memegang keadaan.

Pemusatan ini memastikan bahawa apabila data berubah, semua komponen secara automatik mencerminkan keadaan dikemas kini, membawa kepada tingkah laku yang konsisten dan kod yang lebih mudah. Ia juga menjadikan apl lebih mudah untuk diselenggara dan skala, kerana aliran data diurus daripada satu sumber kebenaran.

Dalam artikel ini, kami akan meneroka cara melaksanakan pengurusan negeri dalam Angular menggunakan NgRx dengan membina aplikasi troli beli-belah yang mudah. Kami akan merangkumi konsep teras NgRx, seperti Stor, Tindakan, Pengurang, Pemilih, dan Kesan, serta menunjukkan cara bahagian ini sesuai untuk mengurus keadaan aplikasi anda dengan berkesan.

Nyatakan dalam Sudut merujuk kepada data yang apl anda perlukan untuk mengurus dan memaparkan, seperti kandungan troli beli-belah.

Mengapa anda memerlukan Pengurusan Negeri

1. Ketekalan: Ia memastikan data adalah seragam merentas semua komponen. Apabila data berubah di satu tempat, gedung pusat mengemas kini semua komponen yang berkaitan secara automatik, mengelakkan ketidakkonsistenan.

2. Aliran Data Dipermudahkan: Daripada menghantar data antara komponen secara manual, pengurusan negeri membenarkan mana-mana komponen mengakses atau mengemas kini data terus dari stor pusat, menjadikan aliran data apl lebih mudah untuk diurus dan difahami.

3. Penyelenggaraan dan Skalabilitas Lebih Mudah: Dengan memusatkan pengurusan data, pengurusan negeri mengurangkan pertindihan dan kerumitan kod. Ini menjadikan apl lebih mudah untuk diselenggara, nyahpepijat dan skala apabila ia berkembang.

4. Pengoptimuman Prestasi: Penyelesaian pengurusan negeri selalunya disertakan dengan alatan untuk mengoptimumkan prestasi, seperti mengemas kini secara terpilih hanya komponen yang perlu bertindak balas terhadap perubahan keadaan, dan bukannya memaparkan semula keseluruhan aplikasi.

Cara NgRx berfungsi

NgRx ialah perpustakaan pengurusan negeri untuk Angular yang membantu mengurus dan mengekalkan keadaan aplikasi anda dengan cara yang boleh diramal.

Mastering Angular State Management using NgRx

1. Komponen

Komponen ialah tempat pengguna berinteraksi dengan apl anda. Ia mungkin butang untuk menambah item pada troli beli-belah.

Komponen dan perkhidmatan diasingkan dan tidak berkomunikasi antara satu sama lain secara langsung, sebaliknya perkhidmatan digunakan dalam kesan sekali gus mewujudkan struktur aplikasi yang berbeza daripada apl Angular tradisional.

2. Tindakan

Tindakan menerangkan perkara yang berlaku dan mengandungi sebarang muatan (data) yang diperlukan.

3. Pengurang

Mengemas kini keadaan berdasarkan tindakan.

4. Kedai

Kedai ialah tempat berpusat yang menyimpan keseluruhan keadaan permohonan anda.

5. Pemilih

Mengekstrak data daripada stor untuk komponen.

6. Kesan

Kesan ialah tempat anda mengendalikan logik yang tidak terdapat dalam pengurang, seperti panggilan API.

7. Perkhidmatan

Perkhidmatan melaksanakan logik perniagaan sebenar atau panggilan API. Effects sering menggunakan perkhidmatan untuk melaksanakan tugas seperti mengambil data daripada pelayan.

Bila Menggunakan NgRx

Gunakan NgRx apabila kerumitan apl anda membenarkannya, tetapi untuk apl yang mudah, berpegang pada kaedah pengurusan keadaan yang lebih mudah. Ikatan perkhidmatan, isyarat dan @Input/@Output Angular antara komponen biasanya mencukupi untuk mengurus keadaan dalam aplikasi yang kurang kompleks.

Contoh: Membina Ciri Tambah ke Troli dengan NgRx

1.Buat Projek Sudut Baharu:

ng new shopping-cart
Salin selepas log masuk

2. Pasang NGRX dan Effects
Untuk memasang NGRX dan Effects, jalankan arahan berikut dalam terminal anda:

ng add @ngrx/store@latest

ng add @ngrx/effects
Salin selepas log masuk

3. Tentukan Model Produk
Di dalam direktori src/app, buat fail bernama product.model.ts

Tentukan antara muka Produk untuk mewakili struktur produk:

export interface Product {
    id: string;
    name: string;
    price: number;
    quantity: number;
}
Salin selepas log masuk

4. Sediakan Pengurusan Negeri
Langkah 1: Buat Folder keadaan di dalam direktori src/apl

Langkah 2: Tentukan Tindakan Troli

Buat cart.actions.ts dalam folder keadaan.

import { createActionGroup, emptyProps, props } from '@ngrx/store';
import { Product } from '../product.model';

export const CartActions = createActionGroup({
  source: 'Cart',
  events: {
    'Add Product': props<{ product: Product }>(),
    'Remove Product': props<{ productId: string }>(),
    'Update Quantity': props<{ productId: string; quantity: number }>(),
    'Load Products': emptyProps,
  },
});

export const CartApiActions = createActionGroup({
  source: 'Cart API',
  events: {
    'Load Products Success': props<{ products: Product[] }>(),
    'Load Products Failure': props<{ error: string }>(),
  },
});
Salin selepas log masuk

Langkah 3: Cipta Pengurangan

Buat cart.reducer.ts dalam folder keadaan.

import { createReducer, on } from '@ngrx/store';
import { Product } from '../product.model';
import { CartActions, CartApiActions } from './cart.actions';

// Initial state for products and cart
export const initialProductsState: ReadonlyArray<Product> = [];
export const initialCartState: ReadonlyArray<Product> = [];

// Reducer for products (fetched from API)
export const productsReducer = createReducer(
  initialProductsState,
  on(CartApiActions.loadProductsSuccess, (_state, { products }) => products)
);

// Reducer for cart (initially empty)
export const cartReducer = createReducer(
  initialCartState,
  on(CartActions.addProduct, (state, { product }) => {
    const existingProduct = state.find(p => p.id === product.id);
    if (existingProduct) {
      return state.map(p =>
        p.id === product.id ? { ...p, quantity: p.quantity + product.quantity } : p
      );
    }
    return [...state, product];
  }),
  on(CartActions.removeProduct, (state, { productId }) =>
    state.filter(p => p.id !== productId)
  ),
  on(CartActions.updateQuantity, (state, { productId, quantity }) =>
    state.map(p =>
      p.id === productId ? { ...p, quantity } : p
    )
  )
);
Salin selepas log masuk

Langkah 4: Buat Pemilih

Dalam folder state, buat cart.selectors.ts

import { createSelector, createFeatureSelector } from '@ngrx/store';
import { Product } from '../product.model';

export const selectProducts = createFeatureSelector<ReadonlyArray<Product>>('products');

export const selectCart = createFeatureSelector<ReadonlyArray<Product>>('cart');

export const selectCartTotal = createSelector(selectCart, (cart) =>
  cart.reduce((total, product) => total + product.price * product.quantity, 0)
);
Salin selepas log masuk

Step 5: Create Effects

Create a new file cart.effects.ts in the state folder that listens for the Load Products action, uses the service to fetch products, and dispatches either a success or failure action.

import { Injectable } from '@angular/core';
import { Actions, createEffect, ofType } from '@ngrx/effects';
import { ProductService } from '../product.service';
import { CartActions, CartApiActions } from './cart.actions';
import { catchError, map, mergeMap } from 'rxjs/operators';
import { of } from 'rxjs';

@Injectable()
export class CartEffects {
  loadProducts$ = createEffect(() =>
    this.actions$.pipe(
      ofType(CartActions.loadProducts),
      mergeMap(() =>
        this.productService.getProducts().pipe(
          map(products => CartApiActions.loadProductsSuccess({ products })),
          catchError(error => of(CartApiActions.loadProductsFailure({ error })))
        )
      )
    )
  );

  constructor(
    private actions$: Actions,
    private productService: ProductService
  ) {}
}
Salin selepas log masuk

5. Connect the State Management to Your App
In a file called app.config.ts, set up configurations for providing the store and effects to the application.

import { ApplicationConfig } from '@angular/core';
import { provideStore } from '@ngrx/store';
import { provideHttpClient } from '@angular/common/http';
import { cartReducer, productsReducer } from './state/cart.reducer';
import { provideEffects } from '@ngrx/effects';
import { CartEffects } from './state/cart.effects';

export const appConfig: ApplicationConfig = {
  providers: [
    provideStore({
      products: productsReducer, 
      cart: cartReducer 
    }),
    provideHttpClient(),
    provideEffects([CartEffects])
],
};
Salin selepas log masuk

6. Create a Service to Fetch Products
In the src/app directory create product.service.ts to implement the service to fetch products

import { Injectable } from '@angular/core';
import { Observable, of } from 'rxjs';
import { Product } from './product.model';

@Injectable({ providedIn: 'root' })
export class ProductService {
  getProducts(): Observable<Array<Product>> {
    return of([
      { id: '1', name: 'Product 1', price: 10, quantity: 1 },
      { id: '2', name: 'Product 2', price: 20, quantity: 1 },
    ]);
  }
}
Salin selepas log masuk

7. Create the Product List Component
Run the following command to generate the component: ng generate component product-list

This component displays the list of products and allows adding them to the cart.

Modify the product-list.component.ts file:

import { Component, OnInit } from '@angular/core';
import { CommonModule } from '@angular/common';
import { Store } from '@ngrx/store';
import { Observable } from 'rxjs';
import { Product } from '../product.model';
import { selectProducts } from '../state/cart.selectors';
import { CartActions } from '../state/cart.actions';

@Component({
  selector: 'app-product-list',
  standalone: true,
  templateUrl: './product-list.component.html',
  styleUrls: ['./product-list.component.css'],
  imports: [CommonModule],
})
export class ProductListComponent implements OnInit {
  products$!: Observable<ReadonlyArray<Product>>;

  constructor(private store: Store) {

  }

  ngOnInit(): void {
    this.store.dispatch(CartActions.loadProducts()); // Dispatch load products action
    this.products$ = this.store.select(selectProducts); // Select products from the store
  }

  onAddToCart(product: Product) {
    this.store.dispatch(CartActions.addProduct({ product }));
  }
}
Salin selepas log masuk

Modify the product-list.component.html file:

<div *ngIf="products$ | async as products">
  <div class="product-item" *ngFor="let product of products">
    <p>{{product.name}}</p>
    <span>{{product.price | currency}}</span>
    <button (click)="onAddToCart(product)" data-test="add-button">Add to Cart</button>
  </div>
</div>
Salin selepas log masuk

8. Create the Shopping Cart Component
Run the following command to generate the component: ng generate component shopping-cart

This component displays the products in the cart and allows updating the quantity or removing items from the cart.

Modify the shopping-cart.component.ts file:

import { Component, OnInit } from '@angular/core';
import { CommonModule } from '@angular/common';
import { Store } from '@ngrx/store';
import { Observable } from 'rxjs';
import { Product } from '../product.model';
import { selectCart, selectCartTotal } from '../state/cart.selectors';
import { CartActions } from '../state/cart.actions';

@Component({
  selector: 'app-shopping-cart',
  standalone: true,
  imports: [CommonModule],
  templateUrl: './shopping-cart.component.html',
  styleUrls: ['./shopping-cart.component.css'],
})
export class ShoppingCartComponent implements OnInit {
  cart$: Observable<ReadonlyArray<Product>>;
  cartTotal$: Observable<number>;

  constructor(private store: Store) {
    this.cart$ = this.store.select(selectCart);
    this.cartTotal$ = this.store.select(selectCartTotal);
  }

  ngOnInit(): void {}

  onRemoveFromCart(productId: string) {
    this.store.dispatch(CartActions.removeProduct({ productId }));
  }

  onQuantityChange(event: Event, productId: string) {
    const inputElement = event.target as HTMLInputElement;
    let quantity = parseInt(inputElement.value, 10);

    this.store.dispatch(CartActions.updateQuantity({ productId, quantity }));
  }
}
Salin selepas log masuk

Modify the shopping-cart.component.html file:

<div *ngIf="cart$ | async as cart">
  <div class="cart-item" *ngFor="let product of cart">
    <p>{{product.name}}</p><span>{{product.price | currency}}</span>
    <input type="number" [value]="product.quantity" (input)="onQuantityChange($event, product.id)" />
    <button (click)="onRemoveFromCart(product.id)" data-test="remove-button">Remove</button>
  </div>
  <div class="total">
    Total: {{cartTotal$ | async | currency}}
  </div>
</div>
Salin selepas log masuk

Modify the shopping-cart.component.css file:

.cart-item {
  display: flex;
  justify-content: space-between;
  align-items: center;
  margin-bottom: 10px;
}

.cart-item p {
  margin: 0;
  font-size: 16px;
}

.cart-item input {
  width: 50px;
  text-align: center;
}

.total {
  font-weight: bold;
  margin-top: 20px;
}
Salin selepas log masuk

9. Put Everything Together in the App Component
This component will display the product list and the shopping cart

Modify the app.component.ts file:

import { Component } from '@angular/core';
import { CommonModule } from '@angular/common';
import { ProductListComponent } from './product-list/product-list.component';
import { ShoppingCartComponent } from './shopping-cart/shopping-cart.component';
import { NgIf } from '@angular/common';

@Component({
  selector: 'app-root',
  standalone: true,
  templateUrl: './app.component.html',
  imports: [CommonModule, ProductListComponent, ShoppingCartComponent, NgIf],
})
export class AppComponent {}
Salin selepas log masuk

Modify the app.component.html file:

<!-- app.component.html -->
<h2>Products</h2>
<app-product-list></app-product-list>

<h2>Shopping Cart</h2>
<app-shopping-cart></app-shopping-cart>
Salin selepas log masuk

10. Running the Application
Finally, run your application using ng serve.

Now, you can add products to your cart, remove them, or update their quantities.

Conclusion

In this article, we built a simple shopping cart application to demonstrate the core concepts of NgRx, such as the Store, Actions, Reducers, Selectors, and Effects. This example serves as a foundation for understanding how NgRx works and how it can be applied to more complex applications.

As your Angular projects grow in complexity, leveraging NgRx for state management will help you maintain consistency across your application, reduce the likelihood of bugs, and make your codebase easier to maintain.

To get the code for the above project, click the link below:
https://github.com/anthony-kigotho/shopping-cart

Atas ialah kandungan terperinci Menguasai Pengurusan Keadaan Sudut menggunakan NgRx. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:dev.to
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan