Tuesday, October 25, 2022

Angular tutorial Part 3 - Learn Angular Services and Dependency injection Forms

Angular

Services and Dependency injection
Forms

Dependency Injection

     Dependencies are services or objects that a class needs to perform its function.

     Dependency injection, or DI, is a design pattern in which a class requests dependencies from external sources rather than creating them.

     Angular's DI framework provides dependencies to a class upon instantiation. Use Angular DI to increase flexibility and modularity in your applications.

     We shall see dependency injection using services

Services

     A service is a class with a focusedpurpose. We often create a service to implement functionality that is independent from any particular component.

     Services are used to share the data and logic across components or to encapsulate external interactions such as data access.

     To create a service class there is no need to do anything angulars pecific,noMetadata,naming convention requirement or some functional interface that needs to be implemented.They'rejust plain old classes that you create to modularize reusable code.

     Services are not only beneficial for modularity and single responsibility,but it also makes the code more testable.

     Dataservice is a class that will handle getting and setting data from your data store

Creating injectable service

     Create an injectable service using the command

     Ng generate service myservice in the angular app folder

     It will create the service shell code

     import { Injectable } from '@angular/core';

      @Injectable({ providedIn: 'root', })

     export class myService { constructor() { } }

     The @Injectable() decorator specifies that Angular can use this class in the DI system. Root gives application wide access to service.

 

       We will add business logic to get todays date to the service.

       showTodayDate() {

           let ndate = new Date();

           return ndate;

        }

       To add this service to component  inject a dependency in a component's constructor(), supply a

        constructor argument with the dependency type.

       1. import { MyserviceService } from './myservice.service';

       2.   constructor(private myservice: MyserviceService ){}

 

         

 

Service

       You can also inject service in the main module

       So in the app-module.ts we can inject the service

       import { MyserviceService } from './myservice.service';

        providers: [MyserviceService],

       Component provider and Module provider are different, If its injected in the module provider,it’s available to the whole app,it can also be specifically be injected into a component.

       @Component({ /* . . . */ providers: [UserService] })

 

 

Forms – two ways to create forms in angular

     Reactive forms provide direct, explicit access to the underlying forms object model. Compared to template-driven forms, they are more robust: they're more scalable, reusable, and testable. If forms are a key part of your application, or you're already using reactive patterns for building your application, use reactive forms.

     Template-driven forms rely on directives in the template to create and manipulate the underlying object model. They are useful for adding a simple form to an app, such as an email list signup form. They're straightforward to add to an app, but they don't scale as well as reactive forms. If you have very basic form requirements and logic that can be managed solely in the template, template-driven forms could be a good fit.

 

 

Core classes

     Both reactive and template-driven forms track value changes between the form input elements that users interact with and the form data in  your component model.

     Both reactive and template-driven forms are built on the following base classes.

 

     FormControl tracks the value and validation status of an individual form control.

 

     FormGroup tracks the same values and status for a collection of form controls.

 

     FormArray tracks the same values and status for an array of form controls.

 

     ControlValueAccessor creates a bridge between Angular FormControl instances and built-in DOM elements.

Building a reactive form

     With reactive forms, you define the form model directly in the component class. The [formControl] directive links the explicitly created FormControl instance to a specific  form element in the view, using an internal value accessor.

     import { Component } from '@angular/core';

     import { FormControl } from '@angular/forms';

     @Component({ selector: 'app-reactive-favorite-color', template: ` Favorite Color: <input type="text" [formControl]="favoriteColorControl"> ` })

     export class FavoriteColorComponent { favoriteColorControl = new FormControl(''); } //creates new form control instance

 

Direct access to forms model in a reactive form

Template uses ngmodel for binding and not form control instance

Building template form

     In template-driven forms, the form model is implicit, rather than explicit. The directive NgModel creates and  manages a FormControl instance for a given form element.

     import { Component } from '@angular/core';

     @Component({ selector: 'app-template-favorite-color', template: ` Favorite Color: <input type="text" [(ngModel)]="favoriteColor"> ` }) export class FavoriteColorComponent {

      favoriteColor = ''; }

     In a template-driven form the source of truth is the template. You do not have direct programmatic access to the FormControl

Data flowin reactive forms

     The user types a value into the input element, in this case the favorite color Blue.

     The form input element emits an "input" event with the latest value.

     The control value accessor listening for events on the form input element immediately relays the new value to the FormControl instance.

     The FormControl instance emits the new value through the valueChanges observable.

     Any subscribers to the valueChanges observable receive the new value.

Data flow in reactive forms

 

     The user calls the favoriteColorControl.setValue() method, which updates the FormControl value.

     The FormControl instance emits the new value through the valueChanges observable.

     Any subscribers to the valueChanges observable receive the new value.

     The control value accessor on the form input element updates the element with the new value.

Data flow in template forms

 

     The user types Blue into the input element.

     The input element emits an "input" event with the value Blue.

     The control value accessor attached to the input triggers the setValue() method on the FormControl instance.

     The FormControl instance emits the new value through the valueChanges observable.

     Any subscribers to the valueChanges observable receive the new value.

     The control value accessor also calls the NgModel.viewToModelUpdate() method which emits an ngModelChange event.

     Because the component template uses two-way data binding for the favoriteColor property, the favoriteColor property in the component is updated to the value emitted by the ngModelChange event (Blue).

Data flow in template forms

     The model-to-view diagram shows how data flows from model to view when the favoriteColor changes from Blue to Red, through the following steps

 

     The favoriteColor value is updated in the component.

     Change detection begins.

     During change detection, the ngOnChanges lifecycle hook is called on the NgModel directive instance because the value of one of its inputs has changed.

     The ngOnChanges() method queues an async task to set the value for the internal FormControl instance.

     Change detection completes.

     On the next tick, the task to set the FormControl instance value is executed.

     The FormControl instance emits the latest value through the valueChanges observable.

     Any subscribers to the valueChanges observable receive the new value.

     The control value accessor updates the form input element in the view with the latest favoriteColor value.

 

Form Validations

     Reactive forms define custom validators as functions that receive a control to validate.

     Template-driven forms are tied to template directives, and must provide custom validator directives that wrap validation functions.

 

Reactive forms with form controls

     There are three steps to using form controls.

 

     Register the reactive forms module in your application. This module declares the reactive-form directives that you need to use reactive forms.

     Generate a new FormControl instance and save it in the component.

     Register the FormControl in the template.

     import { ReactiveFormsModule } from '@angular/forms'; @NgModule({

      imports: [ // other imports ... ReactiveFormsModule ], })

Adding form control to component

     import { FormControl } from '@angular/forms';

     name = new FormControl('');//create a new instance of form control

     After you create the control in the component class, you must associate it with a form control element in the template.

     Bind the variable using form control

     <label for="name">Name: </label> <input id="name" type="text" [formControl]="name">

     Display the value

     <p>Value: {{ name.value }}</p>

 

     updateName() { this.name.setValue('Nancy'); }

     Updates the value from a component method

     <button (click)="updateName()">Update Name</button>

     Method binding to update the variable

Form Group controls

     Forms typically contain several related controls. Reactive forms provide two ways of grouping multiple related controls into a single input form.

     A form group defines a form with a fixed set of controls that you can manage together. Form group basics are discussed in this section. You can also nest form groups to create more complex forms.

     A form array defines a dynamic form, where you can add and remove controls at run time. You can also nest form arrays to create more complex forms. For more about this option, see Creating dynamic forms.

 

Building a form group control

     import { FormGroup, FormControl } from '@angular/forms‘;

     To add a form group to this component, take the following steps.

 

     Create a FormGroup instance.

     Associate the FormGroup model and view.

     Save the form data.

Creating a form group

     import { Component } from '@angular/core';

     import { FormGroup, FormControl } from '@angular/forms';//import the formgroup

 

     @Component({

       selector: 'app-profile-editor',

       templateUrl: './profile-editor.component.html',

       styleUrls: ['./profile-editor.component.css']

     })

     export class ProfileEditorComponent {

       profileForm = new FormGroup({  //instantiate new form group

         firstName: new FormControl(''), //within from group instantiate new form control

         lastName: new FormControl(''),

       });

     }

Associate the FormGroup model and view

     Build the form ,associate the formgroup and submit the data.

 

     <form [formGroup]="profileForm" (ngSubmit)="onSubmit()">

       <label for="first-name">First Name: </label>

       <input id="first-name" type="text" formControlName="firstName">

 

       <label for="last-name">Last Name: </label>

       <input id="last-name" type="text" formControlName="lastName">

 

     </form>

     In Submit method access the form group values

     onSubmit()

     { // TODO: Use EventEmitter with form value

     console.warn(this.profileForm.value);

      }

There are two ways to update the model value:

       Use the setValue() method to set a new value for an individual

       control. The setValue() method strictly adheres to the structure of the form group and replaces the entire value for the control.

 

       Use the patchValue() method to replace any properties defined in the object that have changed in the form model.

       updateProfile() {

         this.profileForm.patchValue({    firstName: 'Nancy',    address: {      street: '123 Drew Street'    }

         });

       }

 

Validations

     <p>Complete the form to enable button.</p>

     <button type="submit" [disabled]="!profileForm.valid">Submit</button>

     The above mark up allows submission only if form control values are valid. These validation rules will be defined by you using validators.

 

 

Validation in template forms

 

     <input type="text" id="name" name="name" class="form-control"

           required minlength="4" appForbiddenName="bob"      [(ngModel)]="hero.name" #name="ngModel">

 

     <div *ngIf="name.invalid && (name.dirty || name.touched)"    class="alert">  <div *ngIf="name.errors?.required">    Name is required.

       </div>  <div *ngIf="name.errors?.minlength">

         Name must be at least 4 characters long.  </div>  <div *ngIf="name.errors?.forbiddenName">

         Name cannot be Bob.

       </div>

 

     </div>

Validation in template forms

     The <input> element carries the HTML validation attributes: required and minlength. It also carries a custom validator directive, forbiddenName. For more information, see the Custom validators section.

 

     #name="ngModel" exports NgModel into a local variable called name. NgModel mirrors many of the properties of its underlying FormControl instance, so you can use this in the template to check for control states such as valid and dirty. For a full list of control properties, see the AbstractControl API reference.

 

     The *ngIf on the <div> element reveals a set of nested message divs but only if the name is invalid and the control is either dirty or touched.

 

     Each nested <div> can present a custom message for one of the possible validation errors. There are messages for required, minlength, and forbiddenName.

Validation in Reactive forms

     Built in validators can be used at the form control level to validate inputs

     Custom validators can also be used to validate. Lets look at an example

     name: new FormControl(this.hero.name, [ Validators.required,  Validators.minLength(4), //these 2 validators check for required and length of input.

      forbiddenNameValidator(/bob/i) // <-- Here's how you pass in the custom validator. ]),

Building a custom validator

      forbiddenNameValidator(nameRe: RegExp): ValidatorFn {

          //first line creates a function which accepts a regularexp.//It also implements interface validatorfn   .A function that receives a control and synchronously returns a map of validation errors if present, otherwise null.

          //the interfae implemented method.is passed the control.valuereturn (control: AbstractControl): ValidationErrors | null => {

              return (control: AbstractControl): ValidationErrors | null => {

                const forbidden = nameRe.test(control.value)//this will return true or false    nbased on whether the control,value, matches with the regular expression          // added in the validatr

                return forbidden ? {forbiddenName: {value: control.value}} : null;

              };

      }//imports of validator,validationerrors,validationfn required from angular/forms

     Ternary operator

     function getFee(isMember) {  return (isMember ? '$2.00' : '$10.00');}

     If true 2 ,if null or false 10

Creating a user registration form demo

     ng g component formapp  to generate form app