Tutorial de Ionic – Firebase – parte 2: Database – Guardar nuestros sitios en la nube.

En el siguiente enlace tienes el índice para acceder al resto de entradas de este tutorial:

 

¡¡Atención!! este tutorial se basa en ionic 3 y está desactualizado por lo que es posible que los ejemplos no funcionen en la última versión de ionic, haz click aquí para acceder a un tutorial mas actual de Ionic.

Hola a todos:

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()
export class FirebaseDbProvider {

  constructor(public afDB: 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(){
    return this.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:

guardaSitio(sitio){
     sitio.id  = Date.now();
     return this.afDB.database.ref('sitios/'+this.auth.getUser()+'/'+sitio.id).set(sitio)
  }

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',
})
export class ModalNuevoSitioPage {

  coords : any = { lat: 0, lng: 0 }
  address: string;
  description: string = '';
  foto: any = '';

  constructor(
    public navCtrl: NavController,
    public navParams: NavParams,
    private viewCtrl : ViewController,
    private camera: Camera,
    // private db: DbProvider
    private dbFirebase :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();

    return new Promise(function(resolve, reject) {
        geocoder.geocode({'location': coords} , function (results, status) { // llamado asincronamente
            if (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 string
        this.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:

getSitios(){
    return this.afDB.list('sitios/'+this.auth.getUser()).valueChanges();
  }

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',
})
export class ListadoPage {

  sitios: any;

  constructor(
    public navCtrl: NavController,
    public navParams: NavParams,
    public db : DbProvider,
    public modalCtrl : ModalController,
    public alertCtrl : AlertController,
    public dbFirebase :FirebaseDbProvider,

  ) {

  }

Ahora en la función ionViewDidEnter() vamos a sustituir this.db.getSitios().then((res)=>{…}) por lo siguiente:

ionViewDidEnter(){
   
    this.dbFirebase.getSitios().subscribe(sitios=>{
      this.sitios = sitios;
    })

}

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:

guardaSitio(sitio){
   if(!sitio.id){
      sitio.id  = Date.now();
    }
    return this.afDB.database.ref('sitios/'+this.auth.getUser()+'/'+sitio.id).set(sitio)
  }

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',
})
export class ModalDetalleSitioPage {

  sitio: any;
  edit : boolean = false;

  constructor(
    public navCtrl: NavController,
    public navParams: NavParams,
    private viewCtrl : ViewController,
    private launchNavigator : LaunchNavigator,
    private camera: Camera,
  //  private db: DbProvider,
    private dbFirebase :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 string
        this.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 firebase
              this.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

 

Si necesitas desarrollar una aplicación móvil no dudes en solicitarme un presupuesto sin compromiso:

Un saludo, y si aún no lo has hecho no olvides suscribirte a mi blog para no perderte los próximos posts  :-),

También puedes seguirme en Twitter en ‎@revigames y no olvides que me ayudas mucho si compartes este post en las redes sociales.

27 comentarios en “Tutorial de Ionic – Firebase – parte 2: Database – Guardar nuestros sitios en la nube.

  1. Genial!!! He completado todo el tutorial desde el inicio en el que creamos la app de sitios sin ningún problema, todo perfecto. Esto no me había pasado nunca, siempre tutoriales incompletos y hechos deprisa. Te felicito.

    Aprovecho para ver si puedes ayudarme con una dudilla, después de ver lo genial que son las bases de datos realtime de Firebase.

    Tengo en la cabeza una aplicación pero no se muy bien si Firebase es la mejor opción o no a la hora de almacenar lo que quiero. Esta aplicación quiero que muestre una galería de imágenes almacenadas en algún sitio (Firebase? Cloudinary?). Todos los usuarios verán las mismas imágenes en la app pero cada uno podrá tener estas imágenes en un estado diferente, por ejemplo visto, no visto, modificado. ¿Como crees que es la mejor opción para hacer algo así?

    Muchas gracias!!

    1. Hola Sergio, me alegro que el tutorial te haya sido de utilidad,
      En cuanto a lo que me preguntas la verdad es que no he usado Cloudinary así que no puedo darte una opinión, lo que si te diría que a pesar de que en este ejemplo del tutorial almacenamos las imágenes de los sitios directamente en la base de datos en formato base64, no es una buena idea cuando son muchas las imágenes que vas a almacenar, creo que lo ideal es tener las imágenes almacenadas en otro sitio, puede ser en firebase Storage o otro sitio y en la base de datos guardar la url de acceso a cada imagen.
      Espero que tengas suerte con tu idea y si quieres comentar por aquí que solución has utilizado al final por si sirve de ayuda a alguien serás bien recibido.

      Un saludo y gracias por comentar :)https://reviblogdotnet.wordpress.com/wp-admin/edit-comments.php#comments-form

  2. Eduardo te felicito por este Tutorial, me ha sido de gran utilidad, pero… sabes de algún manual, libro de Ionic y Firebase?? que pueda ser recomendable… Te lo agradecería , pues estoy muy interesado en aprenderlo a fondo, Muchas Gracias por todo Eduardo.

    1. Hola José María,

      La verdad es que buscando libros sobre ionic 2 solo he encontrado un par de ellos en amazon en ingles y bastante caros, si lo que te interesa es aprenderlo a fondo, teniendo ya una base lo mejor es acudir directamente a la documentación oficial de ionic y firebase respectivamente, no obstante si encuentras algún libro o tutorial al respecto que creas que merece la pena no dudes en recomendarlo en los comentarios para todos los que podamos estar interesados.

      Muchas gracias por comentar.

      Un saludo

  3. Hola Eduardo, estoy intentando añadir cosas a la app de tu tutorial para aprende más pero me encuentro con problemas para acceder a la base de datos de firebase cuando tengo un nivel más, a ver si me puedes echar una manita.

    En tu tutorial para acceder a la base de datos de firebase utilizas:

    return this.afdb.list(‘sitios/’+this.auth.getUser());

    Pero como accedería si mi json tiene otro nivel?

    He eliminado los usuarios para simplificar

    Por ejemplo :

    {
    “sitios”: {
    “Barcelona”:{
    “sitio1”:{
    “name”: “SH6-01”,
    “url-foto”: “http://sitio1.png”},
    “sitio2”:{
    “name”: “SH6-01”,
    “url-foto”: “http://sitio1.png”}
    },
    “Madrid”:{
    “sitio1”:{
    “name”: “SH6-01”,
    “url-foto”: “http://sitio1.png”
    }
    }
    }
    }

    En mi html me gustaría separar los ngFor por Ciudades, es decir, que me liste los de Barcelona por un lado y los de Madrid por otro por si luego me interesa filtrar una búsqueda por ejemplo, pero si lo hago de esta forma me da error, he buscado mucha info pero no me aclaro.

  4. Hola Eduardo, estoy aprendiendo Ionic3 con tu tutorial y encuentro un error al intentar borrar un sitio. La pregunta es ¿como le pasas el id del sitio que se quiere borrar? Justo no está el HTML con el template (¿listado.html tal ves?) y la llamada al método borrarSitio(id).
    He intentado algo así: on-tap=”borrarSitio({{item.$key}})” pero no funciona, tira:
    Error: ExpressionChangedAfterItHasBeenCheckedError: Expression has changed after it was checked. Previous value: ‘undefined’. Current value: ‘[[item.$key]]’.
    He intentado con [[]] y con {{}}, y es lo mismo.
    Alguna idea?
    Gracias de antemano.

    1. Hola Adrián, Si has seguido el tutorial completo no tienes que modificar el html de la pagina listado. Para eliminar un sitio lo haces así (click)=”borrarSitio(sitio.id)”.
      El html los puedes ver en este post cuando utilizábamos la base de datos local en lugar de firebase:
      https://reviblog.net/2017/06/02/tutorial-de-ionic-2-crear-una-aplicacion-para-guardar-nuestros-sitios-geolocalizados-parte-8-eliminar-un-sitio-desde-el-listado-deslizando-el-item-con-itemsliding/

      Un saludo

  5. Muchas gracias por el curso, es una pasada, me ha encantado. He tenido un problema al final, ¿me podrías ayudar?

    Al tratar de suscribirme a los sitios mediante el observable, me dice da un par de errores. Es como si no existiera la propiedad subscribe para AngularFireList.

    A la hora de hacer el import de FirebaseListObservable en firebase-db.ts también me falla, me dice que no tiene ese miembro exportable.

    [20:32:33] typescript: C:/Users/pedro/OneDrive/ionic/misSitios – firebase/src/pages/listado/listado.ts, line: 36
    Property ‘subscribe’ does not exist on type ‘AngularFireList’.

    L36: this.dbFirebase.getSitios().subscribe(sitios => {
    L37: this.sitios = sitios;

    [20:32:33] typescript: …dro/OneDrive/ionic/misSitios – firebase/src/providers/firebase-db/firebase-db.ts, line: 2
    Module ‘”C:/Users/pedro/OneDrive/ionic/misSitios – firebase/node_modules/angularfire2/database/index”‘ has
    no exported member ‘FirebaseListObservable’.

    L1: import { Injectable } from ‘@angular/core’;
    L2: import { AngularFireDatabase, FirebaseListObservable } from ‘angularfire2/database’;
    L3: import { AuthProvider } from ‘../auth/auth’;

      1. Hola Alejandra:

        Acabo de corregir la entrada, el problema era que con la última actualización FirebaseListObservable se ha quedado obsoleto.
        Ya he corregido el código, en firebase-db-ts hay que sustituir FirebaseListObservable por AngularFireList. Después en la función getSitios hay que añadirle .valueChanges() al final:

        getSitios(){
        return this.afDB.list(‘sitios/’+this.auth.getUser()).valueChanges();
        }

        Gracias por comentar.

        Un saludo

        1. en que parte se usa AngularFireList?? no logro acceder a mi base de datos..

          me sale el siguiente error:

          L2: import { Injectable } from ‘@angular/core’;
          L3: import { AngularFireDatabase,AngularFireList} from ‘angularfire2/database’;

          y me metodo es de la siguiente manera nose que parte de estoy haciendo mal :

          getSitios(){
          return this.afDB.list(‘sitios/’+this.auth.getUser()).valueChanges();
          }

          asi no pudiendo obtener mi lista para poder iterar y póder mostrarlo en mi .hmtl

    1. Hola Pedro, perdón por tardar tanto en responder, no había encontrado tiempo para investigar sobre este error.

      Con la última actualización FirebaseListObservable se ha quedado obsoleto.
      Ya he corregido el código, en firebase-db-ts hay que sustituir FirebaseListObservable por AngularFireList. Después en la función getSitios hay que añadirle .valueChanges() al final:

      getSitios(){
      return this.afDB.list(‘sitios/’+this.auth.getUser()).valueChanges();
      }

      Gracias por comentar.

      Un saludo

    1. Hola, se que escribiste hace mucho este comentario, pero estoy intentando hacer lo que vos consultabas y no puedo lograrlo, encontraste solucion? graciasss

  6. Hola, como estas? necesito hacer un consulta, Yo estoy guiandome con los tutoriales y hay algo que no se como logralo.
    yo traslade todo lo que fui aprendiendo aca , a una aplicacion que tengo que entregar para la facultad.
    en mi caso, el usuario crea publicaciones y se muestran en un card con un boton de publicar, en la base de datos se guarda igual que los sitios , como tu explicaste. Cuando el usuario hace click en el boton de “Publicar”, necesito obtener los datos de esa publicacion y es ahi donde no puedo hacerlo, puedo traer todas las publicaciones del usuario, pero no de esa publicacion en particular.Muchas graciasss

  7. Hola:
    Seguí el tutorial al pie de la letra, pero tengo un problema al momento de guardar en firebase (en sqlite se guarda correctamente).

    Al momento de agregar un nuevo sitio, toma correctamente las coordenadas y saca la fotografía, cuando presiono el botón de “guardar sitio” simplemente no lo activa y no guarda nada.

    Al debuggear la app me arroja el siguiente error:
    ERROR CONTEXT DebugContext_ {view: Object, nodeIndex: 58, nodeDef: Object, elDef: Object, elView: Object}View_ModalNuevoSitioPage_0 @ ModalNuevoSitioPage.html:31proxyClass @ vendor.js:102495DebugContext_.logError @ vendor.js:15412ErrorHandler.handleError @ vendor.js:1889IonicErrorHandler.handleError @ vendor.js:159918dispatchEvent @ vendor.js:10382(anonymous function) @ vendor.js:12717schedulerFn @ vendor.js:4728SafeSubscriber.__tryOrUnsub @ vendor.js:20820SafeSubscriber.next @ vendor.js:20758Subscriber._next @ vendor.js:20696Subscriber.next @ vendor.js:20660Subject.next @ vendor.js:24106EventEmitter.emit @ vendor.js:4696NgForm.onSubmit @ vendor.js:32363(anonymous function) @ ModalNuevoSitioPage.html:31handleEvent @ vendor.js:13963callWithDebugContext @ vendor.js:15472debugHandleEvent @ vendor.js:15059dispatchEvent @ vendor.js:10378(anonymous function) @ vendor.js:11003(anonymous function) @ vendor.js:40717t.invokeTask @ polyfills.js:3zone._inner.zone._inner.fork.onInvokeTask @ vendor.js:5125t.invokeTask @ polyfills.js:3r.runTask @ polyfills.js:3e.invokeTask @ polyfills.js:3p @ polyfills.js:2v @ polyfills.js:2
    ModalNuevoSitioPage.html:31 ERROR Error: Reference.set failed: First argument contains undefined in property ‘sitios.IFwhT6NARLdQL5DwpXJWZJPDxX33.1543958848027.address’
    at validateFirebaseData (http://localhost:8080/build/vendor.js:140684:15)
    at http://localhost:8080/build/vendor.js:140730:13
    at Object.forEach (http://localhost:8080/build/vendor.js:76606:13)
    at validateFirebaseData (http://localhost:8080/build/vendor.js:140713:14)
    at validateFirebaseDataArg (http://localhost:8080/build/vendor.js:140672:5)
    at Reference.set (http://localhost:8080/build/vendor.js:153211:9)
    at FirebaseDbProvider.webpackJsonp.288.FirebaseDbProvider.guardaSitio (http://localhost:8080/build/main.js:176:89)
    at ModalNuevoSitioPage.webpackJsonp.513.ModalNuevoSitioPage.guardarSitio (http://localhost:8080/build/0.js:137:25)
    at Object.eval [as handleEvent] (ng:///ModalNuevoSitioPageModule/ModalNuevoSitioPage.ngfactory.js:95:27)
    at handleEvent (http://localhost:8080/build/vendor.js:13963:155)

  8. Hola, bueno el tutorial… pero tengo problemas al momento de mostrar la lista desde firebase…

    me da el siguiente error…
    Uncaught TypeError: Object(…) is not a function
    at SwitchMapSubscriber.project

    al momento de ir a la funcion getSitios…

    me ayudas?!

    1. hola walter si te aparece este error Uncaught TypeError: Object(…) is not a function es por que no tienes permisos con Uncaught.
      a si instalarlos con este comando

      npm install rxjs@6 rxjs-compat@6 –save

Deja una respuesta

Tu dirección de correo electrónico no será publicada. Los campos obligatorios están marcados con *

Este sitio usa Akismet para reducir el spam. Aprende cómo se procesan los datos de tus comentarios.