En los dos artículos anteriores (I y II) del curso hemos visto cómo crear aplicaciones utilizando la nueva versión de la API v2 de Google Maps para Android y hemos descrito las acciones principales sobre el mapa.
En este último artículo de la serie nos vamos a centrar en los eventos del mapa que podemos capturar y tratar, en la creación y gestión de marcadores, y en el dibujo de lineas y polígonos sobre el mapa.
Sin más preámbulos, comencemos con los eventos. Si hacemos un poco de memoria, con la versión anterior de la API debíamos crear una nueva capa (overlay) para capturar los eventos principales de pulsación. Sin embargo, el nuevo componente de mapas soporta directamente los eventos de click, click largo y movimiento de cámara y podemos implementarlos de la forma habitual, mediante su método set correspondiente.
Así por ejemplo, podríamos implementar el evento de onclick llamando al método setOnMapClickListener() con un nuevo listener y sobrescribir el método onMapClick(). Este método recibe como parámetro, en forma de objeto LatLng, las coordenadas de latitud y longitud sobre las que ha pulsado el usuario. Si quisiéramos traducir estas coordenadas física en coordenadas en pantalla podríamos utilizar un objeto Projection (similar al que ya comentamos para la API v1), obteniéndolo a partir del mapa a través del método getProjection() y posteriormente llamando a toScreenLocation() para obtener las coordenadas (x,y) de pantalla donde el usuario pulsó.
Así, por ejemplo, si quisiéramos mostrar un Toast con todos estos datos cada vez que se pulse sobre el mapa podríamos hacer lo siguiente:
mapa.setOnMapClickListener(new OnMapClickListener() {
public void onMapClick(LatLng point) {
Projection proj = mapa.getProjection();
Point coord = proj.toScreenLocation(point);
Toast.makeText(
MainActivity.this,
"Click\n" +
"Lat: " + point.latitude + "\n" +
"Lng: " + point.longitude + "\n" +
"X: " + coord.x + " - Y: " + coord.y,
Toast.LENGTH_SHORT).show();
}
});
De forma similar podríamos implementar el evento de pulsación larga, con la única diferencia de que lo asignaríamos mediante setOnMapLongClickListener() y sobrescribiríamos el método onMapLongClick().
mapa.setOnMapLongClickListener(new OnMapLongClickListener() {
public void onMapLongClick(LatLng point) {
Projection proj = mapa.getProjection();
Point coord = proj.toScreenLocation(point);
Toast.makeText(
MainActivity.this,
"Click Largo\n" +
"Lat: " + point.latitude + "\n" +
"Lng: " + point.longitude + "\n" +
"X: " + coord.x + " - Y: " + coord.y,
Toast.LENGTH_SHORT).show();
}
});
Así, cuando el usuario hiciera una pulsación larga sobre el mapa veríamos los datos en pantalla de la siguiente forma:

También podremos capturar el evento de cambio de cámara, de forma que podamos realizar determinadas acciones cada vez que el usuario se mueve manualmente por el mapa, desplazándolo, haciendo zoom, o modificando la orientación o el ángulo de visión. Este evento lo asignaremos al mapa mediante su método setOnCameraChangeListener() y sobrescribiendo el método onCameraChange(). Este método recibe como parámetro un objeto CameraPosition, que ya vimos en el artículo anterior, por lo que podremos recuperar de él todos los datos de la cámara en cualquier momento.
De esta forma, si quisiéramos mostrar un Toast con todos los datos podríamos hacer lo siguiente:
mapa.setOnCameraChangeListener(new OnCameraChangeListener() {
public void onCameraChange(CameraPosition position) {
Toast.makeText(
MainActivity.this,
"Cambio Cámara\n" +
"Lat: " + position.target.latitude + "\n" +
"Lng: " + position.target.longitude + "\n" +
"Zoom: " + position.zoom + "\n" +
"Orientación: " + position.bearing + "\n" +
"Ángulo: " + position.tilt,
Toast.LENGTH_SHORT).show();
}
});
Hecho esto, cada vez que el usuario se mueva por el mapa veríamos lo siguiente:
El siguiente tema importante que quería tratar en este artículo es el de los marcadores. Rara es la aplicación Android que hace uso de mapas sin utilizar también este tipo de elementos para resaltar determinados puntos en el mapa. Si recordamos el artículo sobre la API v1, vimos cómo podíamos añadir marcadores añadiendo una nueva capa (overlay) al mapa y dibujando nuestro marcador como parte de su evento draw(). En la nueva versión de la API tendemos toda esta funcionalidad integrada en la propia vista de mapa, y agregar un marcador resulta tan sencillo como llamar al método addMarker() pasándole la posición en forma de objeto LatLng y el texto a mostrar en la ventana de información del marcador. En nuestra aplicación de ejemplo añadiremos un menú de forma que cuando lo pulsemos se añada automáticamente un marcador sobre España con el texto “Pais: España“. Veamos cómo escribir un método auxiliar que nos ayuda a hacer esto pasándole las coordenadas de latitud y longitud:
private void mostrarMarcador(double lat, double lng)
{
mapa.addMarker(new MarkerOptions()
.position(new LatLng(lat, lng))
.title("Pais: España"));
}
Así de sencillo, basta con llamar al método addMarker() pasando como parámetro un nuevo objeto MarkerOptions sobre el que establecemos la posición del marcador (método position()) y el texto a incluir en la ventana de información del marcador (métodos title() para el título y snippet() para el resto del texto). Si ejecutamos la aplicación de ejemplo y pulsamos el menú “Marcadores” aparecerá el siguiente marcador sobre el mapa (la ventana de información aparece si además se pulsa sobre el marcador):
Además de facilitarnos la vida con la inclusión de marcadores en el mapa también tendremos ayuda a la hora de capturar el evento de pulsación sobre un marcador, ya que podremos asignar al mapa dicho evento como cualquiera de los comentados anteriormente. En este caso el evento se asignará al mapa mediante el método setOnMarkerClickListener() y sobrescribiremos el método onMarkerClick(). Dicho método recibe como parámetro el objeto Marker pulsado, de forma que podamos identificarlo accediendo a su información (posición, título, texto, …). Veamos un ejemplo donde mostramos un toast con el título del marcador pulsado:
mapa.setOnMarkerClickListener(new OnMarkerClickListener() {
public boolean onMarkerClick(Marker marker) {
Toast.makeText(
MainActivity.this,
"Marcador pulsado:\n" +
marker.getTitle(),
Toast.LENGTH_SHORT).show();
return false;
}
});
Con el código anterior, si pulsamos sobre el marcador de España aparecerá el siguiente mensaje informativo:
Todo lo explicado sobre marcadores corresponde al comportamiento por defecto de la API, sin embargo también es posible por supuesto personalizar determinadas cosas, como por ejemplo el aspecto de los marcadores. Esto se sale un poco de este artículo, donde pretendía describir los temas más básicos, pero para quien esté interesado tan sólo decir que mediante los métodos icon() y anchor() del objeto MakerOptions que hemos visto antes es posible utilizar una imagen personalizada para mostrar como marcador en el mapa. En la documentación oficial (en inglés) podéis encontrar un ejemplo de cómo hacer esto.
Como último tema, vamos a ver cómo dibujar líneas y polígonos sobre el mapa, elementos muy comunmente utilizados para trazar rutas o delimitar zonas del mapa. Para realizar esto en la versión 2 de la API vamos a actuar una vez más directamente sobre la vista de mapa, sin necesidad de añadir overlays o similares, y ayudándonos de los objetos PolylineOptions y PolygonOptions respectivamente.
Para dibujar una linea lo primero que tendremos que hacer será crear un nuevo objeto PolylineOptions sobre el que añadiremos utilizando su método add() las coordenadas (latitud-longitud) de todos los puntos que conformen la linea. Tras esto estableceremos el grosor y color de la linea llamando a los métodos width() y color() respectivamente, y por último añadiremos la linea al mapa mediante su método addPolyline() pasándole el objeto PolylineOptions recién creado.
En nuestra aplicación de ejemplo he añadido un nuevo menú para dibujar un rectángulo sobre España. Veamos cómo queda:
private void mostrarLineas()
{
//Dibujo con Lineas
PolylineOptions lineas = new PolylineOptions()
.add(new LatLng(45.0, -12.0))
.add(new LatLng(45.0, 5.0))
.add(new LatLng(34.5, 5.0))
.add(new LatLng(34.5, -12.0))
.add(new LatLng(45.0, -12.0));
lineas.width(8);
lineas.color(Color.RED);
mapa.addPolyline(lineas);
}
Ejecutando esta acción en el emulador veríamos lo siguiente:
Pues bien, esto mismo podríamos haberlo logrado mediante el dibujo de polígonos, cuyo funcionamiento es muy similar. Para ello crearíamos un nuevo objeto PolygonOptions y añadiremos las coordenadas de sus puntos en el sentido de las agujas del reloj. En este caso no es necesario cerrar el circuito (es decir, que la primera coordenada y la última fueran iguales) ya que se hace de forma automática. Otra diferencia es que para polígonos el ancho y color de la linea los estableceríamos mediante los métodos strokeWidth() y strokeColor(). Además, el dibujo final del polígono sobre el mapa lo haríamos mediante addPolygon(). En nuestro caso quedaría como sigue:
//Dibujo con polígonos
PolygonOptions rectangulo = new PolygonOptions()
.add(new LatLng(45.0, -12.0),
new LatLng(45.0, 5.0),
new LatLng(34.5, 5.0),
new LatLng(34.5, -12.0),
new LatLng(45.0, -12.0));
rectangulo.strokeWidth(8);
rectangulo.strokeColor(Color.RED);
mapa.addPolygon(rectangulo);
El resultado al ejecutar la acción en el emulador debería ser exactamente igual que el anterior.
Puedes consultar y/o descargar el código completo de los ejemplos desarrollados en este artículo accediendo a la pagina del curso en GitHub.
Y con esto habríamos concluido la serie de tres artículos destinados a describir el funcionamiento básico de la nueva API v2 de Google Maps para Android. Espero haber aclarado las dudas principales a la hora de comenzar a utilizar la nueva API. Tampoco descarto algún artículo adicional para comentar temas algo más avanzados sobre esta API, pero eso será más adelante.




solo una pregunta, aún me causa error al correrlo en el emulador
sabes si hay una forma de arreglar esto ya que no tengo cell con adroid
Debes saber que gracias a al curso que estas publicando, y que llevo siguiendo desde hace mas de un año, estoy aprendiendo a programar para Android (Ya he donado en dos ocasiones ;) ).
Queria consultarte si tienes pensado sacar algun capitulo sobre creacion de servicios.
Gracias.
PD:para cuando la V3 del Curso?
Hola Álvaro, gracias por los comentarios y la colaboración, y me alegro de que el curso te haya servido. En cuanto al tema de los servicios la respuesta es Sí, tengo previsto incluir un par de artículos dedicados a estos componentes, lo que no sabría decir exactamente es cuándo podré hacerlo. Como podrás comprobar por el ritmo de publicaciones, últimamente me cuesta un poco encontrar el tiempo necesario para actualizar el curso, pero no te preocupes porque este tema lo incluiré. Y respecto a la versión 3 del PDF ya está en marcha, irá acompañada de la actualización de muchos artículos, y espero poder publicarla en breve, avisaré en el blog y en twitter cuando esté disponible.
Hola Salvador, es poca la información que hay todavía sobre la api v2 de googlemaps para android y es de agradecer todo el esfuerzo que estás realizando.
Estoy intentando acceder desde un marcador en el mapa a una nueva activity y me pregunto cómo podría realizar ésto con el método OnMarkerClickListener.
También me gustaría saber si actualmente existe alguna manera de mostrar la ubicación actual mediante LocationSource.OnLocationChangedListener.
Hola Salvador,
muchas gracias por tus tutoriales, son con diferencia, los mejores.
Estoy intentando implementar el geocoder en la v2 pero sin éxito, no se que estoy haciendo mal.
Tendrías a mano un ejemplo rápido de como hacerlo?
Muchas gracias y saludos.
Genial la vista móvil de este blog.
Estupendo tutorial me ha quedado muy claro y e migrado mi aplicación de v1 a v2 y se nota mucho la fluidez de la nueva versión.
me surge una duda y es como añadir mas de un punto yo antes convertía mi ArrayList que contenía los puntos de una base de datos en ArrayList los paso al Overlay y listo.
Pero estoy teniendo problemas al convertir un ArrayList al nuevo tipo ArrayList.
Podrías ayudarme he googleado pero este asunto es bastante nuevo y no encuentro nada con mi problema.
Y como tu estas echo un fenómeno y he aprendido todo Android de ti seguro que sacas un as de la manga jejeje.
un saludo.
Perdona al final lo resolví es una ventaja en la versión v2 en un solo array de tipo LatLng guarda la longitud y latitud en una misma posición del arras.
LatLng coord = new LatLng(Double.parseDouble(latitud.get(i)), Double.parseDouble(longitud.get(i)));
lat.add(coord);
y después se puede acceder a latitud y longitud contenida en la misma posición de la forma mas simple ahora que lo se claro jejeje.
.position(new LatLng(lat.get(i).latitude,lat.get(i).longitude));
un saludo espero que le sirva a alguien.
gracias y sigue con los tutoriales que a muchos no ayudas un montón
muy buen tutorial, te felicito y gracias
en android 4.0.4 no me mostraba el title y el snippet por default, solo lo mostraba despues de dar click sobre el marker, para mostrarlo en automatico cuando se abre el mapa tuve que llamar al metodo showInfoWindow del marker
¿Alguien ha implementado algo similar o ha usado algo tipo MyLocationOverlay?
Los marcadores, ¿hay alguna forma de personalizarlos totalmente? Es decir, al pulsar sobre ellos mostrar una ventana personalizada totalmente, con botones, imágenes, texto, que permitan un intent a otra actividad…
Hola amigo
muy bueno el tutorial pero tengo una consulta, necesito hacer un infowindow personalizado con un par de botones dentro que hagan diferentes acciones pero resulta que los botones dentro del globo no detectan elevento clic sobre ellos, que me recomiendas o me puedes ayudar para poder hacerlo
gracias
Excelente tutorial!
Solo tengo una duda: Como puedo correr la aplicación en un dispositivo como si fuera en el emulador de forma que se vea el mapa para poder hacer debug cuando lo necesite?
Hola, he puesto todo tal cuál en el tuto pero a la hora de ejecutar la aplicación solo se muestra el tile del mapa en blanco (y los botones de zoom y obtener posición actual) ya he tratado corregirlo generando nuevas firmas, poniendo compatibilidad, he hecho de todoooooo! pero aun así no logro que el mapa se muestre. Agradezco me puedan colaborar
Hola:
muy buenos los tutoriales, casi todo lo que se es gracias a ellos.
Me gustaría saber cómo calcular la ruta de un punto a otro para poder dibujarla en el mapa.
Hola,
Hace algún tiempo aparqué el tema de intentar aplicar geocoder con la v2, como indiquñe en un comentario anterior en este mismo hilo, pero ahora me veo en la tesitura de implantarlo y continuo sin éxito.
Alguien ha conseguido implantarlo?
Muchas gracias y saludos.
En el método
public boolean onOptionsItemSelected(MenuItem item)
{
switch(item.getItemId())
{
case R.id.menu_marcadores:
mostrarMarcador(40.5, -3.5);
break;
case R.id.menu_lineas:
mostrarLineas();
break;
}
return super.onOptionsItemSelected(item);
}
menu_marcadores y menu_lineas no lo reconoce que debo hacer?
Hola
primero que todo déjame felicitarte por tu blog me ha ayudado muchisimo.
tengo el siguiente problema. tengo un ViewPager al que le cargo los fragment a través de un Array de Fragment. mi problema es que a ese array no puedo agregarle un MapFragment me podrías ayudar a adherir el MapFragment junto a los otros Fragment al ViewPager?
Saludos
Hola, lo primero enhorabuena por los magníficos tutoriales, me parecen magníficos.
Una duda, ¿alguien sabe como dibujar la línea entre dos puntos del mapa? Tipo aplicación de los GPS de los coches.
Gracias!! Saludos
Hola, lo primero enhorabuena por el excelente post.
Quería preguntarte cómo puedo hacer para que no se pinten en el mapa los marcadores en un punto por defecto cuando la respuesta web de la codificación geográfica no es correcta porque se ha introducido mal el código postal o address.
Es decir, si hago una lista de peticiones web para geocodificación y algunas address están mal (cosa que no puedo controlar porque cojo de un servidor) se me pintan los marcadores en un punto centrado de la región del mapa en la que trabajo….me gustaría evitar esto.
Gracias crack.
Hola, de nuevo. Tengo un problema.
Estoy desarrollando una aplicación y antes con la versión 2 de Google Play Services me pintaba los mapas sin problema tal y como tú explicas en este post, pues bien, desde hace un tiempo no me pinta la aplicación los mapas, entonces consulté el SDK Manager y ví que había una nueva actualización para Google Play Services. Pensé que al actualizar volvería a pintarmelos sin problemas, cual es mi sorpresa que ahora se ha empeorado el asunto porque el código no me reconoce ni las clases que antes empleaba para pintar los mapas!!!! Ruego me dé cualquier tipo de ayuda para volver a pintar los mapas como hacía hasta hace bien poco, si tengo que desinstalar la actualización de Google Play Services, si hay una nueva forma de pintar los mapas en una app Android….ayuda por favor!!
Gracias, saludos.
Hay alguna manera de poder implementar con un textview y un botón una herramienta de búsqueda de localizaciones en el mapa? (no uso serachview ara poder usarlo en apis de android antiguas y hacer más universal la aplicación).
Me he vuelto loco buscando en inglés, en español,…y no encuentro ninguna que no esté obsoleta o haga referencia a la api v1 de los mapas. Y veo que hacen referencia algunos a la Api de Location, pero eso no es más bien ya para usar aplicaciones de localización vía gps u otros medios? La verdad es que me está generando bastante confusión este tema.
Por otro lado, muy buen blog. En castellano hay dos o tres buenos y este es uno de ellos.