Web/Mobile/ Progressive
App Development - Part 3
ECMAscript 6 & Typescript
Super set di JavaScript, contiene funzionalità già proprie di ES6, con la possibilità di aggiungere variabili tipizzate.
Angular 2+
Angular è un framework per lo sviluppo di web app, basato sulla logica a componenti.
Creato in collaborazione tra Google e Microsoft.
Apache Cordova & IonicFramework 3+
Ionic Framework è un framework per la creazione di interfacce mobile/desktop, basato su TS / ng2+. Con l'aiuto di Apache Cordova, andremo a creare un app mobile ibrida.
App con i tabs
Ionic Templates
// tabs.html
<ion-tabs>
<ion-tab-bar slot="bottom">
<ion-tab-button tab="tab1">
<ion-icon name="flash"></ion-icon>
<ion-label>Tab One</ion-label>
</ion-tab-button>
</ion-tab-bar>
</ion-tabs>
// tabs-routing.ts
// Imports
const routes: Routes = [{
path: 'tabs',
component: TabsPage,
children: [
{
path: 'tab1',
children: [
{
path: '',
loadChildren: () => import('../tab1/tab1.module').then(m => m.Tab1PageModule)
}
]
},
]
}];
Ionic ci mette a disposizione anche una serie di template già pronti all'uso, quelli che useremo di più sono i tabs e il sidemenu.
Caricare dati con il servizio HTTP
ionic generate service planet
// planet.service.ts
import { Injectable } from '@angular/core';
import { HttpClient } from '@angular/common/http';
import { environment } from '../environments/environment';
import { Planet } from '../models/planet';
import { map } from 'rxjs/operators';
@Injectable({
providedIn: 'root'
})
export class PlanetService {
public data: any = null;
constructor(public http: HttpClient) {
console.log('Hello Planet Service');
}
public getPlanets() {
return this.http
.get<Planet[]>(`${environment.apiUrl}/planets/?format=json`, { } )
.pipe( map( response => {
// @ts-ignore
if ( response ) {
// @ts-ignore
response.results.forEach( (el) => {
let url = el.url.replace('https://swapi.co/api/planets/', '');
url = url.replace('/', '');
el.id = url;
});
// @ts-ignore
return response.results;
}
}));
}
}
Creiamo il modello planet.ts
// app/models/planet.ts
export interface Planet {
id?: number;
name: string;
rotation_period: number;
orbital_period: number;
diameter: number;
climate: string;
gravity: string;
terrain: string;
surface_water: number;
population: number;
residents: any;
films: any;
created: string;
edited: string;
url: string;
}
Modifichiamo i tabs
// tabs.html
<ion-tab-button tab="tab1">
<ion-icon name="planet"></ion-icon>
<ion-label>Pianeti</ion-label>
</ion-tab-button>
// Modifichiamo tab1.ts
import { Component, OnInit } from '@angular/core';
import { Planet } from '../../models/planet';
import { PlanetService } from '../../services/planet.service';
@Component({
selector: 'app-tab1',
templateUrl: 'tab1.page.html',
styleUrls: ['tab1.page.scss']
})
export class Tab1Page implements OnInit {
public planets: Planet[] = [];
constructor(
private planetService: PlanetService
) { }
public ngOnInit() {
this.planetService
.getPlanets()
.subscribe( result => {
this.planets = result;
});
}
}
Rinominiamo il tab.
Il template di tab1.html
// tab1.html
<ion-header>
<ion-toolbar>
<ion-title>
Pianeti
</ion-title>
</ion-toolbar>
</ion-header>
<ion-content>
<ion-list>
<ion-item *ngFor="let planet of planets"
[href]="'/tabs/tab1/planet-detail/' + planet.id" [detail]="true">
<h2>{{ planet.name }}</h2>
</ion-item>
</ion-list>
</ion-content>
Aggiungiamo la pagina di dettaglio del pianeta.
ionic generate page PlanetDetail
// planet-detais.ts
import { Component, OnInit } from '@angular/core';
import { ActivatedRoute } from '@angular/router';
import { Planet } from '../../models/planet';
import { PlanetService } from '../../services/planet.service';
@Component({
selector: 'app-planet-detail',
templateUrl: './planet-detail.page.html',
styleUrls: ['./planet-detail.page.scss'],
})
export class PlanetDetailPage implements OnInit {
planetId: number;
planet: Planet;
constructor(
private route: ActivatedRoute,
private planetService: PlanetService
) { }
ngOnInit() {
this.route.params.subscribe( d => {
this.planetId = d.planetId;
this.planetService.getPlanet( this.planetId )
.subscribe( d => { this.planet = d });
});
}
}
Template della pagina di dettaglio del pianeta.
<ion-header>
<ion-toolbar>
<ion-title>Dettaglio Pianeta</ion-title>
</ion-toolbar>
</ion-header>
<ion-content>
<ion-card>
<ion-card-header>
<ion-card-subtitle>{{ planet?.name }}</ion-card-subtitle>
</ion-card-header>
<ion-card-content>
{{ planet | json }}
</ion-card-content>
</ion-card>
</ion-content>
Aggiungiamo la ricerca
// tab1.html
<ion-searchbar [(ngModel)]="searchQuery"
showCancelButton="focus" cancelButtonText="Resetta"
(ionInput)="searchPlanets($event)"
(ionClear)="getPlanets()"></ion-searchbar>
// tab1.ts
export class Tab1Page {
public searchQuery: string;
public ngOnInit() {
this.searchQuery = '';
this.getPlanets();
}
public getPlanets() {
this.planetService
.getPlanets()
.subscribe( result => {
this.planets = result;
});
}
public searchPlanets(event) {
// la ricerca prende il valore dalla search bar
const queryString = event.target.value;
if ( queryString !== undefined ) {
// se il valore è vuoto, non fare la ricerca
if ( queryString.trim() === '' ) {
return;
}
this.planetService
.searchPlanets( queryString )
.subscribe( result => {
this.planets = result;
});
}
}
}
Modifichiamo il service
// planet.service.ts
export class PlanetService {
[...]
public searchPlanets(queryString) {
return this.http
.get<Planet[]>(`${environment.apiUrl}/planets/?search=${queryString}&format=json`, { } )
.pipe( map( response => {
// @ts-ignore
if ( response ) {
response = this._parseResults( response );
// @ts-ignore
return response.results;
}
}));
}
}
Virtual Scrolling
// tab1.html
<ion-virtual-scroll [items]="planets" approxItemHeight="60px">
<ion-item *virtualItem="let planet; let planetBounds = bounds;">
[...]
</ion-item>
</ion-virtual-scroll>
Uno dei problemi di performance che si verificano spesso, è dovuto a lunghe liste di contenuti (non è il caso dei pianeti in questione), ma quando i contenuti superano i 400, inizieremo a vedere rallentamenti e strani comportamenti.
Ci viene in aiuto il Virtual Scrolling, che rendererizza (cioè crea elementi html) solo degli elementi che dobbiamo visualizzare. Vediamo come si fa.
Titoli liste
// tab1.html
<ion-virtual-scroll [items]="planets" approxItemHeight="60px" [headerFn]="customHeaderFn">
<ion-item-divider *virtualHeader="let header">
{{ header }}
</ion-item-divider>
[...]
</ion-virtual-scroll>
// tab1.ts
public customHeaderFn(record, recordIndex, records) {
if ( recordIndex > 0 ) {
if ( record.name.charAt(0) !== records[recordIndex - 1].name.charAt(0) ) {
return record.name.charAt(0);
} else {
return null;
}
} else {
return record.name.charAt(0);
}
}
Il virtual scrolling permette inoltre di utilizzare header e footer dinamici.
Useremo il titolo separatore mostrando la prima lettera in ordine alfabetico per separare.
Debug & Testing
Chrome Inspector
Lo strumento DevTools di Chrome è il primo strumento che useremo per fare sviluppo e testing di quello che stiamo facendo.
https://github.com/FMRiggio/ionicWars
https://www.dev2qa.com/how-to-set-android-sdk-path-in-windows-and-mac/
Web/Mobile/Progressive Web App Development - Part 3
By Filippo Matteo Riggio
Web/Mobile/Progressive Web App Development - Part 3
- 389