-
Εγκατάσταση του Angular CLI
npm install -g @angular/cli@latest
-
Δημιουργία ενός νέου Angular Project
ng new angular-introduction --standalone --skip-tests
-
Επεμβάσεις στο αρχείο
ts.config.json
{ ... "compilerOptions": { ... "baseUrl": "./", "strict": false, ... } ... }
-
Εκκίνηση του Angular Project
❯ ng serve Initial chunk files | Names | Raw size polyfills.js | polyfills | 83.60 kB | main.js | main | 1.67 kB | styles.css | styles | 95 bytes | | Initial total | 85.36 kB Application bundle generation complete. [1.241 seconds] Watch mode enabled. Watching for file changes... ➜ Local: http://localhost:4200/ ➜ press h + enter to show help
-
Η εφαρμογή είναι διαθέσιμη στη διεύθυνση
http://localhost:4200/
-
Δημιουργία online repository στο GitHub (
angular-introduction
) και αποστολή του κώδικαgit remote add origin git@github.com:christodoulos/angular-introduction.git git push -u origin main
-
Δημιουργία του repository
<username>.github.io
αν δεν υπάρχει ήδη. -
Προσθήκη δυνατότητας deployment στις σελίδες gh-pages του GitHub
ng add angular-cli-ghpages
-
Προσθήκη του deploy script στο αρχείο
package.json
{ ... "scripts": { ... "deploy": "ng deploy --base-href=https://<username>.github.io/angular-introduction/" } ... }
-
Αποστολή της εφαρμογής στις σελίδες gh-pages του GitHub
npm run deploy
-
Η εφαρμογή είναι διαθέσιμη στη διεύθυνση
https://<username>.github.io/angular-introduction/
-
Ενεργοποίηση του GitHub Pages για το repository
<username>.github.io/angular-introduction
-
Η εφαρμογή είναι διαθέσιμη στη διεύθυνση
https://<username>.github.io/angular-introduction/
-
Εγκατάσταση του bootstrap
npm install bootstrap
-
Επεξεργασία του αρχείου
angular.json
{ ... "styles": [ "src/styles.css", "node_modules/bootstrap/dist/css/bootstrap.min.css" ], ... }
-
Επανεκκίνηση του Angular Project μετά από κάθε αλλαγή στο αρχείο
angular.json
είναι απαραίτητο να εκκινηθεί ξανά το Angular Project (^C και ξανάng serve
) -
Τοπική εγκατάσταση του
prettier
και δημιουργία του αρχείου.prettierrc
npm install --save-dev prettier
{ "overrides": [ { "files": "*.html", "options": { "parser": "angular" } } ] }
-
Χρήση του placeholder
{{ <atribute_name > }}
για τη δεσμευση του χαρακτηριστικούattribute_name
στο template του component. -
Αν το χαρακτηριστικό της κλάσης είναι αντικείμενο τότε χρησιμοποιούμε τη γνωστή σύνταξη
{{ <object_name>.<attribute_name> }}
.
- Δημιουργία ενός νέου component με την εντολή
ng generate component components/person-table
. - Μεταφορά του πίνακα από το
app.component.html
στο template του νέου component. - Μεταφορά του χαρακτηριστικού
person
από την κλάσηAppComponent
στην κλάσηPersonTableComponent
. - Συμπερίληψη της κλάσης
PersonTableComponent
στον πίνακαimports
στην αρχικοποίηση του decorator στο αρχείοapp.component.ts
. - Χρήση του νέου component στο template του
app.component.html
με την ετικέτα<app-person-table></app-person-table>
.
-
Δημιουργία interface για τα δεδομένα τύπου
Person
ng generate interface shared/interfaces/person
export interface Person { givenName: string; surName: string; age: number; email: string; address: string; }
-
Χρήση του interface
Person
ως τύπο του χαρακτηριστικούperson
στο componentPersonTableComponent
-
Χρήση του decorator
@Input()
στο χαρακτηριστικόperson
τύπουPerson
ήundefined
στο componentPersonTableComponent
-
Χρήση του
@if() {} @else {}
στο template του componentPersonTableComponent
για την υπό συνθήκη εμφάνιση των δεδομένων του χαρακτηριστικούperson
-
Η δέσμευση των χαρακτηριστικών της κλάσης
AppComponent
στο χαρακτηριστικόperson
του componentPersonTableComponent
γίνεται στο template του componentAppComponent
<app-person-table [person]="person0"></app-person-table>
<!-- Χωρίς δέσμευση στο επόμενο -->
<app-person-table></app-person-table>
<app-person-table [person]="person1"></app-person-table>
- Ορισμός χαρακτηριστικού
persons
τύπουPerson[]
στην κλάσηAppComponent
(πίνακας αντικειμένων τύπουPerson
) - Χρήση του template directive
@for(obj of objects); track obj
για την εμφάνιση των δεδομένων του πίνακαpersons
με τη χρήση του componentPersonTableComponent
@for (user of users; track user) {
<app-person-table [person]="user"></app-person-table>
}
-
Δέσμευση μεθόδου της κλάσης (event handler) στο συμβάν
event
του template με χρήση του(eventName)="onEventName($event)"
<button (click)="onAddPerson()">Add Person</button>
-
Χρήση του event
input
από ένα HTML input element για ανάγνωση της τιμής του στην κλάση και στη συνέχεια πέρασμα πίσω στο template με χρήση της απλής δέσμευση με το{{ <atribute_name > }}
<input type="text" (input)="onInput($event)" />
-
Σκοπός μας είναι να κάνουμε επιλογές από το μενού στα αριστερά και τα component να εμφανίζονται στο χώρο δεξιά.
-
Δημιουργία του Welcome component, αυτό που θα εμφανίζεται πρώτο όταν ξεκινήσει η εφαρμογή (χρησιμοποιεί κι ένα λογότυπο από το
/assets
):ng g c welcome
-
Στο αρχείο
app.routes.ts
ο πίνακαςroutes
περιέχει αντικείμενα που είναι ο κατάλογος των path που εμφανίζονται στο μενού της εφαρμογής μαζί με το Angular component που αντιστοιχεί στο path.import { Routes } from "@angular/router"; import { EventBindExampleComponent } from "src/app/components/event-bind-example/event-bind-example.component"; import { WelcomeComponent } from "./components/welcome/welcome.component"; export const routes: Routes = [ { path: "event-bind-example", component: EventBindExampleComponent }, { path: "welcome", component: WelcomeComponent }, { path: "", redirectTo: "/welcome", pathMatch: "full" }, ];
-
Ήδη στο αρχείο
app.config.ts
ο κατάλογος των routes περνάει στοprovideRouter
:import { ApplicationConfig } from "@angular/core"; import { provideRouter } from "@angular/router"; import { routes } from "./app.routes"; export const appConfig: ApplicationConfig = { providers: [provideRouter(routes)], };
-
Το ακριβές σημείο στο template που θα εισάγονται τα component δηλώνεται με τη χρήση του tag
<router-outlet>
:... <span class="flex-grow-1 p-2 text-nowrap"> <router-outlet></router-outlet> </span> ...
-
Παράδειγμα ροής για μια επιλογή του χρήστη:
- Ο χρήστης επιλέγει κάτι από το μενού που στην HTML το tag που αφορά την επιλογή συμπεριλαμβάνει την οδηγία
routerLink
, π.χ. στοapp.component.html
το tag<span role="button" routerLink="event-bind-example">Event Bind Example</span>
. - Ο έλεγχος μεταβιβάζεται στο αρχείο
app.routes.ts
όπου γίνεται αναζήτηση στον πίνακαroutes
για την εύρεση του αντικειμένου που έχει τιμή στο χαρακτηριστικόpath
ίδια με την τιμή τουrouterLink
στο tag από το βήμα 1. - To URL αλλάζει σε αυτό που αντιστοιχεί στο path του αντικειμένου του βήματος 2.
- Στο πλαίσιο του
<router-outlet></router-outlet>
εμφανίζεται το component από το χαρακτηριστικό του αντικειμένου του βήματος 2.
- Ο χρήστης επιλέγει κάτι από το μενού που στην HTML το tag που αφορά την επιλογή συμπεριλαμβάνει την οδηγία
-
Δημιουργία των
ComponentInputExampleComponent
καιForDirectiveExampleComponent
και προσθήκη στο μενού της εφαρμογής:- Ενημέρωση του αρχείου
app.routes.ts
- Ενημέρωση του html μενού με τις κατάλληλες οδηγίες
routerLink
- Ενημέρωση του αρχείου
Βήμα 7: Fancy App Menu με το list-group του Bootstrap
-
Δημιουργία νέου interface
MenuItem
στο αρχείοshared/interfaces/menu-item.ts
:export interface MenuItem { text: string; // Κείμενο που εμφανίζεται στο μενού routerLink: string; // Το path που αντιστοιχεί στο component }
-
Δημιουργία του component
ListGroupMenuComponent
με την εντολή:ng g c components/list-group-menu
-
To μενού της εφαρμογής μας είναι ένας πίνακας αντικειμένων
MenuEntry
:menu: MenuEntry[] = [ { text: 'Component Input Example', routerLink: 'component-input-example' }, { text: '@for Directive Example', routerLink: 'for-directive-example' }, { text: 'Event Bind Example', routerLink: 'event-bind-example' }, ];
-
Χρήση του https://cobbl.io/ για να παράξουμε ένα πίνακα με πολλά δεδομένα τύπου
ΕPerson
που ορίζουμε στο/shared/interfaces/person.ts
:export interface EPerson { givenName: string; surName: string; age: string; email: string; address: string; education: string; } export const ManyPerson: EPerson[] = [ { given_name: 'Sarah', surName: 'Howard', age: '41', email: 's.m.howard@yahoo.com', education: 'Some college, no degree', }, ...
-
Δημιουργία του
SimpleDataTableComponent
: λαμβάνει δεδομένα τύπουEPerson
και τα εμφανίζει σε έναν πίνακα με δυνατότητα ταξινόμησης ανά στήλη -
Δημιουργία του
SimpleDataTableExampleComponent
: χρησιμοποιεί τοSimpleDataTableComponent
-
Ενημέρωση του μενού της εφαρμογής μας
-
app.routes.ts
:... { path: 'simple-data-table-example', component: SimpleDatatableExampleComponent, } ...
-
list-group-menu.component.ts
:... { text: 'Simple Data Table Example', routerLink: 'simple-data-table-example', } ...
-
-
Εγκατάταση του
lodash-es
:npm i lodash-es npm i --save-dev @types/lodash-es
- Δημιουργία του
ComponentOutputExampleComponent
και ενημέρωση του μενού της εφαρμογής μας (στοapp.routes.ts
και στοlist-group-menu.component.ts
). - Ενημέρωση του
SimpleDataTableComponent
ώστε να περνάει σαν έξοδο τη γραμμή του πίνακα που επιλέγεται με διπλό κλικ.- Χρήση του decorator
@Output()
στο χαρακτηριστικόpersonClicked
τύπουEPerson
στοSimpleDataTableComponent
. - Το output είναι ένα
EventEmitter<T>
που μεταφέρει δεδομένα του συγκεκριμένου τύπου<Τ>
.
- Χρήση του decorator
-
Εγκατάσταση του Angular Material και του Angular CDK:
❯ ng add @angular/material ℹ Using package manager: npm ✔ Found compatible package version: @angular/material@17.3.2. ✔ Package information loaded. The package @angular/material@17.3.2 will be installed and executed. Would you like to proceed? Yes ✔ Packages successfully installed. ? Choose a prebuilt theme name, or "custom" for a custom theme: Indigo/Pink [ Preview: https://material.angular.io?theme=indigo-pink ] ? Set up global Angular Material typography styles? No ? Include the Angular animations module? Include and enable animations UPDATE package.json (1396 bytes) ✔ Packages installed successfully. UPDATE src/app/app.config.ts (338 bytes) UPDATE angular.json (3652 bytes) UPDATE src/index.html (516 bytes) UPDATE src/styles.css (181 bytes)
-
Επέμβαση στο
PersonTableComponent
για να χειρίζεται δεδομένα είτεPerson
είτεEPerson
. -
Επέμβαση στο
ComponentOutputExampleComponent
και αντικατάσταση τουalert
με τοdialog
του Angular Material (https://t.ly/JLFka).
- Δημιουργία των
EpersonTemplateDrivenFormComponent
καιTemplateDrivenFormExampleComponent
. - Ενημέρωση του μενού της εφαρμογής μας (στο
app.routes.ts
και στοlist-group-menu.component.ts
). - Επέμβαση στο
SimpleDatatableComponent
για την περίπτωση του κενού πίνακα. - Χρήση του Angular Forms Module.
Η φόρμα ορίζεται στο template και μεταφέρει δεδομένα στο component κατά την υποβολή της. Συνήθως τότε, ένα EventEmitter μεταφέρει τα δεδομένα στο component γονέα.
- Χρήση του
FormsModule
στον πίνακα imports του component (εμπλουτίζει τα templates με επιπλέον HTML markup ώστε να δημιουργούνται objects από τις φόρμες). <form #form="ngForm">...</form>
ορίζει πως η HTML φόρμα δημιουργεί ένα αντικείμενο που είναι διαχειρίσιμο στα πλαίσια του template με τη μεταβλητή (template variable)form
.- Το αντικείμενο
form
περνά σαν όρισμα στοonSubmit(form)
όταν συμβεί το eventonSubmit
(ελέγχεται από το κουμπί Submit που μπορεί να πατηθεί μόνο όταν η φόρμα είναι ορθά συμπληρωμένη (valid)). - Δίνουμε στο name του input το όνομα του χαρακτηριστικού του αντικειμένου που παράγει η φόρμα και σχετίζεται (το χαρακτηριστικό) με το συγκεκριμένο input. Το συγκεκριμένο χαρακτηριστικό συμμετέχει στο αντικείμενο μόνο αν συμπεριλάβουμε την οδηγία
ngModel
. - Με το
#givenName="ngModel"
δηλώνουμε τη μεταβλητή template με όνομαgivenName
που είναι αντικείμενο που μπορεί να εξεταστεί για την ορθότητά του με τοgivenName.errors
και να χρησιμοποιηθεί για την υπο συνθήκη εμφάνιση επεξηγηματικού κειμένου για το ενδεχόμενο λάθος ορθότητας.
- Ξεκινάμε με τα αντίστοιχα βήματα όπως στο βήμα 11.
Η φόρμα ορίζεται στο component και συνδέεται με τα input του template. Ένας click handler μεταφέρει τα δεδομένα στο component και στη συνέχεια ένα EventEmitter μεταφέρει τα δεδομένα στο component γονέα.
- Χρήση του
ReactiveFormsModule
στον πίνακα imports του component (εμπλουτίζει τα templates με επιπλέον HTML markup ώστε να μπορούν να συσχετιστούν με τα χαρακτηριστικά του component). - Χρήση των κλάσεων
FormGroup
καιFormControl
για τη δόμηση του αντικειμένου που παράγεται από τη φόρμα. Χρήση τωνValidators
. - Δέσμευση του χαρακτηριστικού
form
του component με χρήση του<form [formGroup]="form">...</form>
. - Σύνδεση του input με το
FormControl
με χρήση τουformControlName
. - Άμεση πρόσβαση στο πεδίο της φόρμας με το
form.get('όνομα πεδίου')
- Κατά το Submit το χαρακτηριστικό
form
έχει ήδη τιμή στο component.
-
Για να μπορέσουμε να χρησιμοποιήσουμε τον HTTP Client είναι απαραίτητη η επέμβαση στο
app.config.ts
:import { ApplicationConfig } from "@angular/core"; import { provideRouter } from "@angular/router"; import { routes } from "./app.routes"; import { provideAnimationsAsync } from "@angular/platform-browser/animations/async"; import { provideHttpClient, withInterceptorsFromDi } from "@angular/common/http"; export const appConfig: ApplicationConfig = { providers: [provideRouter(routes), provideAnimationsAsync(), provideHttpClient(withInterceptorsFromDi())], };
-
Δημιουργία του
JokesService
με την εντολή:ng generate service shared/services/jokes
-
Συνηθίζουμε να ορίζουμε με
const
το URL του API που θα χρησιμοποιήσουμε:const DAD_JOKES_API_URL = "https://icanhazdadjoke.com/"; const JACK_NORRIS_JOKES_API_URL = "https://api.chucknorris.io/jokes/random";
-
Το service είναι μια κλάση της Typescript με τον decorator
@Injectable({providedIn: 'root'})
που επιτρέπει την ενσωμάτωση του service σε όλα τα Angular components με χρήση τουinject
. -
Ο
HttpClient
είναι ένα έτοιμο Angular service που παρέχει τη δυνατότητα αποστολής HTTP requests και λήψης HTTP responses. Τα service της εφαρμογής μας ενσωματώνουν άμεσα τονHttpClient
με τη χρήση τουinject
.
-
-
Δημιουργία του
HttpClientExampleComponent
για την επίδειξη της λειτουργίας τουHttpClient
μέσω τουJokesService
:
ng g c components/http-client-example
- Έλεγχος του τύπου των δεδομένων που επιστρέφουν οι κλήσεις των API με το
console.log
.
import { Component, inject } from "@angular/core";
import { JokesService } from "src/app/shared/services/jokes.service";
@Component({
selector: "app-http-client-example",
standalone: true,
imports: [],
templateUrl: "./http-client-example.component.html",
styleUrl: "./http-client-example.component.css",
})
export class HttpClientExampleComponent {
jokesService = inject(JokesService);
ngOnInit(): void {
this.jokesService.getDadJoke().subscribe((data) => {
console.log(data);
});
this.jokesService.getChuckNorrisJoke().subscribe((data) => {
console.log(data);
});
}
}
- Δημιουργία των Interfaces
DadJoke
καιChuckNorrisJoke
στο αρχείοshared/interfaces/jokes.ts
:
export interface DadJoke {
joke: string;
}
export interface ChuckNorrisJoke {
value: string;
}
- Χρήση των interfaces για casting στον HttpClient:
getDadJoke() {
return this.http.get<DadJoke>(DAD_JOKES_API_URL, {
headers: {
Accept: 'application/json',
},
});
}
getChuckNorrisJoke() {
return this.http.get<ChuckNorrisJoke>(JACK_NORRIS_JOKES_API_URL, {
headers: {
Accept: 'application/json',
},
});
}
-
Από το σημείο αυτό και στο εξής είναι απαραίτητο να έχετε εγκαταστήσει τη γλώσσα Python και να χρησιμοποιείτε το Python-Flask backend από το repository angular-introduction-backend.
-
Δημιουργία των enviroments με την εντολή:
ng generate environments
-
Ενημέρωση των αρχείων
environment.development.ts
καιenvironment.ts
-
Δημιουργία του
User
interface στο αρχείοshared/interfaces/mongo-backend.ts
:export interface User { givenName: string; surName: string; email: string; password: string; }
-
Δημιουργία του
UserService
με την εντολή:ng generate service shared/services/user
- Η μέθοδος
registerUser
αποστέλλει στο backend τα πλήρη δεδομένα που αφορούν στην εγγραφή ενός νέου χρήστη - Η μέθοδος
check_duplicate_email
ρωτά το backend αν τοemail
που λαμβάνει σαν όρισμα χρησιμοποιείται ήδη σε κάποια εγγραφή στη βάση.
- Η μέθοδος
-
Δημιουργία του
UserRegistrationComponent
που υλοποιεί μια reactive form για τη διαδικασία του registration:- Χρησιμοποιεί το
UserService
με τη χρήση τουinject
, - Αρχικοποιεί το
registrationStatus
- Ορίζει τη φόρμα του registration με δύο πεδία για το password που θα πρέπει να λάβουν από το χρήστη ακριβώς το ίδιο περιεχόμενο
- Δεύτερο όρισμα στον ορισμό της φόρμας μέσω του
FormGroup
είναι ο συνολικός validator της φόρμας, στην περίπτωσή μας η μέθοδος της κλάσης που εξετάζει αν τα δύο password input συμπίπτουν.- Στην περίπτωση εντοπισμού λάθους σε κάποιο input, ο Validator επιστρέφει ένα object με ένα κλειδί ενδεικτικό του λάθους
- To κλειδί αυτό μπορεί μετά να χρησιμοποιηθεί στο template για να εμφανιστεί κατάλληλο μήνυμα λάθους
- Χρησιμοποιεί το
-
Στην υποβολή της φόρμας χρησιμοποιείται το
UserService
για να υποβάλλει στo backend τα δεδομένα της φόρμας. Η εγγραφή στην απάντηση του backend ξεχωρίζει τις περιπτώσεις της απάντησης του backend με τα callbacks στα χαρακτηριστικάnext
καιerror
:next
: το callback που καλείται όταν στο backend στείλει HTTP response20*
.error
: το callback που καλείται όταν το backend στείλει HTTP reponse40*
ή50*
.- Ανάλογα θέτουμε το
registrationStatus
για να έχουμε τον αντίστοιχο έλεγχο στο template.
-
Χρήση του backend για τον έλεγχο ύπαρξης duplicate email στη βάση και χρήση της πληροφορίας κατά το event
blur
για να γίνει το πεδίο email invalid.