[gnome-maps/wip/jonasdn/vector-tiles] Add support to render with vector-tile-glib



commit a6f5f09cd059750b826b08a50d42c76e9b4a007d
Author: Jonas Danielsson <jonas threetimestwo org>
Date:   Sat Apr 18 12:22:51 2015 +0200

    Add support to render with vector-tile-glib

 configure.ac               |    2 +
 data/Makefile.am           |    4 +-
 data/gnome-maps.mapcss     |   85 +++++++++++++++++
 lib/Makefile.am            |   28 ++++--
 lib/maps-mapbox-renderer.c |  223 ++++++++++++++++++++++++++++++++++++++++++++
 lib/maps-mapbox-renderer.h |   75 +++++++++++++++
 src/mapView.js             |   42 ++++++++-
 7 files changed, 450 insertions(+), 9 deletions(-)
---
diff --git a/configure.ac b/configure.ac
index 4d7b796..dced899 100644
--- a/configure.ac
+++ b/configure.ac
@@ -50,6 +50,8 @@ PKG_CHECK_MODULES(GNOME_MAPS_LIB, [
     folks                        >= $FOLKS_MIN_VERSION
     geocode-glib-1.0             >= $GEOCODE_MIN_VERSION
     champlain-0.12               >= $CHAMPLAIN_MIN_VERSION
+    vector-tile-glib-1.0
+    cairo
 ])
 AC_SUBST(GNOME_MAPS_LIB_CFLAGS)
 AC_SUBST(GNOME_MAPS_LIB_LIBS)
diff --git a/data/Makefile.am b/data/Makefile.am
index e0ce3e9..9c7511e 100644
--- a/data/Makefile.am
+++ b/data/Makefile.am
@@ -12,7 +12,9 @@ org.gnome.Maps.data.gresource: org.gnome.Maps.data.gresource.xml $(app_resource_
                --sourcedir=$(srcdir) $<
 
 resourcedir = $(pkgdatadir)
-resource_DATA = org.gnome.Maps.data.gresource
+resource_DATA =                                                        \
+       org.gnome.Maps.data.gresource                                   \
+       gnome-maps.mapcss
 
 appsdir = $(datadir)/applications
 apps_DATA = org.gnome.Maps.desktop
diff --git a/data/gnome-maps.mapcss b/data/gnome-maps.mapcss
new file mode 100644
index 0000000..a59df25
--- /dev/null
+++ b/data/gnome-maps.mapcss
@@ -0,0 +1,85 @@
+canvas {
+    fill-color: #FAEBD7;
+}
+
+way {
+    color: #777777;
+    width: 1;
+    linecap: round;
+}
+
+way[highway] {
+    color: #ffffff;
+    linecap: round;
+    casing-linecap: round;
+    linejoin: miter;
+    casing-linejoin: miter;
+    casing-width: 1;
+    width: 5;
+    casing-color: #C7B8A4;
+}
+
+way[highway][roads=minor_road] {
+    width: 3;
+}
+
+way[highway=motorway], way[highway=motorway_link],
+way[highway=trunk], way[highway=trunk_link] {
+    casing-width: 0.95;
+    color: #ff9966;
+    casing-color: #bb5500;
+    width: 5;
+}
+way[highway=primary], way[highway=primary_link] {
+    casing-width: 0.8;
+    width: 2;
+    color: #ffbb88;
+    casing-color: #cc6600;
+}
+
+way[highway=footway] {
+    color: #770000;
+    width: 1;
+    dashes: 2, 2;
+}
+
+way[highway=cycleway] {
+    color: #000077;
+    width: 1;
+    dashes: 2, 2;
+}
+
+area {
+    color: #FAEBD7;
+    fill-color: #FAEBD7;
+    width: 1;
+}
+
+area[landuse=park], area[landuse=playground], area[landuse=pitch] {
+    color: #73D216;
+    fill-color: #73D216;
+}
+
+area[water], way[water], area[natural=water] {
+    color: #4A90D9;
+    fill-color: #4A90D9;
+}
+
+area[building] {
+    color: #777777;
+    fill-color: #C7B8A4;
+}
+
+way[is_bridge=yes] {
+    casing-width: 1;
+    casing-color: #000000;
+}
+
+way[is_tunnel=yes] {
+    color: #333333;
+    width: 2;
+    dashes: 3, 3;
+}
+
+
+
diff --git a/lib/Makefile.am b/lib/Makefile.am
index 35203da..f7e83f7 100644
--- a/lib/Makefile.am
+++ b/lib/Makefile.am
@@ -4,8 +4,17 @@ BUILT_SOURCES =                                                        \
        maps-enum-types.c                                               \
        maps-enum-types.h
 
-libgnome_maps_headers_private = maps-contact-store.h maps-contact.h maps.h
-libgnome_maps_sources = maps-contact-store.c maps-contact.c
+libgnome_maps_headers_private =                                        \
+       maps-contact-store.h                                            \
+       maps-contact.h                                                  \
+       maps-mapbox-renderer.h                                          \
+       maps.h
+
+libgnome_maps_sources =                                                \
+       maps-contact-store.c                                            \
+       maps-contact.c                                                  \
+       maps-mapbox-renderer.c
+
 libgnome_maps_la_SOURCES =                                             \
        $(libgnome_maps_sources)                                        \
        $(libgnome_maps_headers_private)                                \
@@ -19,12 +28,12 @@ AM_CPPFLAGS =                                                               \
 
 maps-enum-types.h: $(libgnome_maps_headers_private)
        $(AM_V_GEN) ($(GLIB_MKENUMS)                                    \
-               --template $(srcdir)/maps-enum-types.h.template                 \
+               --template $(srcdir)/maps-enum-types.h.template \
                $^ > xgen-$(@F) && mv -f xgen-$(@F) $@)
 
 maps-enum-types.c: $(libgnome_maps_headers_private) maps-enum-types.h
        $(AM_V_GEN) ($(GLIB_MKENUMS)                                    \
-               --template $(srcdir)/maps-enum-types.c.template                 \
+               --template $(srcdir)/maps-enum-types.c.template \
                $^ > xgen-$(@F) && mv -f xgen-$(@F) $@)
 
 if HAVE_INTROSPECTION
@@ -37,10 +46,15 @@ GnomeMaps_1_0_gir_INCLUDES =                                                \
        GObject-2.0                                                     \
        GeocodeGlib-1.0                                         \
        Champlain-0.12
-GnomeMaps_1_0_gir_PACKAGES = gobject-2.0 geocode-glib-1.0
+GnomeMaps_1_0_gir_PACKAGES = gobject-2.0 geocode-glib-1.0 champlain-0.12 cairo
 GnomeMaps_1_0_gir_FILES = $(libgnome_maps_la_SOURCES)
-GnomeMaps_1_0_gir_CFLAGS = $(MAPS_CFLAGS) -I$(top_srcdir) -I$(top_builddir) -I$(srcdir)
-GnomeMaps_1_0_gir_LIBS = libgnome-maps.la
+GnomeMaps_1_0_gir_CFLAGS =                                             \
+       $(GNOME_MAPS_LIB_CFLAGS)                                        \
+       $(GNOME_MAPS_LIB_LIBS)                                          \
+       -I$(top_srcdir)                                         \
+       -I$(top_builddir)                                               \
+       -I$(srcdir)
+GnomeMaps_1_0_gir_LIBS = libgnome-maps.la champlain-0.12
 GnomeMaps_1_0_gir_EXPORT_PACKAGES = gnome-maps-1.0
 GnomeMaps_1_0_gir_SCANNERFLAGS =                                       \
        --symbol-prefix=maps                                            \
diff --git a/lib/maps-mapbox-renderer.c b/lib/maps-mapbox-renderer.c
new file mode 100644
index 0000000..adf148b
--- /dev/null
+++ b/lib/maps-mapbox-renderer.c
@@ -0,0 +1,223 @@
+/*
+ * Copyright (c) 2015 Jonas Danielsson
+ *
+ * GNOME Maps is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by the
+ * Free Software Foundation; either version 2 of the License, or (at your
+ * option) any later version.
+ *
+ * GNOME Maps is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
+ * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
+ * for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with GNOME Maps; if not, see <http://www.gnu.org/licenses/>.
+ *
+ * Author: Jonas Danielsson <jonas threetimestwo org>
+ */
+
+#include <vector-tile-mapbox.h>
+#include <cairo.h>
+
+#include "maps-mapbox-renderer.h"
+
+struct _MapsMapboxRendererPrivate
+{
+  gchar *data;
+  guint size;
+
+  VTileMapCSS *stylesheet;
+};
+
+typedef struct _RenderData
+{
+  ChamplainTile *tile;
+  ClutterContent *canvas;
+  gchar *data;
+  guint size;
+  VTileMapCSS *stylesheet;
+} RenderData;
+
+G_DEFINE_TYPE_WITH_PRIVATE (MapsMapboxRenderer, maps_mapbox_renderer, CHAMPLAIN_TYPE_RENDERER)
+
+static void set_data (ChamplainRenderer *renderer,
+                      const gchar *data,
+                      guint size);
+static void render (ChamplainRenderer *renderer,
+                    ChamplainTile *tile);
+
+static void
+maps_mapbox_renderer_dispose (GObject *object)
+{
+  G_OBJECT_CLASS (maps_mapbox_renderer_parent_class)->dispose (object);
+}
+
+
+static void
+maps_mapbox_renderer_finalize (GObject *object)
+{
+  MapsMapboxRenderer *renderer = MAPS_MAPBOX_RENDERER (object);
+  g_free (renderer->priv->data);
+
+  G_OBJECT_CLASS (maps_mapbox_renderer_parent_class)->finalize (object);
+}
+
+
+static void
+maps_mapbox_renderer_class_init (MapsMapboxRendererClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+  ChamplainRendererClass *renderer_class = CHAMPLAIN_RENDERER_CLASS (klass);
+
+  object_class->finalize = maps_mapbox_renderer_finalize;
+  object_class->dispose = maps_mapbox_renderer_dispose;
+
+  renderer_class->set_data = set_data;
+  renderer_class->render = render;
+}
+
+
+static void
+maps_mapbox_renderer_init (MapsMapboxRenderer *renderer)
+{
+  renderer->priv = maps_mapbox_renderer_get_instance_private (renderer);
+  renderer->priv->data = NULL;
+  renderer->priv->size = 0;
+}
+
+
+/**
+ * maps_mapbox_renderer_new:
+ *
+ * Returns: (transfer full): a new #MapsMapboxRenderer object, use g_object_unref() when done.
+ *
+ */
+MapsMapboxRenderer *
+maps_mapbox_renderer_new (void)
+{
+  return g_object_new (MAPS_TYPE_MAPBOX_RENDERER, NULL);
+}
+
+void
+maps_mapbox_renderer_load_css (MapsMapboxRenderer *renderer,
+                                    const char *filename,
+                                    GError **error)
+{
+  renderer->priv->stylesheet = vtile_mapcss_new ();
+  vtile_mapcss_load (renderer->priv->stylesheet, filename, error);
+}
+
+
+static void
+set_data (ChamplainRenderer *renderer_base, const gchar *data, guint size)
+{
+  MapsMapboxRenderer *renderer = (MapsMapboxRenderer *) renderer_base;
+
+  if (renderer->priv->data)
+    g_free (renderer->priv->data);
+
+  renderer->priv->data = g_memdup (data, size);
+  renderer->priv->size = size;
+}
+
+static void
+on_mapbox_rendered (VTileMapbox *mapbox,
+                    GAsyncResult *res,
+                    RenderData *data)
+{
+
+}
+
+gboolean
+on_canvas_draw (ClutterCanvas *canvas,
+                cairo_t *cr,
+                gint width,
+                gint height,
+                RenderData *data)
+{
+  VTileMapbox *mapbox;
+  cairo_surface_t *surface;
+  char output[512];
+  ClutterActor *actor = NULL;
+  GError *error = NULL;
+  gboolean success;
+
+  mapbox = vtile_mapbox_new (data->data,
+                             data->size,
+                             width,
+                             champlain_tile_get_zoom_level (data->tile));
+
+  vtile_mapbox_set_stylesheet (mapbox, data->stylesheet);
+
+
+  cairo_save (cr);
+  cairo_set_operator (cr, CAIRO_OPERATOR_CLEAR);
+  cairo_paint (cr);
+  cairo_restore (cr);
+  cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
+
+  vtile_mapbox_render (mapbox, cr, NULL);
+
+  actor = clutter_actor_new ();
+  clutter_actor_set_size (actor,
+                          champlain_tile_get_size (data->tile),
+                          champlain_tile_get_size (data->tile));
+  clutter_actor_set_content (actor, data->canvas);
+  g_object_unref (data->canvas);
+
+  /*  clutter_actor_set_offscreen_redirect (actor,
+      CLUTTER_OFFSCREEN_REDIRECT_AUTOMATIC_FOR_OPACITY);  */
+
+ finish:
+  if (actor)
+    champlain_tile_set_content (data->tile, actor);
+
+  g_signal_emit_by_name (data->tile, "render-complete",
+                         data->data, data->size, success);
+
+
+  g_object_unref (data->tile);
+  g_free (data->data);
+  g_free (data);
+
+  /*
+  sprintf (output, "%d-%d-%d.png\n",
+           champlain_tile_get_x (data->tile),
+           champlain_tile_get_y (data->tile),
+           champlain_tile_get_zoom_level (data->tile));
+  surface = cairo_get_target (cr);
+  cairo_surface_write_to_png (surface, output);
+  */
+
+
+  return TRUE;
+}
+
+static void
+render (ChamplainRenderer *renderer_base, ChamplainTile *tile)
+{
+  MapsMapboxRenderer *renderer = (MapsMapboxRenderer *) renderer_base;
+  RenderData *data;
+
+  if (!renderer->priv->data || renderer->priv->size == 0)
+    {
+      g_signal_emit_by_name (tile, "render-complete",
+                             renderer->priv->data, renderer->priv->size, TRUE);
+      return;
+    }
+
+  data = g_new0 (RenderData, 1);
+  data->tile = g_object_ref (tile);
+  data->data = g_memdup (renderer->priv->data, renderer->priv->size);
+  data->size = renderer->priv->size;
+  data->stylesheet = renderer->priv->stylesheet;
+  data->canvas = clutter_canvas_new ();
+
+  clutter_canvas_set_size ((ClutterCanvas *) data->canvas,
+                           champlain_tile_get_size (tile),
+                           champlain_tile_get_size (tile));
+  g_signal_connect (data->canvas, "draw",
+                    G_CALLBACK (on_canvas_draw), data);
+  clutter_content_invalidate (data->canvas);
+}
diff --git a/lib/maps-mapbox-renderer.h b/lib/maps-mapbox-renderer.h
new file mode 100644
index 0000000..d72fda4
--- /dev/null
+++ b/lib/maps-mapbox-renderer.h
@@ -0,0 +1,75 @@
+/*
+ * Copyright (c) 2015 Jonas Danielsson
+ *
+ * GNOME Maps is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by the
+ * Free Software Foundation; either version 2 of the License, or (at your
+ * option) any later version.
+ *
+ * GNOME Maps is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
+ * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
+ * for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with GNOME Maps; if not, see <http://www.gnu.org/licenses/>.
+ *
+ * Author: Jonas Danielsson <jonas threetimestwo org>
+ */
+
+#ifndef __MAPS_MAPBOX_RENDERER_H__
+#define __MAPS_MAPBOX_RENDERER_H__
+
+#include <champlain/champlain.h>
+
+G_BEGIN_DECLS
+
+#define MAPS_TYPE_MAPBOX_RENDERER maps_mapbox_renderer_get_type ()
+
+#define MAPS_MAPBOX_RENDERER(obj) \
+  (G_TYPE_CHECK_INSTANCE_CAST ((obj), MAPS_TYPE_MAPBOX_RENDERER, MapsMapboxRenderer))
+
+#define MAPS_MAPBOX_RENDERER_CLASS(klass) \
+  (G_TYPE_CHECK_CLASS_CAST ((klass), MAPS_TYPE_MAPBOX_RENDERER, MapsMapboxRendererClass))
+
+#define MAPS_IS_MAPBOX_RENDERER(obj) \
+  (G_TYPE_CHECK_INSTANCE_TYPE ((obj), MAPS_TYPE_MAPBOX_RENDERER))
+
+#define MAPS_IS_MAPBOX_RENDERER_CLASS(klass) \
+  (G_TYPE_CHECK_CLASS_TYPE ((klass), MAPS_TYPE_MAPBOX_RENDERER))
+
+#define MAPS_MAPBOX_RENDERER_GET_CLASS(obj) \
+  (G_TYPE_INSTANCE_GET_CLASS ((obj), MAPS_TYPE_MAPBOX_RENDERER, MapsMapboxRendererClass))
+
+typedef struct _MapsMapboxRendererPrivate MapsMapboxRendererPrivate;
+typedef struct _MapsMapboxRenderer MapsMapboxRenderer;
+typedef struct _MapsMapboxRendererClass MapsMapboxRendererClass;
+
+/**
+ * MapsMapboxRenderer:
+ *
+ * The #MapsMapboxRenderer structure contains only private data
+ * and should be accessed using the provided API
+ */
+struct _MapsMapboxRenderer
+{
+  ChamplainRenderer parent;
+
+  MapsMapboxRendererPrivate *priv;
+};
+
+struct _MapsMapboxRendererClass
+{
+  ChamplainRendererClass parent_class;
+};
+
+GType maps_mapbox_renderer_get_type (void);
+
+MapsMapboxRenderer *maps_mapbox_renderer_new (void);
+
+void maps_mapbox_renderer_load_css (MapsMapboxRenderer *renderer,
+                                    const char *filename,
+                                    GError **error);
+G_END_DECLS
+
+#endif /* __MAPS_MAPBOX_RENDERER_H__ */
diff --git a/src/mapView.js b/src/mapView.js
index 36c2b7c..2fb762f 100644
--- a/src/mapView.js
+++ b/src/mapView.js
@@ -21,6 +21,7 @@
 
 const Champlain = imports.gi.Champlain;
 const Clutter = imports.gi.Clutter;
+const GLib = imports.gi.GLib;
 const GObject = imports.gi.GObject;
 const Geocode = imports.gi.GeocodeGlib;
 const GtkChamplain = imports.gi.GtkChamplain;
@@ -29,6 +30,7 @@ const Lang = imports.lang;
 const Application = imports.application;
 const ContactPlace = imports.contactPlace;
 const Geoclue = imports.geoclue;
+const Maps = imports.gi.GnomeMaps;
 const MapWalker = imports.mapWalker;
 const Place = imports.place;
 const PlaceMarker = imports.placeMarker;
@@ -144,11 +146,49 @@ const MapView = new Lang.Class({
         }).bind(this));
     },
 
+    _vectorSource: function() {
+        let id = 'vector-tile-glib';
+        let renderer = new Maps.MapboxRenderer();
+        renderer.load_css(GLib.build_filenamev([pkg.pkgdatadir,
+                                                'gnome-maps.mapcss']));
+
+        let tile = Champlain.NetworkTileSource.new_full(
+            id,
+            'OpenStreetMap Mapbox vector tiles',
+            'Map Data ODBL OpenStreetMap Contributors',
+            'http://creativecommons.org/licenses/by-sa/2.0/',
+            2,
+            17,
+            256,
+            Champlain.MapProjection.MAP_PROJECTION_MERCATOR,
+            'http://vector.mapzen.com/osm/all/#Z#/#X#/#Y#.mapbox',
+            renderer);
+
+        let error = this._factory.create_error_source(256);
+        let file = Champlain.FileCache.new_full(100000000,
+                                                 null,
+                                                 renderer);
+        let mem = Champlain.MemoryCache.new_full (100, renderer);
+        let chain = new Champlain.MapSourceChain();
+
+        chain.push(error);
+        chain.push(tile);
+        chain.push(file);
+        chain.push(mem);
+
+        return chain;
+    },
+
     setMapType: function(mapType) {
         if (this.view.map_source.id === mapType)
             return;
 
-        let source = this._factory.create_cached_source(mapType);
+        let source;
+        if (mapType === MapType.AERIAL)
+            source = this._vectorSource();
+        else
+            source = this._factory.create_cached_source(mapType);
+
         this.view.map_source = source;
     },
 


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]