h1(#basic-angular-tutorial). Basic Angular tutorial bq. This page has been fully written by "Domitille Bertin":https://www.linkedin.com/in/domitillebertin/ (Epita, AppingX-2020) "Christian Martins":https://www.linkedin.com/in/christian-martins-074517ba/ (Epita, AppingX-2020) and validated by Thomas Broussard h2(#what-is-angular). What is Angular? h3(#general-purpose). General purpose Angular is a web front-end framework that helps developers building modern, responsive apps that work on a wide variety of platforms. Indeed, Angular can be used to build progressive web apps, which can be run on web browsers or as regular apps on smartphones (see "Ionic framework":https://ionicframework.com/) or computers (see "Electron framework":https://electronjs.org/ ). More precisely, Angular allows developers to build _Single Page Applications_ (SPA). SPAs are essentially websites on which navigation does not require any page reloading when clicking on links. "Google Maps":https://www.google.com/maps is the perfect example of this concept. If a user types an address on the search bar, there is no page reloading, which would be critical for user experience. Instead, the current page is simply reorganized to show the requested information: map position changes, and the UI displays new relevant data etc. h3(#angular-versions). Angular versions In 2009, Google released the first version of AngularJS (note the "JS" suffix), which can be considered as Angular's "big brother", Angular version 1. Later on, in 2016, Angular came out as the new version of AngularJS. This new *major* version was a complete change of paradigm and implied many breaking changes. One of them is the fact that Angular is written in TypeScript, which has become the officially supported language for the framework. Moreover, since Angular 2, an Angular CLI has been introduced allowing developers to manage dependencies, generating code, building and deploying their apps more easily. As these lines a being written, AngularJS has been put under _Long Term Support_ (LTS). This means Google no longer delivers new features, but only provides bug fixes. Meanwhile, there are still many projects running on AngularJS which have not upgraded to Angular. This is mainly because the gap between these two versions is too important and the cost of upgrading too high, particularly for larger projects. h3(#typescript). TypeScript "TypeScript":https://www.typescriptlang.org/ brings, among other features, strict typing and backwards-compatibility to regular JavaScript. While TypeScript is a programming language on its own, it is considered to be a _superset_ of JavaScript because it is always eventually compiled to JavaScript. Angular is built with TypeScript and developers are highly advised to use it, even if they could still use the framework with regular JavaScript. h2(#angular-key-paradigms). Angular key paradigms h3(#mvc-pattern). MVC pattern The way Angular apps are built strongly relies on the _Model-View-Controller_ (MVC) pattern and Angular makes it easy to implement. bc(MVC with angular concept mappings).. @startuml left to right direction skinparam linetype ortho rectangle Controller{ rectangle "service" as service rectangle "component class" as cclass } rectangle Model{ rectangle "transport class" as transport } rectangle View{ rectangle "component view" as cview } transport <--> service transport <--> cclass cclass <--> cview @enduml p. * *Model (Angular Service)* In Angular, the Model is, in most cases referred as _"Service"_. It is responsible for retrieving and managing data, from a REST API for example. * *View (Angular Templates)* Angular terminology defines the View as a _"Template"_. A template is basically HTML that displays data made available by the Controller (see below). Templates are often styled with CSS. * *Controller (Angular Component)* Finally, the Controller (called _"Component"_ in Angular), is responsible for orchestrating interaction between the Service and Template. The fact that MVC separates the data logic (Model/Service) from the UI logic (View/Template) greatly improves maintainability, modularity and makes it easy to implement new features. Just to make it clear, here is a recap of the Angular MVC terminology in a table. table(table). |_. MVC concept |_. Angular equivalent |_. Purpose | | Model | Transport (TypeScript)| Retrieves and manages data.
Can communicate with a REST API. | | View | Template | Displays data.
Composed of HTML and CSS. | | Controller | Component + Service | Map the user actions to backend features | Concrete examples are provided later in this tutorial. h3(#dependency-injection). Dependency injection Angular also relies on the _Dependency Injection_ (DI) pattern and exposes its own internal system to manage it easily. Below is a simplified diagram of how _DI_ works in Angular. !./images/DI.png (Dependency Injection in Angular)! Classes usually need _Dependencies_ to perform their job, which are, most of the time, Angular Services classes that provide data, and data management _API_ s. The _DI pattern_ states that it is the _DI system_ that is responsible for *instantiating* the service classes *once*, rather than the dependent classes themselves.This works amazingly with the MVC pattern and makes the final app more efficient, modular, and maintainable. h2(#representational-state-transfer-rest). REpresentational State Transfer (REST) Most apps usually have two parts: a _front-end_ (the app itself) and a _back-end_. The back-end stores and manages the data, it is invisible to the user. Usually, it is a server which can send data in an organized way and handle requests that modify this data. The _front-end_, on the other hand, is what the user sees: the interface he is presented with. The _front-end_ displays data and forms that allow modifying this data. When the user validates his input, the _front-end_ sends the modifications to the _back-end_, which saves it. As a _back-end_, many Angular apps use _REpresentational State Transfer_ (REST) APIs. _REST_ is a software architecture that defines a set of rules (a norm) for managing _**Resources**_ over a server. It is possible to build a _REST API_ with any programming language that handles the _HTTP_ protocol. Indeed, most _REST API_ s communicate through _HTTP_, even though other protocols can be used as a replacement, like _FTP_. _For this tutorial, we will consider that REST APIs solely rely on the HTTP protocol._ Basically, a _REST API_ exposes _interfaces_ on which actions can be made. A concrete example would be a @student@ interface on a College student management server. Typically, this interface would exposed at an URL like @https://epita.net/api/students@ on which the front-end could send requests with different _HTTP methods_ like: * @PUT@ to create a student * @POST@ to modify a student's information * @DELETE@ to unregister a student that has left the School h2(#creating-a-student-management-app). Creating a student-management app In the last part of this tutorial, we will create a student management app that : * Displays students * Allows editing students We will cover, in a concrete way the concepts explained at the beginning of the tutorial. h3(#prerequisites). Prerequisites For this tutorial, you will need: * *NodeJS 8.x or 10.x installed* Refer to the "official website":https://nodejs.org/en/ for installation guide. * *Angular CLI installed* Run @npm i @angular/cli -g@. * *Your favorite Code editor* Two great options are "Visual Studio Code":https://code.visualstudio.com/ (free) and "WebStorm":https://www.jetbrains.com/webstorm/ (free for students). * *Our app's back-end* This tutorial provides you with a RESTful HTTP Server built for this tutorial with NodeJS and "Express":http://expressjs.com/. Please download the given archive, extract it on your computer and run @node .@ at the root directory of that project. When navigating to @http://localhost:3000/@, you should get the following message: @Server is up and running...@ *Note:* this tutorial does not aim at teach you how to create a REST API but only how to use it with Angular. If you are interested on how to create a RESTful HTTP Server, the code is also provided, but not explained here. h3(#creating-a-new-project). Creating a new project h4(#initializing-a-project-with-the-angular-cli). Initializing a project with the Angular CLI Once you have got everything you need, open up a terminal, navigate to the directory you want you project to be in and run the following command: bc. ng new college-app p. When asked to choose: * Choose "yes" for Routing * SCSS for CSS preprocessor This command initializes a new project. The main folder we will work on is the @src@ folder, which currently contains different files: * @app.component.html@: the main Template (View) of the app * @app.component.scss@: some CSS that styles the Template * @app.component.spec.ts@: don't mind about it for this tutorial, it is a function test script * @app.component.ts@: the Component (Controller) of our main Template * @app.module.ts@: the root Module of the app h4(#cleaning-the-default-template). Cleaning the default template Now that we have this ready, lets open the @app.component.html@ Template and replace all of its content by: bc.

Hello World!

h4(#serving-the-app-with-the-angular-cli). Serving the app with the Angular CLI Check that everything is working fine, run the following command on your terminal: bc. ng serve p. When "Compiles successfully" appears on the console, open your favorite browser and navigate to @http://localhost:4200@. At this point, you should get a webpage that displays "Hello World!". h3(#creating-a-student-service). Creating a Student Service h4(#generating-a-service-with-the-angular-cli). Generating a Service with the Angular CLI The first thing we will is to create a @StudentService@ that will fetch the students from our back-end server, our REST API. To do so, run the following command on your terminal: bc. ng generate service Student p. This command generates a new empty Angular Service by creating a @student.service.ts@ file. Let's take a look at it. bc.. import { Injectable } from '@angular/core'; @Injectable({ providedIn: 'root' }) export class StudentService { constructor() { } } p. Notice the @@Injectable@ annotation. It makes the @StudentService@ class available for Dependency Injection. The @providedIn@ metadata option specifies that the Service will be injectable through the @root@ injector, which means the entire app in our case. h4(#creating-a-data-model). Creating a data Model Our Service will manage students. From a programming point-of-view, this is a Model. In most apps, this would translate into a @Student@ class that allows manipulating instances.However, for our example, we will simply create an @IStudent@ that describes what our back-end server can send us. Let's create a @Student.ts@ file inside the @src/app@ folder of our project containing the following code. bc.. export interface IStudent { id: number; firstName: string; lastName: string; age: number; grade: number; } h4(#communicating-over-http). Communicating over HTTP h5(#introduction). Introduction We want out @StudentService@ to fetch the @IStudent@ objects from the College server. Once the Service has got that data, all Components on which it will be injected into will be able to get the list of @IStudent@ objects. Below is a diagram that shows how our @StudentService@ will interact. !./images/\Diagram.png! h5(#importing-the-httpclientmodule-and-injecting-the-httpclient-service). Importing the @HttpClientModule@ and injecting the @HttpClient@ service First, we need to import Angular's @HttpClientModule@ into our app. To do so, open the @app.module.ts@ file, locate the @imports@ array in the @@NgModule@ annotation and add @HttpClientModule@ to it. This requires importing the symbol from @@angular/common/http@. Your file should look like so: bc.. import { BrowserModule } from '@angular/platform-browser'; import { NgModule } from '@angular/core'; import { AppComponent } from './app.component'; import { HttpClientModule } from '@angular/common/http'; @NgModule({ declarations: [ AppComponent ], imports: [ BrowserModule, HttpClientModule ], providers: [], bootstrap: [AppComponent] }) export class AppModule { } p. Now that the @HttpClientModule@ is available in our app, we need to inject it inside the @StudentService@. To do so, change the Service's constructor signature in the @student.service.ts@ file: bc. constructor(private _httpClient: HttpClient) { } By adding a private attribute of type @HttpClient@ in the constructor's signature, Angular knows that it needs to inject the service into our @StudentService@ service. (Yeah, services can be injected in other services!) h5(#using-the-httpclient-service). Using the @HttpClient@ service Now that we have the @HttpClient@ service available, let's code a @getStudents@ method that will use it to make a HTTP request to our back-end server in order to retrieve the list of all students. bc. public getStudents(): Observable { return this._httpClient.get('http://localhost:3000/students'); } There are a few things that need to be discussed here: * *The return type of the method:* @Observable@ The Angular @HttpClient@ Service relies on the Observer pattern. If you don't know what it is, take a look at "RxJS's explanations about it":https://rxjs-dev.firebaseapp.com/guide/observable. Basically, our Service will return an @Observable@ object on which @Observer@ s will be able to @subscribe@. When the @Observable@ changes, all @Observer@ s will be notified and take action. Here, the Observer will be an Angular Component, which will be notified when the @StudentService@ retrieves the student list to display it to the user. * *The* @get@ *method* The @HttpClient@ Service exposes different methods for performing different kinds of HTTP requests. Here, we want to perform a HTTP @GET@ request to our server, that's why we use @HttpClient@ 's @get@ method. * *The* @get@ *method argument* Our back-end server is available on @localhost:3000@, hence the first argument of the @get@ method. That's it! Our @StudentService@ is ready to fetch our Student list. h3(#creating-a-students-component). Creating a Students Component Now that we have a Service to fetch all the students from our back-end server, we want to display it. Remember that in Angular, we use Components and Templates to display data. We are going to create a @StudentsComponent@ and inject our @StudentService@. This way, the Component will be able to get the students and display it on our app. h5(#generate-a-component-with-angular-cli). Generate a Component with Angular CLI To generate our component, run the following command: bc. ng generate component Students p. This command creates a @students@ directory inside the @src/app@ folder. h5(#injecting-and-consuming-the-studentservice). Injecting and consuming the @StudentService@ First of all, we need to inject the @StudentService@ we previously built into our new Component. To do so, open the @students.component.ts@ file and modify its constructor prototype: bc. constructor(private _studentsService: StudentService) { } Now Angular knows it has to inject the @StudentService@ inside our @StudentsComponent@. Our Component also need an attribute to store the list of students that will be fetched from the @StudentService@. Just add it in the class like so: bc. public students: IStudent[]; The next step is to Observe the Students list from the Service. To do so, we will subscribe to the @Observable@ in the @ngOninit@ method of our Component. Have a look at the following code. bc. ngOnInit() { this._studentsService .getStudents() .subscribe((students: IStudent[]) => this.students = students); } The @Observable.subscribe@ method takes a callback that will be called when the students list has been fetched from the server. The list is then passed as the first argument to the callback, which will set our components internal students list. At this point, your @students.component.ts@ file should look like this: bc.. import { Component, OnInit } from '@angular/core'; import {StudentService} from '../student.service'; import {IStudent} from '../Student'; @Component({ selector: 'app-students', templateUrl: './students.component.html', styleUrls: ['./students.component.scss'] }) export class StudentsComponent implements OnInit { public students: IStudent[]; constructor(private _studentsService: StudentService) { } ngOnInit() { this._studentsService .getStudents() .subscribe((students: IStudent[]) => { console.log(students); this.students = students; }); } } h5(#data-binding-from-component-to-template). Data binding from Component to Template p. Now that our @StudentsComponent@ retrieves data from our server through the @StudentService@, we need to display it in the View (Angular Template). To do so, open the @students.component.html@ file and fill it with the following code: bc..

Students list

  • {{student.firstName}} {{student.lastName | uppercase}}
    Age: {{student.age}}
    Grade: {{student.grade}}/20

There are no students!

p. Many lines need explanations here. * @
    @ Here we use the @ngIf@ directive of Angular. What it does is that it only displays the element if the expression inside the quotes is true. Here the @students@ variable refers to the @StudentsComponent.students@ method, which is the list of students that is filled with the data received from our back-end server through the @StudentService@. If the list does not exist (resolves to @undefined@) or its length is @0@, then we don't display the @ul@ element. * @
  • @ Here is another Angular directive: @ngFor@. This acts as a @for@, iterates over all students in the list and creates an @li@ element for each one of them. * @{{student.firstName}} {{student.lastName | uppercase}}@ You may have noticed Angular's _interpolation_ mechanism on this line. What this does is evaluating and displaying whatever is inside of it. It is also possible to format the data using the @|@ (pipe) operator. Here we use formatting to uppercase the students name. * @

    @ We use @ngIf@ once again to display a message if there are no students (@!students.length@ which is equivalent to @students.length === 0@ in JavaScript) or if the students list has not been retrieved from the server yet (@!students@). h5(#using-the-app-student-component). Using the @app-student@ component At this point, we have: * A @StudentService@ Service that retrieves a student list from a REST API * A @StudentsComponent@ Component that *observes* the data from the @StudentService@ * A _Template_ that displays the data contained inside the @StudentsComponent@ Component Now the last thing we need to do is to use the use the consume the @StudentsComponent@ we have created. Let's first go back to the @students.component.ts@ file and notice line 6: @selector: 'app-students',@. This line declares the HTML tag of the Component. We must add a @app-students@ tag to our @app.component.html@ file to instantiate the @StudentsComponent@ on our app. Let's do so: bc. Save this file, run the @ng serve@ command if needed, and navigate to @http://localhost:4200@ on your web browser. You should obtain something like this: !./images/scr.PNG! There it is! Our students list has been fetched from our back-end server and displayed on our app! h3(#to-go-further). To go further This is a very basic app that only displays data. A more serious app would obviously be more stylish and allow editing students, changing their grade etc. Take a deep look at the "Official Angular Documentation":https://angular.io/docs to learn more and go further with this app! h2(#authors). Credits * Domitille BERTIN (domitille.bertin@epita.fr) * Christian MARTINS (christian.martins@epita.fr) This tutorial has been made as an assignment for Mr. Thomas BROUSSARD. _EPITA APPING_ X 2 2020_ p(breadcrumb). "Tutorials":../../../../tutorials > Advanced > "angular":../../angular > "tutorial":../tutorial > basic-angular-tutorial.textile