Esta semana he estado trabajando en una aplicación con ionic para un cliente.
La aplicación funcionaba estupendamente en todos los dispositivos Android donde la he probado, evidentemente en dispositivos más antiguos el funcionamiento era un poco menos fluido pero aceptable en cualquier caso, sin embargo a la hora de probar la app en un dispositivo iOS, concretamente en un iPhone 6 que tengo para probar las apps en un dispositivo real me he llevado una decepción.
La aplicación tiene que mostrar una lista con imágenes e información bastante grande, además tiene un buscador para filtrar el listado por varios campos.
El scroll no iba muy fino a pesar de utilizar [virtualScroll] para el listado, cuando filtraba el listado por algunos campos a la hora de refrescar el contenido del listado iba a trompicones y tardaba en responder… en fin, la experiencia de usuario dejaba bastante que desear.
Lo primero que he hecho es pulir el código todo lo que he podido intentando que sea lo más eficiente posible, pero no ha sido suficiente.
Investigando un poco en el blog oficial de ionic me he encontrado con WKWebView.
Podéis leer la entrada del blog de ionic en este enlace:
Como sabemos ionic utiliza apache cordova que a su vez utiliza la webview del sistema para mostrar el contenido de nuestra aplicación.
Actualmente, la plataforma iOS proporciona dos webviews diferentes.
Está la webview más antigua (y más lenta) llamada «UIWebView» y otra mas nueva llamada «WKWebView«.
El navegador web predeterminado de iOS es Safari, internamente Safari utiliza WKWebView, sin embargo ionic debido a diversas incompatibilidades y problemas técnicos utiliza UIWebView.
WKWebView es más rápida que UIWebView, además Apple proporciona actualizaciones en cada versión de iOS.
Para utilizar WKWebView solo tenemos que instalar en nuestra aplicación un plugin que proporciona ionic-team.
Para instalar el plugin desde consola ejecutamos el siguiente comando:
Solo con instalar este plugin mi aplicación iba muchísimo mas fluida en mi iPhone.
Probad vuestra aplicación con este plugin y si todo va bien perfecto, sin embargo yo tuve problemas con CORS (Cross Origin Resource Sharing) al hacer peticiones al servidor ya que yo no tenia acceso para cambiar nada en el servidor y por lo tanto no podía cambiar las cabeceras de respuesta del servidor para aceptar peticiones cross origin.
Después en app.module.ts tendríamos que añadir lo siguiente:
import { NgModule } from '@angular/core';
import { NativeHttpFallback, NativeHttpModule } from 'ionic-native-http-connection-backend';
import { RequestOptions, Http } from '@angular/http';@NgModule({ declarations:[], imports:[ NativeHttpModule], bootstrap:[], entryComponents:[], providers:[{provide: Http, useClass: Http, deps:[NativeHttpFallback, RequestOptions]}],})
export class AppModule {}
Una vez hecho esto ya podemos realizar peticiones http sin problemas, si quieres saber como relizar peticiones http con ionic puedes consultar el siguiente post: Tutorial de Ionic – Peticiones http – API REST
Con estas dos cosas he conseguido que la aplicación funcione correctamente y con una fluidez mas que aceptable en iOS.
Bueno, por hoy aquí lo dejo con estos pequeños consejos, espero que os sea útil.
El el post anterior vimos como autenticar un usuario en firebase con email y contraseña, hoy vamos ha ver como utilizar Firebase Database para guardar nuestros sitios en la nube.
Para ello vamos a crear un provider para gestionar nuestros sitios en firebase Database, por lo tanto desde consola nos situamos dentro de la carpeta de nuestro proyecto y creamos en nuevo provider:
ionic g provider firebaseDb
Ahora editamos el archivo firebase-db.ts que se acaba de generar dentro de la carpeta providers/firebase-db, eliminamos el import Http y rxjs/add/operator/map e importamos AngularFireDatabase,FirebaseListObservable y nuestro provider AuthProvider quedando de la siguiente manera:
import{ Injectable } from '@angular/core';import{ AngularFireDatabase, AngularFireList } from 'angularfire2/database';
import { AuthProvider } from '..auth/auth';/* Generated class for the FirebaseDbProvider provider. See https://angular.io/docs/ts/latest/guide/dependency-injection.html for more info on providers and Angular DI.*/@Injectable()exportclass FirebaseDbProvider {
constructor(publicafDB: AngularFireDatabase, public auth: AuthProvider){
console.log('Hello FirebaseDbProvider Provider');}}
Como necesitamos el id del usuario lo primero que vamos ha hacer es añadir la función getUser al provider auth que creamos en el post anterior, por lo tanto editamos auth.ts y añadimos esta función:
// Obtenemos el id de usuario.
getUser(){returnthis.afAuth.auth.currentUser.uid;}
Bien, ahora que ya podemos obtener el id de usuario vamos a añadir en el archivo firebase-db.ts un método para guardar nuestros sitios en Firebase database:
Como vemos la función recibe como parámetros sitio que será un objeto con los datos de nuestro sitio.
Al objeto sitio le añadimos un campo id para identificarlo y así poder luego modificarlo. Como necesitamos que el id sea diferente cada vez vamos a utilizar Date.now() que nos devuelve los milisegundos transcurridos desde el 1 de enero de 1970, con esto nos aseguramos que no se repita el id, a no ser que seas capaz de guardar dos sitios en menos de un milisegundo ;-P.
En firebase se guarda la información con estructura de árbol en formato JSON. Para acceder a ella tenemos que hacer referencia a la «ruta» a la que queremos acceder.
En este caso le estamos diciendo que guarde nuestro sitio con esta estructura ‘sitio/_id_usuari_/_id_sitio_/_sitio_’.
El id de usuario lo obtenemos con la función que acabamos de definir en AuthProvide this.auth.getUser().
Dentro de sitio colgarán los diferentes id de usuarios de los cuales a su vez colgarán los diferentes sitios de cada usuarios.
Para verlo mas claro en el modal nuevo-sitio vamos a modificar el método guardarSitio para que en lugar de guardar el sitio en la base de datos local SQlite lo guarde en firebase y podamos ver la estructura de como se guarda la información en firebase.
Editamos modal-nuevo-sitio.ts y hacemos los siguientes cambios:
import{ Component } from '@angular/core';import{ IonicPage, NavController, NavParams, ViewController } from 'ionic-angular';import{ Camera, CameraOptions } from '@ionic-native/camera';// import { DbProvider } from '../../providers/db/db';import{ FirebaseDbProvider } from '../../providers/firebase-db/firebase-db';/** * Generated class for the ModalNuevoSitioPage page. * * See http://ionicframework.com/docs/components/#navigation for more info * on Ionic pages and navigation.*/@IonicPage()@Component({selector:'page-modal-nuevo-sitio',templateUrl:'modal-nuevo-sitio.html',})exportclass ModalNuevoSitioPage {coords :any={lat:0,lng:0}address:string;description:string='';foto:any='';
constructor(publicnavCtrl: NavController,publicnavParams: NavParams,privateviewCtrl : ViewController,privatecamera: Camera,// private db: DbProviderprivatedbFirebase :FirebaseDbProvider,){}
ionViewDidLoad(){
console.log('ionViewDidLoad ModalNuevoSitioPage');this.coords.lat =this.navParams.get('lat');this.coords.lng =this.navParams.get('lng');this.getAddress(this.coords).then(results=>{this.address = results[0]['formatted_address'];}, errStatus =>{// Aquí iría el código para manejar el error});}
cerrarModal(){this.viewCtrl.dismiss();}getAddress(coords):any{
var geocoder =new google.maps.Geocoder();returnnew Promise(function(resolve, reject){
geocoder.geocode({'location': coords},function(results, status){// llamado asincronamenteif(status == google.maps.GeocoderStatus.OK){
resolve(results);}else{
reject(status);}});});}
sacarFoto(){
let cameraOptions : CameraOptions ={quality:50,encodingType:this.camera.EncodingType.JPEG,targetWidth:800,targetHeight:600,destinationType:this.camera.DestinationType.DATA_URL,sourceType:this.camera.PictureSourceType.CAMERA,correctOrientation:true}this.camera.getPicture(cameraOptions).then((imageData)=>{// imageData is a base64 encoded stringthis.foto ="data:image/jpeg;base64,"+ imageData;},(err)=>{
console.log(err);});}
guardarSitio(){
let sitio ={lat:this.coords.lat,lng:this.coords.lng ,address:this.address,description:this.description,foto:this.foto
}this.dbFirebase.guardaSitio(sitio).then(res=>{
console.log('Sitio guardado en firebase:');
this.cerrarModal();
})}}
Como ahora vamos a guardar nuestros sitios en firebase he comentado el import de DbProvider ya que ahora no lo vamos a utilizar, y en la función guardarSitio hemos sustituido la linea this.db.addSitio… por this.dbFirebase.guardaSitio…
Como vemos llamamos a la función guardaSitio del provider FirebaseDbProvider que hemos creado más arriba y le pasamos como parámetro el objeto sitio con los datos de nuestro sitio.
Ahora ejecutamos nuestra aplicación y vamos a la consola de firebase, antes de guardar ningún sitio si seleccionamos Database en el menú de la izquierda de la consola de firebase veremos algo como esto:
Ahora vamos a guardar un nuevo sitio desde nuestra app, rellenamos los campos del formulario del modal nuevo sitio y le damos a guardar, si todo ha ido bien ahora en la consola de firebase veremos algo como esto (pulsa en el icono ‘+’ para desplegar los campos):
Como vemos los datos de guardan en una estructura de árbol, en el primer nivel esta sitio, de sitio ‘cuelga’ los id de usuario, en este caso solo tenemos el nuestro pero si distribuís la aplicación por cada usuario habrá un nodo, de cada usuario ‘cuelgan’ los ids de cada sitio, y de cada id a su vez cuelgan los campos del sitio.
Ahora que ya podemos guardar nuestros sitios en firebase vamos a ver como podemos obtener todos los sitios que tenemos guardados para mostrarlos en el listado:
Lo primero que vamos ha hacer es crear una función en firebase-db.ts para obtener el listado de sitios guardados en firebase database:
Para obtener el listado de sitios guardados utilizamos el método list de AngularFireDatabase pasando como parámetro la ruta a partir de la cual queremos obtener los datos, en este caso queremos obtener todo lo que cuelgue de sitios/id_usuario, es decir todos los sitios de nuestro usuario, el id de usuario una vez más lo obtenemos con this.auth.getUser() que hemos creado en nuestro provider AuthProvider.
Con .valueChanges() devolvemos un observable cuando se produzcan cambios en la base de datos.
Bien, llegados a este punto vamos a modificar el controlador de la página listado para que en lugar de obtener los datos de la base de datos local los obtenga directamente de firebase.
Editamos listado.ts e importamos FirebaseDbProvider:
import{ Component } from '@angular/core';import{ AlertController, IonicPage, NavController, NavParams, ModalController } from 'ionic-angular';import{ DbProvider } from '../../providers/db/db';import{ FirebaseDbProvider } from '../../providers/firebase-db/firebase-db';/** * Generated class for the ListadoPage page. * * See http://ionicframework.com/docs/components/#navigation for more info * on Ionic pages and navigation.*/@IonicPage()@Component({selector:'page-listado',templateUrl:'listado.html',})exportclass ListadoPage {sitios:any;
constructor(publicnavCtrl: NavController,publicnavParams: NavParams,publicdb : DbProvider,publicmodalCtrl : ModalController,publicalertCtrl : AlertController,publicdbFirebase :FirebaseDbProvider,){}
Ahora en la función ionViewDidEnter() vamos a sustituir this.db.getSitios().then((res)=>{…}) por lo siguiente:
La función getSitios() que como hemos visto a su vez llama al método list de AngularFireDatabase nos devuelve un observable, por lo que nos suscribimos al resultado. Aquí podemos ver claramente la diferencia entre un observable y una promesa.
Guardamos en this.sitios la lista de sitios que obtenemos lo que hará que se refresque automáticamente en la vista del listado.
Como ya sabemos una promesa ejecuta lo que tengamos definido en then(res=>{ …}) una vez tenga listo el resultado, pero esto se ejecuta una única vez, sin embargo un observable va a ejecutar lo que tengamos definido en .subscribe(res=>{…}) cada vez que haya un cambio en el resultado. Por ejemplo si desde la consola de firebase cambiamos a mano el campo description de nuestro sitio, este se verá automáticamente reflejado en nuestra aplicación, es interesante hacer la prueba.
En este momento ya podemos guardar sitios en firebase y mostrarlos en el listado:
Lo siguiente que vamos ha hacer es la modificación de los sitios guardados.
Para ello primero vamos ha hacer una pequeña modificación a la función guardaSitio() de nuestro provider firebase-db.ts:
Hemos añadido un if para comprobar si el sitio que recibimos en la función tiene el campo id definido.
Como utilizamos el id del sitio para establecer la ruta del registro, si el sitio que recibimos para guardar no tiene id significa que es un sitio nuevo y entonces le damos un id, si ya tiene un id significa que es un sitio que ya existe y hay que modificar.
Modificar un registro en firebase se hace exactamente igual que crear uno nuevo, si la ruta a la que hacemos referencia no existe crea el registro, si ya existe entonces modifica el registro existente en firebase.
Ahora vamos a modificar el archivo modal-detalle-sitio.ts para hacer que al guardar los cambios al editar un sitio existente se guarden los cambios en firebase, para ello vamos a importar FirebaseDbProvider para poder llamar a la función guardaSitio que acabamos de modificar, y vamos también a modificar la función guardarCambios para que guarde los cambios en firebase en lugar de en la base de datos local:
import{ Component } from '@angular/core';import{ IonicPage, NavController, NavParams, ViewController } from 'ionic-angular';import{ LaunchNavigator } from '@ionic-native/launch-navigator';import{ Camera, CameraOptions } from '@ionic-native/camera';// import { DbProvider } from '../../providers/db/db';import{ FirebaseDbProvider } from '../../providers/firebase-db/firebase-db';/** * Generated class for the ModalDetalleSitioPage page. * * See http://ionicframework.com/docs/components/#navigation for more info * on Ionic pages and navigation.*/@IonicPage()@Component({selector:'page-modal-detalle-sitio',templateUrl:'modal-detalle-sitio.html',})exportclass ModalDetalleSitioPage {sitio:any;edit : boolean =false;
constructor(publicnavCtrl: NavController,publicnavParams: NavParams,privateviewCtrl : ViewController,privatelaunchNavigator : LaunchNavigator,privatecamera: Camera,// private db: DbProvider,privatedbFirebase :FirebaseDbProvider){this.sitio =this.navParams.data;}
ionViewDidLoad(){
console.log('ionViewDidLoad ModalDetalleSitioPage');}
cerrarModal(){this.viewCtrl.dismiss();}
comoLlegar(){
let destino =this.sitio.lat+','+this.sitio.lng;this.launchNavigator.navigate(destino).then(
success => console.log('Launched navigator'),
error => console.log('Error launching navigator', error));}
editar(){this.edit =true;}
sacarFoto(){
let cameraOptions : CameraOptions ={quality:50,encodingType:this.camera.EncodingType.JPEG,targetWidth:800,targetHeight:600,destinationType:this.camera.DestinationType.DATA_URL,sourceType:this.camera.PictureSourceType.CAMERA,correctOrientation:true}this.camera.getPicture(cameraOptions).then((imageData)=>{// imageData is a base64 encoded stringthis.sitio.foto ="data:image/jpeg;base64,"+ imageData;},(err)=>{
console.log(err);});}guardarCambios(){
let sitio ={id :this.sitio.id,lat:this.sitio.lat,lng:this.sitio.lng ,address:this.sitio.address,description:this.sitio.description,foto:this.sitio.foto
}this.dbFirebase.guardaSitio(sitio).then(res=>{
console.log('Sitio modificado en firebase');this.cerrarModal();})}}
Hemos comentado el import de DbProvider porque ya no lo estamos utilizando.
Ahora si pruebas la aplicación con el panel de firebase abierto podrás ver que si modificas un sitio que tengas guardado automáticamente se verá reflejado este cambio en el registro de firebase.
Para concluir solo nos queda eliminar sitios.
Vamos a añadir a FirebaseDbProvider en el archivo firebase-db.ts una función para eliminar un sitio de la base de datos de firebase:
public borrarSitio(id){
this.afDB.database.ref('sitios/'+this.auth.getUser()+'/'+id).remove();
}
Como puedes ver es muy sencillo, solo necesitamos recibir el id del sitio que queremos eliminar y haciendo referencia a la ruta de nuestro sitio (que una vez más es sitios/_id_usuario_/id_sitio) utilizamos la función remove() para eliminar el sitio.
Ahora en el listado solo tenemos que sustituir la llamada a borrarSitio() de la base de datos local de DbProvider por la función que acabamos de crear en FirebaseDbProvider, por lo tanto editamos el archivo listado.ts y dejamos la función borrarSitio() de la siguiente manera:
borrarSitio(id){
let alert =this.alertCtrl.create({
title:'Confirmar borrado',
message:'¿Estás seguro de que deseas eliminar este sitio?',
buttons:[{
text:'No',
role:'cancel',
handler:()=>{// Ha respondido que no así que no hacemos nada}},{
text:'Si',
handler:()=>{// AquÍ borramos el sitio en firebasethis.dbFirebase.borrarSitio(id);}}]});
alert.present();}
Como podemos observar ya no necesitamos obtener de nuevo los sitios una vez borrado para que se refresque el listado ya que al ser firebase una base de datos en tiempo real el listado se actualiza automáticamente.
Podéis probar a borrar un sitio y veréis como el sitio se elimina automáticamente en el panel de firebase y el listado se actualiza.
Eso es todo por hoy, con esto ya podemos hacer muchas cosas interesantes utilizando firebase, seguro que se os ocurren grandes ideas para realizar apps utilizando firebase como backend. Podéis dejarme en los comentarios esas grandes ideas, no se lo contaré a nadie ;-P
Hola a todos, en el último post vimos como comunicarnos con el backend a través de peticiones http, Hoy vamos a aprender a utilizar Firebase como backend.
Firebase es una plataforma de Google que nos ofrece un BaaS (Backend as a Service), lo que nos permite librarnos de la tarea de tener nuestro propio servidor y programar toda la lógica del backend, autenticación de usuarios, etc, ahorrando tiempo e infraestructura.
Firebase nos permite tener una base de datos en la nube y ver los cambios que se produzcan en tiempo real, entre otros servicios también ofrece almacenamiento de archivos en la nube, mensajería, hosting y autenticación de usuarios.
Para aprender a integrar firebase con Ionic vamos a retomar nuestra App de guardar sitios y vamos ha hacer que nuestros sitios se guarden en la base de datos de firebase para tenerlos almacenados en la nube en lugar de localmente.
Para ello tenemos también que crear un usuario y autenticarnos para que en la base de datos cada usuarios tenga sus sitios guardados.
Vamos a empezar por la autenticación de usuarios, pero antes de nada debemos crear una aplicación en firebase.
Para crear una aplicación en firebase debemos acceder a la consola de firebase desde la siguiente url:
Debemos estar logeados con nuestra cuenta de google, si no tienes cuanta de google tendrás que crearte una.
Para acceder a la consola seleccionamos la opción «IR A LA CONSOLA» que se encuentra en la parte superior derecha como se muestra en la siguiente imagen.
Una vez en la consola debemos añadir un nuevo proyecto, por lo tanto seleccionamos añadir proyecto y nos saldrá una ventana para introducir el nombre del proyecto y la región:
En este caso he puesto como nombre Mis Sitios y como región España.
Pulsamos el botón CREAR PROYECTO y accederemos al panel de nuestra aplicación donde tememos varias opciones.
Dejamos abierta la pestaña del navegador con el panel de firebase ya que mas tarde tenemos que volver para obtener los datos de configuración, pero ahora vamos a volver a nuestra aplicación Mis Sitiosy vamos a instalar firebase y angularfire2.
Abrimos una consola y nos situamos dentro del directorio de nuestra aplicación y escribimos el siguiente comando:
npm install angularfire2 firebase --save
Ahora editamos el archivo app.module.ts e importamos los y declaramos los módulos AngularFireModule, AngularFireDatabaseModule y AngularFireAuthModule, también declaramos la constante firebaseConfig que utilizamos en la declaración AngularFireModule.initializeApp(firebaseConfig) :
import{ BrowserModule } from '@angular/platform-browser';import{ ErrorHandler, NgModule } from '@angular/core';import{ IonicApp, IonicErrorHandler, IonicModule } from 'ionic-angular';import{ SplashScreen } from '@ionic-native/splash-screen';import{ StatusBar } from '@ionic-native/status-bar';import{ MyApp } from './app.component';import{ Geolocation } from '@ionic-native/geolocation';import{ Camera } from '@ionic-native/camera';import{ DbProvider } from '../providers/db/db';import{ SQLite } from '@ionic-native/sqlite';import{ LaunchNavigator } from '@ionic-native/launch-navigator';import{ AuthProvider } from '../providers/auth/auth';import{ AngularFireModule } from 'angularfire2';import{ AngularFireDatabaseModule } from 'angularfire2/database';import{ AngularFireAuthModule } from 'angularfire2/auth';exportconst firebaseConfig ={apiKey:"xxxxxxxxxx",authDomain:"your-domain-name.firebaseapp.com",databaseURL:"https://your-domain-name.firebaseio.com",storageBucket:"your-domain-name.appspot.com",messagingSenderId:'<your-messaging-sender-id>'};@NgModule({declarations:[
MyApp
],imports:[
BrowserModule,
IonicModule.forRoot(MyApp),AngularFireModule.initializeApp(firebaseConfig),
AngularFireDatabaseModule,
AngularFireAuthModule],bootstrap:[IonicApp],entryComponents:[
MyApp
],providers:[
StatusBar,
SplashScreen,{provide: ErrorHandler,useClass: IonicErrorHandler},
Geolocation,
Camera,
DbProvider,
SQLite,
LaunchNavigator,
AuthProvider
]})exportclass AppModule {}
Los valores que hemos puesto en el objeto firebaseConfig son provisionales, ahora debemos sustituirlos por los que nos de la consola de Firebase por lo tanto regresamos a la web de Firebase.
Aunque nuestra aplicación es una aplicación para móviles estamos utilizando tecnologías web para su desarrollo ya que no es una app nativa sino híbrida (revisar el primer post de este tutorial para saber más sobre tipo de aplicaciones), por lo tanto vamos a seleccionar Añade Firebase a tu aplicación web para ver las variables de configuración que luego vamos a necesitar:
Al seleccionar añade Firebase a tu aplicación web mostrará algo como esto:
Por lo tanto en nuestro archivo app.module.ts sustituimos los valores de apiKey, authDomain, databaseURL, projectId, storageBucket y messagingSenderId del objeto firebaseConfig por los que nos ha generado.
Una vez copiado los valores cerramos la ventana.
Firebase nos da diferentes métodos para registrar usuarios:
Correo electrónico/contraseña
Teléfono
Google
Facebook
Twitter
GitHub
Anónimo
Para este ejemplo vamos a utilizar el método Correo electrónico/contraseña.
El panel de la izquierda de nuestro proyecto en firebase seleccionamos «Authentication» y «METODO DE INICIO DE SESIÓN«, y hacemos click en Correo electrónico/contraseña como muestro en la siguiente imagen:
El la ventana que nos muestra habilitamos y le damos a Guardar:
Bien, ya tenemos configurado el método de registro de usuarios en firebase.
Ahora lo primero que necesitamos es un provider que utilizaremos para autenticar usuarios.
Por lo tanto abrimos una consola escribimos el siguiente comando que ya conocemos para crear un provider:
ionic g provider auth
En el provider que acabamos de crear eliminamos los imports de Http y rxjs/add/operator/map ya que no los vamos a necesitar, e importamos AngularFireAuth y firebase:
import{ Injectable } from '@angular/core';import{ AngularFireAuth } from 'angularfire2/auth';import* as firebase from 'firebase/app';/* Generated class for the AuthProvider provider. See https://angular.io/docs/ts/latest/guide/dependency-injection.html for more info on providers and Angular DI.*/@Injectable()exportclass AuthProvider {
constructor(privateafAuth : AngularFireAuth){
console.log('Hello AuthProvider Provider');}}
Ahora vamos a crear una nueva página de login para que el usuario pueda darse de alta y autenticarse, para ello desde la consola escribimos el siguiente comando:
ionic g page login
Esto nos habrá generado una nueva página llamada login, vamos a editar el archivo login.html para crear un formulario para autenticar y/o dar de alta a un usuario:
<!-- Generated template for the LoginPage page. See http://ionicframework.com/docs/components/#navigation for more info on Ionic pages and navigation.--><ion-header><ion-navbar><ion-title>login</ion-title></ion-navbar></ion-header><ion-content><ion-list><form><ion-item><ion-label floating>Email</ion-label><ion-input [(ngModel)]="user.email"name="email"type="text" autocapitalize="off"required></ion-input></ion-item><ion-item><ion-label floating>Contraseña</ion-label><ion-input [(ngModel)]="user.password"name="password"type="password"required></ion-input></ion-item><div padding><button ion-button (click)="login()"type="submit"color="primary" block>
Login
</button><button ion-button (click)="signin()"type="submit"color="secondary" block>
Dar de alta
</button></div></form></ion-list></ion-content>
Para simplificar vamos a utilizar el mismo formulario tanto para dar de alta a un usuario como para autenticarnos.
Como podemos ver tenemos un formulario con dos inputs, uno para el email y otro para la contraseña.
Al label de los inputs le decimos que sea de tipo floating, esto produce un efecto de desplazamiento del label al situarnos en el campo. Para saber más sobre el elemento ion-input y sus posibilidades podéis consultar el componente ion-input en la documentación oficial de ionic.
Tendremos una variable miembro llamada user que es un objeto que va a contener el email y la contraseña del usuario, si os fijáis en el campo email hacemos un data binding con [(ngModel)]= «user.email», lo mismo en el campo password con [(ngModel)]=»user.password».
Por lo tanto los cambios que hagamos en estos campos serán reflejados automáticamente en user.email y user.password.
Lo primero que haremos va a ser definir la variable user en el controlador. Abrimos el archivo login.ts y añadimos lo siguiente:
import{ Component } from '@angular/core';import{ IonicPage, NavController, NavParams } from 'ionic-angular';/** * Generated class for the LoginPage page. * * See http://ionicframework.com/docs/components/#navigation for more info * on Ionic pages and navigation.*/@IonicPage()@Component({selector:'page-login',templateUrl:'login.html',})exportclass LoginPage {user={email :'',password :''};
constructor(publicnavCtrl: NavController,publicnavParams: NavParams){}
ionViewDidLoad(){
console.log('ionViewDidLoad LoginPage');}}
En el formulario tenemos dos botones, uno para dar hacer login y otro para darnos de alta.
Como todavía no tenemos ningún usuario creado lo primero que vamos ha hacer es el alta de usuario.
Lo primero que necesitamos hacer es crear en el provider authProvider una función para dar de alta a usuarios. Editamos el archivo auth.ts y añadimos la función registerUser:
// Registro de usuario
registerUser(email:string,password:string){returnthis.afAuth.auth.createUserWithEmailAndPassword( email, password).then((res)=>{
// El usuario se ha creado correctamente.
}).catch(err=>Promise.reject(err))}
Como podemos observar mediante el método createUserWithEmailAndPassword pasándole como parámetros el email y password creamos un nuevo usuario.
Si nos fijamos el botón «Dar de alta» de nuestra página de login tiene asignado al evento click la función signin().
Por lo tanto vamos a definir la función signin en el controlador, para ello debemos importar también el provider AuthProvider que hemos creado anteriormente, también vamos a importar AlertController para mostrar los mensajes de error:
import{ Component } from '@angular/core';import{ IonicPage, NavController, NavParams,AlertController} from 'ionic-angular';import{ AuthProvider } from '../../providers/auth/auth';/** * Generated class for the LoginPage page. * * See http://ionicframework.com/docs/components/#navigation for more info * on Ionic pages and navigation.*/@IonicPage()@Component({selector:'page-login',templateUrl:'login.html',})exportclass LoginPage {
user ={email :'',password :''};
constructor(publicnavCtrl: NavController,publicnavParams: NavParams,publicauth : AuthProvider,publicalertCtrl : AlertController){}
ionViewDidLoad(){
console.log('ionViewDidLoad LoginPage');}
signin(){this.auth.registerUser(this.user.email,this.user.password).then((user)=>{// El usuario se ha creado correctamente}).catch(err=>{
let alert =this.alertCtrl.create({
title:'Error',
subTitle: err.message,
buttons:['Aceptar']});
alert.present();})}}
Como vemos en la función signin() llamamos al método registerUser que hemos creado anteriormente en el provider authProvider pasándole como parametros el email y la contraseña que las tenemos en this.user.email y this.user.password respectivamente.
Si se ha producido un error mostramos un alert con el mensaje de error que hayamos recibido.
Bien para que salga la página de login al iniciar la aplicación tenemos que cambiar el root de nuestra aplicación tal y como vimos en este post. Para ello vamos editamos el archivo app.components.ts y sustituimos:
rootPage:any ='MisTabsPage';
Por:
rootPage:any ='LoginPage';
En este momento ya podemos probar la creación de un usuario.
Ejecutamos la aplicación en el terminal, si al ejecutar os muestra un error como este:
Error:./node_modules/firebase/utils/promise.js
Module not found: Error: Can't resolve 'promise-polyfill'
Lo resolvemos instalando lo siguiente:
npm install promise-polyfill --save-exact
Una vez funcionando nuestra app aparecerá la página de login que hemos creado con el formulario para introducir usuario y contraseña. Si vamos al panel de firebase y seleccionamos Authentication y la pestaña USUARIOS veremos algo como esto:
Como podemos ver aún no hay ningún usuario creado, podríamos crear uno desde el propio panel de firebase pulsando el botón añadir usuario, pero lo que nos interesa es crearlos directamente desde nuestra aplicación.
Al ejecutar nuestra aplicación ahora mostrará la pagina de login que hemos creado con el formulario para introducir el email y la contraseña:
Si introducimos una dirección de correo y una contraseña y pulsamos el botón «Dar de alta» al refrescar la pantalla de firebase ahora veréis el usuario que acabáis de crear en el panel.
Ahora que ya podemos crear usuarios tenemos que autenticarlos y una vez autenticados tenemos que cambiar la página activa por la página misTabsPage.
Vayamos por partes.
Lo primero que necesitamos es crear en authProvider un método para autenticarnos con Correo electrónico / Contraseña, por lo tanto editamos auth.ts y añadimos el siguiente método:
// Login de usuario
loginUser(email:string,password:string){returnthis.afAuth.auth.signInWithEmailAndPassword(email, password).then(user=>Promise.resolve(user)).catch(err=>Promise.reject(err))}
Con el método signInWithEmailAndPassword(email, password) nos autenticamos en firebase, si todo va bien nos devolverá el usuario, si ha habido algo incorrecto nos devolverá un error.
Ahora en login.ts vamos a crear el método login() que se llama al pulsar el botón Login:
login(){this.auth.loginUser(this.user.email,this.user.password ).then((user)=>{}).catch(err=>{
let alert =this.alertCtrl.create({
title:'Error',
subTitle: err.message,
buttons:['Aceptar']});
alert.present();})}
Como vemos lo que hace login es llamar a la función loginUser de authProvider pasandole el email y la contraseña.
Ahora vamos a necesitar un método que nos devuelva la sesión, editamos auth.ts y añadimos la siguiente función:
// Devuelve la sessionget Session(){returnthis.afAuth.authState;}
Si no estas familiarizado con TypeScript puede que pienses que hay un error ya que entre get y Session() hay un espacio. Esta es una peculiaridad de TypeScript que nos permite hacer getters y setters de esta manera. Para obtener la sesión en lugar de llamar a la función simplemente llamamos a Session como si fuese un atributo de la clase, es decir sin el get y sin paréntesis de esta manera: this.auth.Session.subscribe(…), en este caso usamos subscribe porque Session devuelve this.afAuth.authState que a su vez devuelve un observable.
Ahora vamos a comprobar la sesión en el evento platform.ready de app.components.ts que se ejecuta cuando la app está cardada y lista y si estamos autenticados haremos que la página activa sea misTabsPage, pero si la sesión no está activa la página que se muestre se la de login.
Editamos app.components.ts y añadimos lo siguiente:
import{ Component } from '@angular/core';import{ Platform } from 'ionic-angular';import{ StatusBar } from '@ionic-native/status-bar';import{ SplashScreen } from '@ionic-native/splash-screen';import{ DbProvider } from '../providers/db/db';import{ AuthProvider } from '../providers/auth/auth';@Component({templateUrl:'app.html'})exportclass MyApp {rootPage:any='LoginPage';
constructor(platform: Platform,statusBar: StatusBar,splashScreen: SplashScreen,publicdb: DbProvider,privateauth: AuthProvider){
platform.ready().then(()=>{// Okay, so the platform is ready and our plugins are available.// Here you can do any higher level native things you might need.this.auth.Session.subscribe(session=>{if(session){this.rootPage ='MisTabsPage';}else{this.rootPage ='LoginPage';}});
statusBar.styleDefault();
splashScreen.hide();this.db.openDb().then(()=>this.db.createTableSitios())});}}
Tenemos que importar e inyectar en el constructor nuestro provider AuthProvider.
Después en patform.ready nos suscribimos a Session que como he explicado es un getter que nos devuelve un observable, si hay un cambio en la sesión automáticamente cambiara rootPage para mostrar la página de login o las pestañas en función de si tenemos la sesión activa o no.
También vamos a necesitar un método para cerrar sesión (logout), por lo tanto editamos auth.ts y añadimos el método logout:
// Logout de usuario
logout(){this.afAuth.auth.signOut().then(()=>{// hemos salido})}
Por simplificar vamos a colocar un botón de cerrar sesión en la página info que no la estamos utilizando para nada, abrimos info.html y añadimos el siguiente código:
<!-- Generated template for the InfoPage page. See http://ionicframework.com/docs/components/#navigation for more info on Ionic pages and navigation.--><ion-header><ion-navbar><ion-title>Info</ion-title></ion-navbar></ion-header><ion-content padding><button ion-button block (click)="cerrarSesion()">Cerrar Sesión</button></ion-content>
Simplemente hemos añadido un botón que al hacer click llamara a la función cerrarSession().
Vamos ahora a definir la función cerrarSesion en el controlador, abrimos info.ts y añadimos el siguiente código:
import{ Component } from '@angular/core';import{ IonicPage, NavController, NavParams } from 'ionic-angular';import{ AuthProvider } from '../../providers/auth/auth';/** * Generated class for the InfoPage page. * * See http://ionicframework.com/docs/components/#navigation for more info * on Ionic pages and navigation.*/@IonicPage()@Component({selector:'page-info',templateUrl:'info.html',})exportclass InfoPage {
constructor(publicnavCtrl: NavController,publicnavParams: NavParams,publicauth : AuthProvider){}
ionViewDidLoad(){
console.log('ionViewDidLoad InfoPage');}cerrarSesion(){this.auth.logout();}}
Simplemente importamos el provider AuthProvider, lo inyectamos en el constructor y en la función cerrarSesion llamamos al método logout del provider que acabamos de crear.
Si probamos ahora nuestra aplicación y accedemos a la pestaña info, al pulsar sobre el botón cerrar sesión nos mostrará de nuevo la página de login. Esto ocurre como ya he explicado porque en app.components.ts estamos «escuchando» a la variable Session que es un observable, al cerrar la sesión automáticamente cambia rootPage por ‘LoginPage’.
Para que no queden dudas de como debe quedar el provider AuthProvider muestro a continuación como debe de quedar el código completo del archivo auth.ts:
import{ Injectable } from '@angular/core';import{ AngularFireAuth } from 'angularfire2/auth';import* as firebase from 'firebase/app';/* Generated class for the AuthProvider provider. See https://angular.io/docs/ts/latest/guide/dependency-injection.html for more info on providers and Angular DI.*/@Injectable()exportclass AuthProvider {
constructor(privateafAuth : AngularFireAuth){
console.log('Hello AuthProvider Provider');}// Registro de usuario
registerUser(email:string,password:string){returnthis.afAuth.auth.createUserWithEmailAndPassword( email, password).then((res)=>{this.afAuth.auth.signInWithEmailAndPassword(email, password)}).then(user=>Promise.resolve(user)).catch(err=>Promise.reject(err))}// Login de usuario
loginUser(email:string,password:string){returnthis.afAuth.auth.signInWithEmailAndPassword(email, password).then(user=>Promise.resolve(user)).catch(err=>Promise.reject(err))}// Logout de usuario
logout(){this.afAuth.auth.signOut().then(()=>{// hemos salido})}// Devuelve la sessionget Session(){returnthis.afAuth.authState;}}
Eso es todo por hoy.
Hoy hemos aprendido a crear usuario y autenticarnos mediante correo electronico y contraseña en firebase. El el próximo post aprenderemos a guardar nuestros sitios en la base de datos de firebase.
Hola a todos, hoy vamos a ver como podemos comunicar una aplicación desarrollada con ionic con una API REST, para ello vamos a aprender como realizar peticiones a un servidor remoto a través de http.
En esta pequeña prueba vamos a ver como podemos acceder a una API REST para obtener desde un servidor remoto un listado de usuarios. Para este pequeño ejemplo vamos a utilizar RANDOM USER GENERATOR que como se indica en su web es una API libre de código abierto para generar datos de usuario aleatorios para realizar pruebas. Como Lorem Ipsum, pero con personas.
Lo que vamos a hacer es simplemente realizar una llamada a esta API donde recibiremos como respuesta un listado de usuarios que mostraremos en nuestra vista.
Antes de nada vamos a crear una nueva aplicación de prueba:
ionic start pruebahttp1 blank
Ahora vamos a crear un provider donde gestionaremos la comunicación con el servidor remoto:
ionic g provider http
Se habrá creado una carpeta providers (si no existía) y dentro una carpeta con el nombre del provider que acabamos de crear (http) y dentro un archivo .ts con el mismo nombre.
Por defecto contendrá el siguiente código:
import{ HttpClient } from '@angular/common/http';import{ Injectable } from '@angular/core';/* Generated class for the HttpProvider provider. See https://angular.io/guide/dependency-injection for more info on providers and Angular DI.*/@Injectable()exportclass HttpProvider {
constructor(publichttp: HttpClient){
console.log('Hello HttpProvider Provider');}}
Vemos que al crear un provider ya se importa el módulo HttpClient.
Para poder utilizar HttpClient tenemos que importar y declarar HttpClientModule en el archivo app.module.ts:
import{ BrowserModule } from '@angular/platform-browser';import{ ErrorHandler, NgModule } from '@angular/core';import{ IonicApp, IonicErrorHandler, IonicModule } from 'ionic-angular';import{ SplashScreen } from '@ionic-native/splash-screen';import{ StatusBar } from '@ionic-native/status-bar';import{ MyApp } from './app.component';import{ HomePage } from '../pages/home/home';import{ HttpProvider } from '../providers/http/http';import{ HttpClientModule } from '@angular/common/http';@NgModule({declarations:[
MyApp,
HomePage
],imports:[
BrowserModule,
IonicModule.forRoot(MyApp),HttpClientModule],bootstrap:[IonicApp],entryComponents:[
MyApp,
HomePage
],providers:[
StatusBar,
SplashScreen,{provide: ErrorHandler,useClass: IonicErrorHandler},
HttpProvider
]})exportclass AppModule {}
http.get
Ahora vamos a añadir un método que llamaremos loadUsers a nuestro provider para obtener la lista de usuarios desde el servidor, por lo tanto editamos el archivo http.ts y añadimos el siguiente código:
import{ HttpClient } from '@angular/common/http';import{ Injectable } from '@angular/core';/* Generated class for the HttpProvider provider. See https://angular.io/guide/dependency-injection for more info on providers and Angular DI.*/@Injectable()exportclass HttpProvider {path :string='https://randomuser.me/api/?results=25';
constructor(publichttp: HttpClient){
console.log('Hello HttpProvider Provider');}loadUsers(){returnthis.http
.get(this.path)}}
http.get devuelve el resultado de la solicitud en la forma de un observable.
El listado de usuarios tendrá un formato similar a este:
Ahora vamos a crear la vista en home.html para mostrar un botón que llamará a la función
cargarUsuarios,y un listado de items con los usuarios que crearemos recorriendo con *ngFor el array usuarios que posteriormente vamos a crear en el controlador:
<ion-header><ion-navbar><ion-title>
Usuarios
</ion-title></ion-navbar></ion-header><ion-content><ion-list><ion-item *ngFor="let usuario of usuarios"><ion-avatar item-start><img [src]="usuario.picture.medium"></ion-avatar><h2>{{ usuario.name.first }}</h2><p>{{ usuario.email }}</p></ion-item></ion-list><button ion-button full (click)="cargarUsuarios()">Cargar Usuarios</button></ion-content>
Ahora vamos a modificar home.ts para obtener los datos desde el provider y mostrarlos en la vista.
Editamos home.ts e importamos el provider httpProvider que acabamos de crear:
import{ HttpProvider } from '../../providers/http/http';
Para poder utilizarlo debemos inyectarlo en el constructor:
constructor(public navCtrl: NavController, public http: HttpProvider){}
Justo antes del constructor vamos a definir una variable miembro donde guardaremos el array de usuarios que recibamos desde el servidor:
usuarios : any[];
Promesas y Observables
Ahora vamos a crear un método que que a su vez llamará al método loadUsers de nuestro provider para recibir los datos de los usuarios:
import{ Component } from '@angular/core';import{ NavController } from 'ionic-angular';import{ HttpProvider } from '../../providers/http/http';@Component({selector:'page-home',templateUrl:'home.html'})exportclass HomePage {usuarios :any[];
constructor(publicnavCtrl: NavController,publichttp: HttpProvider){}cargarUsuarios(){this.http.loadUsers().subscribe((res)=>{this.usuarios = res['results'];},(error)=>{
console.error(error);})}}
Como podemos observar llamamos al método loadUsers que hemos definido en el provider, pero no utilizamos then sino subscribe, esto es porque http no devuelve una promesa si no que devuelve un observable. Un observable se queda a la espera de recibir datos y nosotros nos «subscribimos» recibiendo estos datos en cuanto estén disponibles. Esta cualidad se puede utilizar para suscribirnos a la url y observar si ha habido cambios, como por ejemplo si es un sitio de noticias donde se están continuamente renovando.
La diferencia entre promesas y observables a groso modo es que la promesa devuelve los datos una única vez cuando estos son recibidos mientras que un observable se queda «vigilando» si se han producido cambios y se ejecuta cada vez que un cambio se produce, aunque hasta que no te suscribes a un observable éste no se ejecutará.
Si solo necesitamos recibir los datos una única vez sin necesidad de observar si se han producido cambios podemos utilizar promesas. Una promesa se ejecuta una vez que se haya resuelto la llamada y recibimos los datos en la función .then(res=>{… }).
El código del provider en el archivo http.ts utilizando una promesa quedaría de la siguiente manera:
import{ HttpClient } from '@angular/common/http';import{ Injectable } from '@angular/core';import'rxjs/add/operator/toPromise';/* Generated class for the HttpProvider provider. See https://angular.io/guide/dependency-injection for more info on providers and Angular DI.*/@Injectable()exportclass HttpProvider {path :string='https://randomuser.me/api/?results=25';
constructor(publichttp: HttpClient){
console.log('Hello HttpProvider Provider');}
loadUsers(){returnthis.http
.get(this.path).toPromise();}}
Debemos importar el operador toPromise para convertir el observable en promesa con import ‘rxjs/add/operator/toPromise’; y simplemente añadir toPromise() después del get.
Ahora en home.ts solo tenemos que sustituir subscribe por then:
Si ejecutáis la aplicación con ionic serve -l y pulsáis en el botón Cargar Usuarios podréis ver algo como esto:
Listado de usuarios
Podéis observar como cada vez que pulséis el botón la lista de usuarios cambia ya que Random User Generator como su propio nombre indica devuelve una lista aleatoria de usuarios.
http.post
RANDOM USER GENERATOR nos ha servido para aprender ha hacer una petición get a una API REST para recibir datos, en este caso una lista de usuarios.
Si queréis conectaros con un servicio que tengáis corriendo en un servidor remoto y necesitáis pasarle datos desde la aplicación para que se guarden en el servidor tenemos que usar http.post.
Vamos a imaginar que tenemos corriendo un servicio en PHP cuya url sea www.miservicio.com/adduser/ que está programado para recibir vía post un nuevo usuario para guardarlo en la base de datos del servidor.
Tendremos que crear una función similar a esta:
postDatos(){
let datos ={ nombre:'Edu',email:'edu.revilla.vaquero@gmail.com'}
let options ={ headers:{'Content-Type':'application/x-www-form-urlencoded'}};
var url ='www.miservicio.com/adduser/';return new Promise(resolve =>{this.http.post(url,JSON.stringify(datos),options).subscribe(data =>{
resolve(data);});});}
En la variable datos tenemos un objeto con los datos del usuario que queremos enviar al servidor, en este caso nombre y email.
Después definimos la variable options donde a su vez definimos las cabeceras en la variable headers, dependiendo de la configuración del servidor podría necesitar parámetros diferentes en la cabecera.
Como vemos http.post es bastante parecido a http.get, solo que como segundo parámetro le pasamos la variable que contiene los datos que queremos enviar convertida a formato JSON con JSON.stringify(datos), como tercer parámetro le pasamos options.
En data recibiremos la respuesta que nos de el servidor.
Vamos a ver un ejemplo de como obtendríamos los datos enviados desde nuestra aplicación con un servicio desarrollado en PHP en el servidor:
Evidentemente en otros lenguajes de programación del lado del servidor el código para recibir los datos sería diferente, pero se escapa del propósito de este tutorial el abordar como sería en cada leguaje. Si alguno hace la prueba en otro lenguaje y quiere compartirlo en los comentarios para ayudar a los demás será bien recibido :-).
Tutorial de Ionic - Crear una aplicación para guardar nuestros sitios geolocalizados - Parte 8 - Eliminar un sitio desde el listado deslizando el item con "ItemSliding".
En el post anterior vimos como modificar un sitio, hoy vamos a mejorar nuestra app para que se puedan eliminar nuestros sitio, pero para aprender más sobre las posibilidades de ionic lo vamos ha hacer desde el listado, deslizando el elemento del listado hacia la izquierda nos mostrará el botón de eliminar. También aprovecharemos para aprender a utilizar AlertController para mostrar un dialogo que pida confirmación antes de eliminar el sitio.
Vamos a ver como sería:
Lo primero que vamos a hacer es modificar el listado para añadir un ion-item-sliding que es un componente que nos permite deslizar el item, debemos introducir el item dentro de ion-item-sliding y le vamos a añadir despues del item un elemento ion-item-options que nos permite añadir opciones que se mostrarán al deslizar el item. Se pueden añadir tantas opciones como desees, nosotros solo vamos a necesitar una para borrar el elemento. Vamos a ver como tiene que quedar el código, editamos el archivo listado.html y lo dejamos de la siguiente manera:
<!-- Generated template for the ListadoPage page. See http://ionicframework.com/docs/components/#navigation for more info on Ionic pages and navigation.--><ion-header><ion-navbar><ion-title>Listado</ion-title></ion-navbar></ion-header><ion-content padding><ion-list><ion-item-sliding #item *ngFor="let sitio of sitios"><ion-item (click)="muestraSitio(sitio)"><ion-thumbnail item-left><img [src]="sitio.foto"></ion-thumbnail><h2>{{ sitio.address }}</h2><p>{{ sitio.description }}</p></ion-item><ion-item-options side="right"><button ion-button color="danger" (click)="borrarSitio(sitio.id)"><ion-iconname="trash"></ion-icon>
Borrar
</button></ion-item-options></ion-item-sliding></ion-list></ion-content>
Como podemos observar hemos puesto el *ngFor en la etiqueta ion-item-sliding que ahora envuelve al ion-item que ya teníamos.
Despues del ion-item hemos añadido un componente ion-item-options al que le indicamos que se muestre en el lado derecho (side=»right»).
Dentro de ion-item-options tenemos un botón con in icono (ion-icon) que muestra el icono «trash» que es el típico cubo de basura y el texto «borrar».
En el evento click del botón le decimos que llame a la función borrarSitio.
Si probamos la aplicación podéis comprobar en el listado que los items ahora se pueden arrastrar hacia la izquierda y aparece el botón de borrar:
Borrando un sitio
Ahora debemos definir la función borrar sitio en el controlador del listado. Antes de borrar vamos a sacar un dialogo de confirmación que nos preguntará si realmente queremos eliminar el sitio, para ello vamos a utilizar el componente Alert.
Alerts
Los alerts son una excelente manera de ofrecer al usuario la posibilidad de elegir una acción específica o una lista de acciones. También pueden proporcionar al usuario información importante, o requerir que tomen una decisión.
Ionic nos proporciona los siguientes tipos de Alerts:
En este caso como lo que queremos es que el usuario confirme el borrado del sitio vamos a utilizar alert de confirmación.
Para poder utilizar un alert lo primero que debemos hacer es importar AlertController e inyectarlo en el constructor, por lo tanto vamos a editar el archivo listado.ts y añadimos el siguiente código:
import{ Component } from '@angular/core';import{ IonicPage, NavController, NavParams, ModalController } from 'ionic-angular';import{ DbProvider } from '../../providers/db/db';import{ AlertController } from 'ionic-angular';/** * Generated class for the ListadoPage page. * * See http://ionicframework.com/docs/components/#navigation for more info * on Ionic pages and navigation.*/@IonicPage()@Component({selector:'page-listado',templateUrl:'listado.html',})exportclass ListadoPage {sitios:any;
constructor(publicnavCtrl: NavController,publicnavParams: NavParams,publicmodalCtrl : ModalController,publicdb : DbProvider,publicalertCtrl : AlertController){}....
Ahoravamos a a crear la función borrarSitio() y definiremos dentro de ella el alert de la siguiente manera:
borrarSitio(id){
let alert =this.alertCtrl.create({
title:'Confirmar borrado',
message:'¿Estás seguro de que deseas eliminar este sitio?',
buttons:[{
text:'No',
role:'cancel',
handler:()=>{// Ha respondido que no así que no hacemos nada}},{
text:'Si',
handler:()=>{// AquÍ borramos el sitio en la base de datos}}]});
alert.present();}
Para crear un alert utilizamos el método create donde tenemos que definir el título en este caso «Confirmar borrado», el mensaje: «¿Estas seguro de que deseas eliminar este sitio?».
Después definimos los dos botones de acción uno para SI y otro para NO, en text definimos el texto del botón y en handler definiremos la acción que debemos realizar, en caso de pulsar «No» no hacemos nada, en caso de pulsar «Si» procederemos a eliminar el sitio.
Una vez definido el alert con alert.present() hacemos que se muestre.
Ahora que ya sabemos como se usan los alerts vamos a escribir el código del botón «Si» para borrar el sitio.
Lo primero que tenemos que hacer es crear un nuevo método en el provider DbProvider para eliminar los sitios, por lo tanto editamos el archivo db.ts y añadimos la función borrarSitio():
public borrarSitio(id){
let sql ="DELETE FROM sitios WHERE id= ? ";returnthis.db.executeSql(sql,[id]);}
Recibimos como parámetro id que es el id del sitio que queremos eliminar de la base de datos, ejecutamos la query y si todo ha ido bien e sitio se habrá eliminado para siempre de la base de datos.
Ahora que ya tenemos preparada la función para borrar el sitio en el provider vamos a volver al alert de confirmación en listado.ts para definir las operaciones a realizar al pulsar el botón «Si», en la función handler del botón «Si» añadimos el siguiente código:
borrarSitio(id){
let alert =this.alertCtrl.create({
title:'Confirmar borrado',
message:'¿Estás seguro de que deseas eliminar este sitio?',
buttons:[{
text:'No',
role:'cancel',
handler:()=>{// Ha respondido que no así que no hacemos nada }},{
text:'Si',
handler:()=>{this.db.borrarSitio(id).then((res)=>{// Una vez borrado el sitio recargamos el listadothis.db.getSitios().then((res)=>{this.sitios =[];for(var i =0; i < res.rows.length; i++){this.sitios.push({
id : res.rows.item(i).id,
lat: res.rows.item(i).lat,
lng: res.rows.item(i).lng,
address: res.rows.item(i).address,
description: res.rows.item(i).description,
foto: res.rows.item(i).foto
});}},(err)=>{/* alert('error al sacar de la bd'+err) */})},(err)=>{/* alert('error al borrar de la bd'+err) */});}}]});
alert.present();}
Lo primero que hacemos es llamar al método que acabamos de crear en el provider con this.db.borrarSitio(id) pasándole como parámetro el id del sitio.
En el then lo que vamos ha hacer es refrescar el listado para ello debemos modificar la variable miembro this.sitios que contiene los sitios que se muestran en el listado, lo que hacemos es volver a cargar de la base de datos los sitios, como acabamos de borrar un sitio este no aparecerá. Para cargar los sitios de la base de datos utilizamos el método this.db.getSitios que ya tenemos definido y hacemos exactamente el mismo proceso que en la función ionViewDidEnter para rellenar el array this.sitios con los datos obtenidos desde la base de datos. De hecho podemos definir una función que se encargue de recoger los sitios de la base de datos y rellenar el array this.sitios en lugar de repetir el código de ionViewDidEnter, eso ya lo dejo como mejora opcional.
Si ejecutáis el código comprobaréis que ya podéis borrar los sitio que no os interesen conservar en vuestro dispositivo.
Tutorial de Ionic - Crear una aplicación para guardar nuestros sitios geolocalizados - Parte 6 - Mostrar detalles del sitio guardado y abrir navegador gps para llegar a el.
Hola a todos, en el post anterior vimos como guardar nuestros sitios en la base de datos y como mostrar los sitios que tenemos guardados en la página listado.
Hoy vamos a mostrar el detalle del sitio al pulsar sobre el en el listado. La idea es que nos muestre la dirección, la foto y la descripción que hayamos introducido, además vamos a añadir un botón de «Como llegar» para que nos abra el navegador gps del móvil con la dirección hacía dicho sitio. Ya no volveremos a olvidar donde hemos aparcado el coche, o podremos recordar como volver a ese restaurante que tanto nos gustó.
Lo primero que tenemos que hacer es crear un nuevo modal con ionic generator:
ionic g page modalDetalleSitio
Ahora en listado.ts debemos importar el componente ModalController de la librería ionic-angular, por lo tanto después de NavController y NavParams importamos también ModalController.
import{ IonicPage, NavController, NavParams, ModalController } from 'ionic-angular';
Debemos inyectar también en el constructor el componente ModalController al que hemos llamado modalCtrl:
constructor(
public navCtrl: NavController,
public navParams: NavParams,public modalCtrl : ModalController,
public db: DbProvider){......
Ahora ya podemos crear un método en listado.ts para abrir el modal que acabamos de crear pasandole como parámetro el sitio que queremos mostrar:
muestraSitio(sitio){
let modalSitio =this.modalCtrl.create( 'ModalDetalleSitioPage', sitio );
modalSitio.present();}
Vamos a modificar el listado para que al pulsar sobre un sitio de la lista se abra el modal por lo tanto editamos listado.html y añadimos lo siguiente:
<!-- Generated template for the Listado page. See http://ionicframework.com/docs/components/#navigation for more info on Ionic pages and navigation.--><ion-header><ion-navbar><ion-title>Listado</ion-title></ion-navbar></ion-header><ion-content padding><ion-list><ion-item *ngFor="let sitio of sitios" (click)="muestraSitio(sitio)"><ion-thumbnail item-left><img [src]="sitio.foto"></ion-thumbnail><h2>{{ sitio.address }}</h2><p>{{ sitio.description }}</p></ion-item></ion-list></ion-content>
Con esto llamamos al método muestraSitio que acabamos de definir en el controlador listado.ts y le pasamos como parámetro el sitio sobre el se ha hecho click.
Ahora vamos a editar el archivo modal-detalle-sitio.ts y vamos a importar e inyectar en en constructor el elemento ViewController que lo utilizaremos para poder cerrar el modal:
import{ Component } from '@angular/core';import{ IonicPage, NavController, NavParams,ViewController} from 'ionic-angular';/** * Generated class for the ModalDetalleSitio page. * * See http://ionicframework.com/docs/components/#navigation for more info * on Ionic pages and navigation.*/
@IonicPage()
@Component({selector:'page-modal-detalle-sitio',templateUrl:'modal-detalle-sitio.html',})exportclass ModalDetalleSitioPage {
constructor(publicnavCtrl: NavController,publicnavParams: NavParams,privateviewCtrl : ViewController){}
...
Vamos a crear también una función para cerrar el modal a la que luego llamaremos desde el botón de cerrar de la vista:
cerrarModal(){this.viewCtrl.dismiss();}
Ahora vamos a crear una variable miembro llamada sitio donde vamos a guardar el objeto con los datos del sitio que recibimos desde NavParams como parámetro al abrir el modal desde la página listado:
...export class ModalDetalleSitioPage { sitio: any;
constructor(public navCtrl: NavController, public navParams: NavParams, private viewCtrl : ViewController){ this.sitio = this.navParams.data;}...
Ahora ya estamos listos para poder mostrar los datos del sitio el la vista, así que editamos el archivo modal-detalle-sitio.html e introducimos el siguiente código:
<!-- Generated template for the ModalDetalleSitio page. See http://ionicframework.com/docs/components/#navigation for more info on Ionic pages and navigation.--><ion-header><ion-navbar><ion-title>{{ sitio.address }}</ion-title><ion-buttonsstart><button ion-button (click)="cerrarModal()"><ion-iconname="md-close"></ion-icon></button></ion-buttons></ion-navbar></ion-header><ion-content padding><ion-card><img [src]="sitio.foto" *ngIf="sitio.foto"/><ion-card-content><ion-itemtext-wrap><h2>{{ sitio.address }}</h2><p>{{ sitio.description }}</p></ion-item></ion-card-content><ion-row><ion-coltext-center><button ion-button icon-leftclear small (click)="comoLlegar()"><ion-iconname="navigate"></ion-icon><div>Como llegar</div></button></ion-col></ion-row></ion-card></ion-content>
Como título del modal hemos puesto la dirección que está almacenada en sitio.address.
Después añadimos el botón para cerrar el modal, para ello le indicamos que en el evento click llame a la función cerrarModal que hemos definido en el controlador.
Después dentro de ion-content creamos un elemento ion-card y dentromostramos la foto solo si existe con el condicional *ngIf.
Después dentro de ion-card-content mostramos la dirección y la descripción.
Por último creamos un botón con un icono de tipo navigate dentro de un elemento ion-col que a su vez esta dentro de un elemento ion-row. Hacemos esto para conseguir que el botón se muestre centrado gracias a la propiedad text-center que hemos añadido a la etiqueta ion-col.
Como podemos observar en el evento click del botón llamamos a la función comoLlegar().
La idea es que cuando pulsemos en el botón nos habrá el navegador gps que tengamos instalado por defecto en el móvil para indicarnos la ruta a seguir para llegar al lugar.
Vamos ello vamos a crear la función comoLlegar en el controlador.
Aunque podríamos utilizar la función que ya expliqué en esté post perteneciente a la serie de posts que publiqué hace tiempo sobre google maps, vamos a utilizar en su lugar el plugin LaunchNavigator que ya nos ofrece esta funcionalidad de forma nativa (gracias a writ3r por el aporte).
Lo primero que debemos hacer es instalar el plugin escribiendo los siguientes comandos desde consola:
Ahora debemos importar y declarar el plugin en app.module.ts:
import { BrowserModule } from '@angular/platform-browser';import{ ErrorHandler, NgModule } from '@angular/core';import{ IonicApp, IonicErrorHandler, IonicModule } from 'ionic-angular';import{ SplashScreen } from '@ionic-native/splash-screen';import{ StatusBar } from '@ionic-native/status-bar';import{ MyApp } from './app.component';import{ Geolocation } from '@ionic-native/geolocation';import{ Camera } from '@ionic-native/camera';import{ DbProvider } from '../providers/db/db';import{ SQLite } from '@ionic-native/sqlite';import{ LaunchNavigator } from '@ionic-native/launch-navigator';@NgModule({
declarations:[
MyApp
],
imports:[
BrowserModule,
IonicModule.forRoot(MyApp)],
bootstrap:[IonicApp],
entryComponents:[
MyApp
],
providers:[
StatusBar,
SplashScreen,{provide: ErrorHandler, useClass: IonicErrorHandler},
Geolocation,
Camera,
DbProvider,
SQLite,LaunchNavigator]})export class AppModule {}
Ahora editamos el archivo modal-detalle-sitio.ts, importamos el plugin LauchNavigator, lo inyectamos en el constructor y añadimos el método comoLlegar:
import{ Component } from '@angular/core';import{ IonicPage, NavController, NavParams, ViewController } from 'ionic-angular';import{ LaunchNavigator } from '@ionic-native/launch-navigator';/** * Generated class for the ModalDetalleSitioPage page. * * See http://ionicframework.com/docs/components/#navigation for more info * on Ionic pages and navigation.*/@IonicPage()@Component({selector:'page-modal-detalle-sitio',templateUrl:'modal-detalle-sitio.html',})exportclass ModalDetalleSitioPage {sitio:any;
constructor(publicnavCtrl: NavController,publicnavParams: NavParams,privateviewCtrl : ViewController,privatelaunchNavigator : LaunchNavigator){this.sitio =this.navParams.data;}
ionViewDidLoad(){
console.log('ionViewDidLoad ModalDetalleSitioPage');}
cerrarModal(){this.viewCtrl.dismiss();}comoLlegar(){
let destino =this.sitio.lat+','+this.sitio.lng;this.launchNavigator.navigate(destino).then(
success => console.log('Launched navigator'),
error => console.log('Error launching navigator', error));}}
El método navigate de launchNavigator recibe como parámetro la dirección donde queremos llegar, en este caso las coordenadas separadas por una coma que tenemos en las variables this.sitio.lat y this.sitio.lng.
Si ejecutamos nuestra aplicación en en móvil obtendremos algo como esto al entrar en el detalle de un sitio de la lista:
Modal detalle del sitio
Bien , por hoy lo dejamos aquí, ya podemos guardar nuestros sitios favoritos, o recordar donde hemos aparcado el coche por ejemplo, podemos mostrar los sitios que hemos guardado y podemos abrir el navegador gps para poder llegar asta el lugar.
Aunque se puede mejorar mucho esta aplicación, ya podemos decir que es funcional y nos ha servido para aprender muchos conceptos sobre ionic. Como siempre os animo a que experimentéis y tratéis de mejorar la app por vuestra cuenta ya que es una magnifica forma de aprender y mejorar.
En el próximo post seguiremos desarrollando la app y veremos como modificar un sitio guardado.