IP Hackers App

Objectif

  • Apprendre à concevoir une application front gérant un ensemble d’items + CRUD.

  • Apprendre à utiliser un formulaire structuré

  • Apprendre à mettre en place une communication entre composants (via event)

  • Apprendre à sauvegarder des données de l’application sur le client (localStorage)

L’idée de l’application

On souhaite concevoir une application web qui gère une liste d’IP à l’origine de requêtes malveillantes et qui tente de les géolocaliser via l’appel à un service distant, avec clé d’authentification.

Dans notre contexte, on appelle Hacker une machine à l’origine d’une requête HTTP malveillante. Cela ne veut pas pour autant dire que l’action est intentionnelle, en référence aux machines zombies…​

Machine zombie ? Voir ici Machine zombie sur wikipedia

L’analyse

On concevra une application qui présentera deux pages :

  • Une page Home présentant la liste des IP et un formulaire de modification/création.

  • Une page About

Initialisation de l’application (CLI)

ng new hackers-app --routing --defaults

Les dépendances

On décide de s’appuyer sur bootstrap pour la mise en forme CSS.

Installation des modules bootstrap

(racine du projet)$ npm install --save bootstrap
(racine du projet)$ npm install --save bootstrap-icons

L’option save indique que le paquetage s’installe uniquement dans le projet et non globalement.

Pour nous simplifier le travail de présentation, nous utiliserons le module bootstrap pour angular (https://ng-bootstrap.github.io/#/home)

ng add @ng-bootstrap/ng-bootstrap

Ajouter les dépendances dans angular.json :

[...]
"styles": [
   "node_modules/bootstrap/dist/css/bootstrap.min.css",
   "node_modules/bootstrap-icons/font/bootstrap-icons.css",
   "src/styles.css"
 ],
 "scripts": [
   "node_modules/bootstrap/dist/js/bootstrap.bundle.min.js"
 ]
[...]
Sous Visual Code, on gagnerait à installer l’extension IntelliSense for CSS class names in HTML pour profiter de la complétion des classes CSS.

Les composants

L’application proposera plusieurs "pages", qui devront présenter la même structure.

Nous allons donc factoriser le modèle de présentation par une entête et un pied de page, chacun étant représenté par un composant.

Nous allons avoir besoin de deux services. Un pour gérer l’appel à une API externe et un autre pour la gestion des sauvegardes des données sur le client (local storage)

Création du composant Header

ng g component component/header

Nous prendrons un template classique de la documentation bootstrap.

/src/app/compoment/header/header.component.html
<nav class="navbar navbar-expand-xl navbar-light bg-light">
    <div class="container-fluid">
      <a class="navbar-brand" href="#">
        <h1>MyApp</h1>
      </a>
      <button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarSupportedContent"
        aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation">
        <span class="navbar-toggler-icon"></span>
      </button>
      <div class="collapse navbar-collapse" id="navbarSupportedContent">
        <ul class="navbar-nav me-auto mb-2 mb-lg-0">
          <li class="nav-item">
            <a class="nav-link active" aria-current="page" href="#">Home</a>
          </li>
          <li class="nav-item dropdown">
            <a class="nav-link dropdown-toggle" href="#" id="navbarDropdown" role="button" data-bs-toggle="dropdown"
              aria-expanded="false">
              Dropdown
            </a>
            <ul class="dropdown-menu" aria-labelledby="navbarDropdown">
              <li><a class="dropdown-item" href="#">Action</a></li>
              <li><a class="dropdown-item" href="#">Another action</a></li>
              <li>
                <hr class="dropdown-divider">
              </li>
              <li><a class="dropdown-item" href="#">Something else here</a></li>
            </ul>
          </li>
          <li class="nav-item">
            <a class="nav-link disabled" href="#" tabindex="-1" aria-disabled="true">Disabled</a>
          </li>
        </ul>
        <form class="d-flex">
          <input class="form-control me-2" type="search" placeholder="Search" aria-label="Search">
          <button class="btn btn-outline-success" type="submit">Search</button>
        </form>
      </div>
    </div>
  </nav>

Conception d’autres composants

Dans le même esprit, créer les composants footer et home (pour ce dernier composant, on présente un peu plus loin comment organiser son template, pour l’instant accepter le contenu par défaut)

Conception des routes

On se contente de définir la route par défaut (composant Home)

src/app/app-routing.modules.ts
import { NgModule } from '@angular/core';
import { RouterModule, Routes } from '@angular/router';
import { HomeComponent } from './component/home/home.component';

const routes: Routes = [
{path: '', component: HomeComponent}
];

@NgModule({
imports: [RouterModule.forRoot(routes)],
exports: [RouterModule]
})
export class AppRoutingModule { }

Intégration

Nous ajoutons maintenant les composants de présentation à notre composant principal.

app.component.html
<app-header></app-header>
<router-outlet></router-outlet>
<app-footer></app-footer>

Test

À ce niveau, l’application devrait être opérationnelle :

ng serve

Réglez les problèmes éventuels avant de poursuivre.

Faire une pause.

Travaux pratiques

  1. Ajouter une page APropos (associé à une route, avec un lien dans la barre de navigation) qui présente à l’utilisateur le lien vers ce document ainsi que le nom du ou des développeurs qui ont réalisé le travail demandé.

Structure des données

Il est temps de définir une structure de données qui caractérise un hacker dans notre application (vous enrichirez cette représentation plus tard).

Cela passe par la définition d’une interface (sens UML) et d’une classe qui l’implémente. Ainsi, si l’on modifie l’interface, le développeur sera dans l’obligation de retoucher la ou les classes qui l’implémentent.

Hacker dans le modèle
Figure 1. Hacker dans le modèle

Nous souhaitons obtenir l’arborescence provisoire suivante :

Arborescence partielle du projet
├src
   ├── app
   │   ├── component
   │   │   ├── footer
   │   │   ├── header
   │   │   └── home
   │   ├── models
   │   │   ├── Hacker.ts
   │   │   └── IHacker.ts
   │   └── service
   ├── assets

Créer le dossier models, ainsi que les 2 fichiers qui sont présentés dans l’arborescence ci-dessus. Voici leur implémentation.

src/app/models/iHacker.ts
/**
 * Représente la structure de données d'un Hacker
 * L'id sera autogénéré par l'application (null si nouveau)
 */
export interface IHacker {
    ip: string,
    countryName: string,
    regionName: string,
    city: string,
    id?: string
}

et

src/app/models/Hacker.ts
import { IHacker } from "./IHacker";

export class Hacker implements IHacker {

   constructor(
       public ip: string,
       public countryName: string,
       public regionName: string,
       public city: string,
       public id?: string) {
       // rien à faire de plus ici
   }

}

À ce niveau, l’application devrait être opérationnelle. Réglez les problèmes éventuels avant de poursuivre.

Création des services

Nous allons créer deux services, l’un aura la responsabilité d’obtenir la géolocalisation d’une IP en appelant un service externe (déjà vu dans le TD Hello World), et l’autre de gérer l’enregistrement des données de "hackers" sur le localStorage du client.

Création des deux services (toujours à partir de la racine du projet)

ng generate service service/lookupIp
ng generate service service/managerHacker

API de géolocalisation

Nous utiliserons le service https://ipstack.com/product dans sa version gratuite.

Réaliser les actions ci-dessous.

  1. Créer un compte sur cette plateforme

  2. Choisir une formule

  3. Copier votre clé d’accès à l’API

  4. Tester votre clé :

Exemple de résultat attendu.

location IP
Figure 2. 103.125.234.210.png

⇒ Remarquez la présence d’un lien vers le drapeau du pays (format vectoriel svg).

Intégration de la clé API dans l’environnement

Nous allons stocker la clé API du développeur dans un fichier de ressource qui nous permettra de définir des variables d’environnement.

Cette possibilité est intégrée à Angular.

(racine de l'application) ng generate environments
CREATE src/environments/environment.ts (31 bytes)
CREATE src/environments/environment.development.ts (31 bytes)
UPDATE angular.json (3171 bytes)

Les données déclarées dans environment.development.ts ne seront accessibles qu’en mode dev.

Intégrons à ce fichier quelques variables, dont la clé de l’API d’ipstack.com du développeur :

src/app/environments/environment.ts (production)
export const environment = {
  production: true,
  apiBaseUrl: 'https://api.ipstack.com/',
  keyAPI: 'A RENSEIGNER'
};
src/app/environments/environment.development.ts (dev)
export const environment = {
  production: false,
  apiBaseUrl: 'http://api.ipstack.com/',
  keyAPI: 'VOTRE CLE DE DEV'
};
Attention, avec ce service, en mode gratuit, seul le protocole http non sécurisé est possible (pas de https)

Implémentation du service lookupIp

Nous ajoutons une méthode que nous nommons getGeoLocationIp qui prend un paramètre nommé ip et retourne une référence à un objet de type Observable (pour rafraichir votre mémoire, reportez-vous au premier TD HelloWorld)

src/app/service/lookup-ip.service.ts
import { HttpClient } from '@angular/common/http';
import { Injectable } from '@angular/core';
import { environment } from 'src/environments/environment'; (1)

@Injectable({
providedIn: 'root'
})
export class LookupIpService {

  constructor(private http:HttpClient) { }

  public getGeoLocationIp(ip: string) : Observable<any> {
    return this.http.get(environment.apiBaseUrl + ip + '?output=json&access_key=' + environment.keyAPI);
  }
}
1 En phase de développement, le fichier src/environments/environment.ts est dynamiquement remplacé par src/environments/environment.development.ts.

Ce mécanisme est inscrit dans le fichier ̀angular.json. Voir dans le fichier build/configuration/development/fileReplacements.

Merci Angular !

À ce niveau, l’application devrait toujours être opérationnelle. Réglez les problèmes éventuels avant de poursuivre.

Maquette de home.component.html

Nous souhaitons que la page principale présente à la fois la liste des hackers détenus par l’applicaton dans son localStorage et un formulaire pour l'édition et la création.

Il est d’usage de réaliser une maquette avant de se lancer dans la création de la vue. Cette maquette peut être produite à la main, sur papier, ou via des logiciels spécialisés. C’est rapide à réaliser, et on peut ainsi soumettre plus efficacement nos idées au client.

Voir ici le concept de : maquette
png

La conception de la maquette a été réalisée en utilisant plantuml et salt

Vous trouverez le code de cette maquette ici : code de la maquette

Composants inclus dans la page Home

Nous allons créer 2 composants : HackerFormComponent et HackerListComponent

Création des composants

ng generate component component/hackerForm
ng generate component component/hackerList

Inclusion des composants dans Home

On donne 1/3 de la page au formulaire et le reste à la liste (en bootstrap, la somme des colonnes est 12)

src/app/component/home/home.component.html
<div class="w-100 p-2 ">
    <div class="row">
        <div class="col-md-3">
            <app-hacker-form></app-hacker-form>
        </div>
        <div class="col-md-9">
            <app-hacker-list></app-hacker-list>
        </div>
    </div>
</div>

À ce niveau, l’application devrait être opérationnelle.

Exemple : maquette1

Réglez les problèmes éventuels avant de poursuivre.

Conception du formulaire

Il y a 2 façons d’implémenter les formulaires en Angular.

  • Les formulaires réactifs

  • Les formulaires pilotés par les templates

Nous ferons usage des formulaires réactifs (approche plus structurée)

Prenez le temps de consulter le guide officiel : https://angular.io/guide/reactive-forms

Ajouter la dépendance à ReactiveFormsModule

Commençons par ajouter une dépendance à notre projet (ReactiveFormsModule)

app.module.ts
[...]
import { ReactiveFormsModule } from '@angular/forms';

 imports: [
    [...],
    ReactiveFormsModule
  ],

Implémenter la classe du composant HackerFormComponent

app/component/hacker-form/hacker-form.component.ts
import { Component } from '@angular/core';
import { FormControl, FormGroup } from '@angular/forms';
import { Hacker } from 'src/app/models/Hacker';

@Component({
  selector: 'app-hacker-form',
  templateUrl: './hacker-form.component.html',
  styleUrls: ['./hacker-form.component.css']
})
export class HackerFormComponent {

  hacker: Hacker =  new Hacker('','','','')

  hackerForm = new FormGroup({ (1)
    ip: new FormControl(''),
    countryName: new FormControl(''),
    regionName: new FormControl(''),
    city: new FormControl(''),
    id: new FormControl(undefined)
  })

  onSubmit() {
    console.log("Submit")
    console.log(this.hackerForm.value)
  }

   clear() {
    this.hackerForm.controls.ip.setValue("IP à renseigner") (2)
    console.log("cancel")
    console.log(this.hackerForm.value)
  }
}
1 Cet objet permet de créer un formulaire réactif. Toute modification apportée dans le formulaire HTML sera répercutée sur ces objets. La structure de l’objet FormGroup reproduit la structure de données IHacker (un interface au sens UML).
2 La mise à jour de la valeur du contrôle sera répercutée sur la vue.

Implémenter le template du composant HackerFormComponent

Conformément au guide d’angular, le formulaire fait référence à un formGroup de la classe.

<form [formGroup]="hackerForm"  ...>

On fait usage des classes et icons de bootStrap (pour le choix des icones voir https://icons.getbootstrap.com/)

<div>
  <h4>Hacker</h4>

  <form [formGroup]="hackerForm" (ngSubmit)="onSubmit()"> (1) (2)
     <div class="form-group">
         <i class="bi bi-person"></i>
         <input class="d-inline" formControlName="ip" (3)
                placeholder="IP du hacker" required>
         <button type="button" class="btn btn-success m-2 d-inline">Lookup</button>
     </div>
    <div class="form-group">
        <i class="bi bi-globe"></i>
        <input class="form-control" formControlName="countryName"
            placeholder="Pays" required>
    </div>
    <div class="form-group">
        <i class="bi bi-pentagon"></i>
        <input class="form-control" formControlName="regionName"
            placeholder="Région" required>
    </div>
    <div class="form-group">
        <i class="bi bi-house"></i>
        <input class="form-control" formControlName="city"
            placeholder="Ville" required>
    </div>
    <div class="form-group visually-hidden">
        <i class="bi bi-person"></i>
        <input class="form-control" formControlName="id"
            placeholder="id">
    </div>

    <div class="form-group pt-2">
        <div class="form-group float-end">
            <button type="button" class="btn btn-success m-2" (click)="clear()">Cancel</button> (4)
            <button type="submit" class="btn btn-success" [disabled]="hackerForm.invalid">
                <span>
                    <i class="bi bi-plus"></i>
                    Ajouter / Mettre à jour
                </span>
            </button>
        </div>
    </div>
  </form>
</div>
1 "hackerForm" est le nom de la propriété de la classe du composant, de type FormGroup
2 (ngSubmit)="onSubmit()". ngSubmit est un événement généré par Angular lorsque l’utilisateur soumet le formulaire. onSubmit() est le nom de la méthode de la classe du composant qui sera appelée.
3 formControlName="ip", il faut reporter ici le nom des variables de type FormControl de la classe du component (par exemple ip)
4 (click)="clear()", même logique que (2), sur le clic d’un bouton.

À ce niveau, l’application est toujours opérationnelle. Réglez les problèmes éventuels avant de poursuivre.

Consulter la console sur le client (F12, onglet Console) pour vérifier que les méthodes associées aux événements submit et clic fonctionnent.

Voici ce que peut donner notre template lorsqu’il est interprété par un navigateur :

ui formulaire hacker

Travaux pratiques

  1. En vous basant sur le travail réalisé dans le TD Hello World (getIPAddress), appeler le service LookupIpService sur le clic du bouton Lookup afin de valoriser automatiquement les valeurs des input du formulaire. (une compétence normalement acquise, même si vous aurez besoin de consulter la documentation)

  2. Faire en sorte que l’action sur le bouton Cancel réinitialise le formulaire en totalité. (réalisable sans avoir besoin de chercher une solution sur le net. Observez bien le code actuel.)

Enregistrer des données sur le client

Il existe 2 solutions pour stocker des données sur le client.

  • Web Storage API une solution pour enregistrer et retrouver de "petites" données, une donnée est stockée sous la forme de couple (clé, valeur).

  • IndexedDB API le navigateur dispose ici d’un système de base de de données pour stocker des données complexes comme des enregistrements de données structurés ou encore des flux/fichier audio ou video.

Nous utiliserons la solution Web Storage API (plus simple à mettre en oeuvre, quitte à transformer une structure "complexe" en JSON).

Web Storage API

Il y a 2 API, localStorage (persistant) et sessionStorage (le temps d’une session de page).

La zone de stockage est dédiée à une origin (nom de domaine, ip). Ainsi du code JS d’une application provenant d’un domaine, disons domaineA.com, ne peut exploiter les données stockées sur le client d’un autre domaine, comme domaineB.com par exemple.
LocalStorage is similar to sessionStorage, except that while localStorage data has no expiration time, sessionStorage data gets cleared when the page session ends — that is, when the page is closed. (localStorage data for a document loaded in a "private browsing" or "incognito" session is cleared when the last "private" tab is closed.)
Ne jamais sauvegarder des données sensibles sur le client !

Composant ManagerHackerService

Pour une bonne répartition des responsabilités, nous décidons de placer la logique de gestion de la persistance des données dans la classe de service ManagerHackerService.

L’exemple ci-dessous implémente la fonction qui permet de placer les données des hackers dans la mémoire vive sous la forme d’un tableau d’objets.

import { Injectable } from '@angular/core';
import { Hacker } from '../models/Hacker';

@Injectable({
  providedIn: 'root'
})
export class ManagerHackerService {

  constructor() { }

  /**
   * Get hackers stored locally on client side (localStorage)
   * @returns list of Hackers
   */
  getAllHackers(): Hacker[] {
    return JSON.parse(localStorage.getItem('badguys') || '[]');
  }

}

Nous avons fait le choix de sauvegarder le tableau des Hackers sous la forme d’un tableau JSON au format texte, obtenu via la méthode JSON.stringify. C’est pourquoi nous utilisons ici la fonction inverse JSON.parse pour charger le tableau en mémoire.

Les données du localstorage sont consultables sur le navogateur en mode développeur.

localstorage consult

Composant HackerListComponent

Ce composant présente à l’utilisateur la liste des hackers stockés sur le client.

Pour réaliser sa fonction, ce composant s’appuie sur une instance de ManagerHackerService transmise par Angular.

Conception de la classe du composant

Nous déclarons une propriété de type ManagerHackerService, directement en tant que paramètres du constructeur (c’est un sucre syntaxique bien pratique).

On en profite pour initialiser la propriété hackers de ce composant (en fait, c’est le rôle d’un constructeur d’initialiser les attributs d’instance)

hacker-list.component.ts
import { Component } from '@angular/core';
import { Hacker } from 'src/app/models/Hacker';
import { ManagerHackerService } from 'src/app/service/manager-hacker.service';

@Component({
  selector: 'app-hacker-list',
  templateUrl: './hacker-list.component.html',
  styleUrls: ['./hacker-list.component.css']
})
export class HackerListComponent {

  hackers: Hacker[]

  constructor(private managerHackerServie: ManagerHackerService) {
    this.hackers = managerHackerServie.getAllHackers()
  }

}

Conception du template du composant

hacker-list.component.html
<style>
    table.center {
        margin-left: auto;
        margin-right: auto;
    }
</style>

<table *ngIf="hackers.length > 0; else hackersEmpty" class="table is-striped center">
    <thead>
        <tr>
            <th>IP</th>
            <th>Pays</th>
            <th>Région</th>
            <th>Ville</th>
        </tr>
    </thead>
    <tbody>
        <tr *ngFor="let hacker of hackers"> (1)
            <td>{{ hacker.ip }}</td>
            <td>{{ hacker.countryName }}</td>
            <td>{{ hacker.regionName }}</td>
            <td>{{ hacker.city }}</td>
        </tr>
    </tbody>
    <tfoot>
        <div style="font-weight:bold;">Nombre d'Hackers : {{ hackers.length }}</div>
    </tfoot>
</table>

<ng-template #hackersEmpty> (2)
    <p>Pas de hackers ! </p>
</ng-template>
1 Exploitation de la propriété hackers de la classe du composant (une itération sur l’ensemble des éléments de la collection)
2 Template nommé. Très utile lorsque que l’on souhaite différencier certaines parties, comme ici en cas de liste vide (voir la balise ouvrante de <table…​)

On vient de voir comment le composant hacker-list.component.ts obtient la liste des hackers enregistrés sur le poste client.

Pour enregistrer un nouvel Hacker, composant HackerFormComponent fera également appel à ManagerHackerService, qui devra donc être injecté dans le composant HackerFormComponent.

Par contre, comment le composant HackerFormComponent obtiendra-t-il l’objet Hacker à modifier ?

Communication entre composants

D’après la Maquette de home.component.html, un lien "modifier" sera placé sur chaque ligne des hackers de la liste. Ceci nous laisse penser que le HackerListComponent connait HackerFormComponent…​ Or, ce n’est pas dans les bonnes pratiques de faire un tel couplage, car ces 2 composants n’ont pas à être en dépendance directe. En effet, le fait que les deux templates soient placés sur la même page tient de la logique de l’UI seulement.

La solution la plus propre consiste à passer par ManagerHackerService, qui est déjà le composant commun à ces 2 composants.

Diagram

L’idée est de permettre à l’utilisateur de déclencher un événement (Event), par une action sur un lien par exemple, qui sera intercepté (exploité) par HackerFormComponent.

Voyons cela en détail.

Voici la nouvelle version de hacker-list.component.html

hacker-list.component.html
<style>
    table.center {
        margin-left: auto;
        margin-right: auto;
    }
</style>

<table *ngIf="hackers.length > 0; else hackersEmpty" class="table is-striped center">
    <thead>
        <tr>
            <th>IP</th>
            <th>Pays</th>
            <th>Région</th>
            <th>Ville</th>
            <th>Opérations</th>
        </tr>
    </thead>
    <tbody>
        <tr *ngFor="let hacker of hackers">
            <td>{{ hacker.ip }}</td>
            <td>{{ hacker.countryName }}</td>
            <td>{{ hacker.regionName }}</td>
            <td>{{ hacker.city }}</td>
            <td>
              <i class="bi bi-pencil-square"
                 (click)="editHacker(hacker)" (1)
                 style="color: green; cursor: pointer;">
              </i>
            </td>
        </tr>
    </tbody>
    <tfoot>
        <div style="font-weight:bold;">Nombre d'Hackers : {{ hackers.length }}</div>
    </tfoot>
</table>

<ng-template #hackersEmpty>
    <p>Pas de hackers ! </p>
</ng-template>
1 editHacker(hacker), une nouvelle méthode du composant.

Cette méthode soutraite au manager le travail demandé.

hacker-list.component.ts
[...]
  editHacker(hacker: Hacker) {
    this.managerHackerService.editHacker(hacker)  (1)
  }
[...]
1 Une nouvelle méthode de la classe ManagerHackerService

Pour communiquer avec un composant, nous avons vu (TD Multiplication) qu’un composant parent peut contrôler ses composants enfants via leurs attributs (propriété décorée @Input dans la classe du composant).

Nous voyons ici un autre façon d’opérer, plus générale, qui permet à 2 composants, et plus, de communiquer entre eux, sans nécessairement être liés par une relation parent-enfant. Cette façon de faire passe par la gestion d’événements, sorte de signaux transmis globalement dans l’arbre DOM. C’est ainsi qu’un enfant pourrait passer des informations à son parent par exemple.

Nous souhaitons que notre manager puisse lancer un événement relatif à la demande de modification (edition, update) d’un objet Hacker. Nous nommons editHackerEvent cet événement.

manager-hacker.service.ts
import { EventEmitter, Injectable, Output } from '@angular/core';
import { Hacker } from '../models/Hacker';
import { IHacker } from '../models/IHacker';

@Injectable({
  providedIn: 'root'
})
export class ManagerHackerService {

  constructor() { }

  @Output() editHackerEvent = new EventEmitter<IHacker>() (1)

  editHacker(hacker: IHacker) {
    this.editHackerEvent.emit(hacker)  (2)
  }

  /**
   * Get hackers stored locally on client side (localStorage)
   * @returns list of Hackers
   */
  getAllHackers(): Hacker[] {
    return JSON.parse(localStorage.getItem('badguys') || '[]');
  }

}
1 Déclaration et initialisation d’un objet émetteur d’événements (@Output)
2 Lancement d’un événement.

L’événement est paramétré - d’après sa déclaration - ainsi la référence à un objet Hacker sera transmise à tous les objets qui se déclareront concernés par cet événement - dans notre cas, ce sera le formulaire (l’objet HackerFormComponent)

Dans l’état, le projet devrait être stable.

Gérer les problèmes éventuels avant de poursuivre.

Exploitation des données de formulaire

Vous vous êtes rendu compte que l’événement lancé au déclenchement de l’édition d’un hacker ne génère aucune action visible ! et pour cause, l’événement est tout simplement ignoré par l’application ! Corrigeons cela.

hacker-form.component.ts
[...]
  ngOnInit(): void {
    this.managerHackerService.editHackerEvent
      .subscribe((hacker: IHacker) => { (1)
        console.log('Event message editEvent')
        this.hacker_to_hackerForm(hacker) (2)
      })
  }
[...]
1 L’instance de notre formulaire s’abonne aux événements editHackerEvent du manger.
2 Ici l’opération réalisée consiste à placer les valeurs des propriétés de l’instance référencée par hacker dans les champs de contrôle du formulaire.

Le composant formulaire a la charge de permettre la création et la modification d’un Hacker.

Pour cela il s’appuie sur 2 types de composants essentiels :

  1. FormControl : Angular s’assure que ces objets sont synchronisés avec leur représentation dans le DOM (rendu et interaction avec l’utilisateur). Ces objets techniques peuvent être associés à des objets Validators (hos du champ de ce TD).

  2. ManagerHackerService pour la persistance (lecture et écriture) de la liste des objets Hacker. On demande à Angular de nous injecter une instance de cette classe via le constructeur (comme pour HackerListComponent)

Le composant HackerFormComponent va avoir besoin de transférer les données du formulaire dans les propriétés d’un objet Hacker et vice-versa. Nous décidons de représenter ces fonctions par 2 méthodes privées que nous nommons hackerForm_to_hacker et hacker_to_hackerForm.

hacker-form.component.ts
  /**
   * Create instance of Hacker from hackerForm data
   * @returns a ref to Hacker object
   */
  private hackerForm_to_hacker(): IHacker {
    return new Hacker(
      this.hackerForm.controls.ip.value ?? '', (1)
      this.hackerForm.controls.countryName.value ?? '',
      this.hackerForm.controls.regionName.value ?? '',
      this.hackerForm.controls.city.value ?? '',
      this.hackerForm.controls.id.value ?? undefined)
  }

  /**
   * Initializes this.hackerForm from parameter hacker instance (object)
   */
  private hacker_to_hackerForm(hacker: IHacker): void {
    this.hackerForm.patchValue({ (2)
      ip: hacker.ip,
      // à compléter !
    })
  }
1 L’opérateur ?? nous permet de définir une valeur par défaut
2 La méthode patchValue permet de sélectionner les attributs à mettre à jour. Ici tous doivent l’être.

Travaux pratiques

  1. Rendre opérationnel l’enregistrement d’un nouvel hacker

  2. Rendre opérationnel la modification d’un hacker de la liste

  3. Faire en sorte qu’au moment de la création d’un nouvel hacker, un ID unique lui soit alloué. C’est le ManagerHackerService qui se chargera de créer cet ID unique juste avant la sauvegarde (déclenchée par l’utilisateur lorsqu’il actionne le bouton "Ajouter" du formulaire)

    Indice 1 : Cela fait référence au concept de UUID - wikipedia

  4. Dans la vue en liste, ajouter l’opérations de suppression d’un item (se référer à la Maquette de home.component.html)

  5. Le bouton submit du formulaire est actuellement labellisé "Ajouter / Mettre à jour".

    Modifier dynamiquement (comprendre par du code, pas statiquement) ce comportement afin d’afficher soit Ajouter, soit Mettre à jour, pour mieux informer l’utilisateur (UX)

  6. Ajouter le lien vers le drapeau du pays (obtenu par l’API externe) comme nouvelle propriété à la Structure des données d’un Hacker, et présenter le drapeau dans la liste et la vue formulaire.

  7. Vérifier que l’application ne soumet à l’API distant que des valeurs d’IP saisie par l’utilisateur qui soient au format IPV4 ou IPV6 (prévoir un filtre qui utilise la technologie des expressions régulières)

  8. Ne pas soumettre non plus des adresses non routables, car il est, dans ce cas, inutile de consommer du crédit API pour rien. Voir : Adresses non routables

That’s all !!