Makina Blog

Le blog Makina-corpus

Affichage d'un Modèle Numérique de Terrain avec Babylon.js à partir de données PostgreSQL


Récupération et exploitation de données depuis PostGIS dans le but de visualiser un Modèle Numérique de Terrain avec Babylon.js .

On possède un Modèle Numérique de Terrain sous forme de raster dans une base de données PostgreSQL (PostGIS 2), et on veut le visualiser en 3D sur le Web.

On a utilisé Python et Flask pour exécuter la requête SQL qui va extraire les données du MNT, et transformer le résultat en JSON. Babylon.JS va consommer ce JSON et, via la technologie WebGL, afficher un rendu 3D du terrain dans le navigateur.

Une démonstration du résultat est disponible ici.

Dans cet article, l'accent est mis principalement sur la requête SQL qui va nous permettre d'extraire les données.

Récupération des données

  • Requête SQL et PostGIS

Remarque : Cette partie suppose que l'on connaisse un minimum le langage SQL.

Pour alléger les calculs, on va extraire seulement les informations du raster qui intersectent une certaine zone. Il faut donc choisir une zone, de préférence rectangulaire, mais surtout comprise dans le MNT pour espérer avoir un résultat !

Ensuite, la fonction generate_series(), associée à la magie de SQL, va nous permettre de créer une sorte de ''grille'' de points 2D à partir du rectangle (i.e. un échantillonnage, comme pour le drapé de lignes).

Ainsi on a toutes les coordonnées en x et y, il ne nous reste plus qu'à récupérer les valeurs d'altitude de ces points grâce à la fonction ST_Value() de PostGIS.

/!\ On n'oublie pas de faire ceci seulement pour les points du raster qui intersectent les points créés !  (d'où le WHERE ST_Intersects… ). Sinon, la requête interroge le raster pour chaque cellule inutilement…

WITH    
    -- On part d'une zone rectangulaire arbitraire
    buffer AS (
           SELECT ST_MakePolygon('SRID=32632;LINESTRING(
                                  338166 4904645,339302 4904645,339302 4905408,
                                  338166 4905408,338166 4904645)'::geometry) AS geom),


    -- On récupère les extrémités du rectangle
    buffer_extent AS (
           SELECT ST_Xmin(geom) AS xmin, ST_Xmax(geom) AS xmax,
                  ST_Ymin(geom) AS ymin, ST_Ymax(geom) AS ymax
           FROM buffer),


    -- On subdivise les lignes et colonnes du rectangle : par pas de 50m
    columns AS (
        SELECT generate_series(b.xmin::int, b.xmax::int, 50) AS x 
        FROM buffer_extent AS b),
    lines AS (
        SELECT generate_series(b.ymin::int, b.ymax::int, 50) AS y 
        FROM buffer_extent AS b),


    -- On instancie les points de la "grille" à partir des lignes et colonnes
    points2d AS (
        SELECT ST_SetSrid(ST_MakePoint(x, y), 32632) AS geom 
        FROM lines, columns),


    -- On drape la grille de points sur le MNT pour avoir x, y, z
    drape AS (
        SELECT  ST_X(p.geom) AS x, 
                ST_Y(p.geom) AS y, 
                ST_Value(mnt.rast, p.geom) AS z
        FROM mnt, points2d p
        WHERE ST_Intersects(mnt.rast, p.geom)), -- limite aux points de la grille


    average AS (
        SELECT  AVG(z) AS avgz
        FROM drape)


-- On récupère :
SELECT  x, y, z, -- les coordonnées de tous les points
    (xmax+xmin)/2 AS avgx, (ymax+ymin)/2 AS avgy, avgz, -- les 3 moyennes pour le centre
    xmin, xmax, ymin, ymax -- les extrémités 
FROM drape, average, buffer_extent;
  • Construction des données

A partir de ces données, on construit 3 choses nécessaires ensuite pour la modélisation en 3D :

  • les points (vertices),

  • le centre du terrain (center) : utile pour la position de la caméra,

  • La résolution, qui comprend le nombre de points sur x (w_sub) et le nombre de points sur y (h_sub).

/!\ Le système de coordonnées de PostGIS est un peu différent de celui de Babylon.js  dans le sens où le premier va associer l'axe z à la hauteur (altitude) alors que pour l'autre, l'altitude sera matérialisée par l'axe des y.

Pour l'exportation des données récupérées, j'ai choisi d'utiliser le format JSON (on notera que le format choisi n'est pour l'instant qu'arbitraire, le but était surtout ici d'afficher les données pour se donner une idée de rendu).

  • Limiter la grandeur des données

J'ai remarqué que l'on avait des nombres assez grands (de l'ordre de 10000 ou 100000) mais qu'ils restaient quand même très proches. Donc pour éviter de travailler avec des nombres si grands, j'ai procédé à 2 manipulations.

   - une soustraction des positions minimums en x et en y,

   - une division par 10 de toutes les valeurs x, y et z.

  • Flask.jsonify

La transformation des données obtenues de le requête sous forme de JSON se fait grâce à la méthode jsonify de Flask.
On obtient ainsi ce genre de JSON, que l'on va pouvoir exploiter pour initialiser la scene Babylon.js.

{
 "center": [
 56.8, 
 38.15, 
 278.506438678244
 ],

 "resolution": {
 "h_sub": 15.26, 
 "w_sub": 22.72
 },

 "vertices": [
 0.0, 
 273.330004882812, 
 0.0, 
 5.0, 
 269.519995117188, 
 0.0, 
 10.0, 
 269.144995117188,
 ...
 110.0, 
 286.889990234375, 
 75.0
 ]
}

 

Exploitation des données

L'affichage est réalisé grâce à Babylon.js: une bibliothèque Javascript très puissante qui utilise la technologie WebGL pour faire relativement simplement de la 3D dans le navigateur !

La fonction CreateGround() de la classe Mesh semblait toute appropriée pour afficher ce genre de terrain : On crée un rectangle et on lui donne le nombre de subdivisions.

Ici, nos longueurs et largeurs ne possèdent pas le même nombre de points, donc pas le même nombre de subdivisions ! Il a donc fallu modifier un peu cette fonction pour passer de : 

CreateGround(name, width, height, subdivisions, scene, updatable)

à :

MyCreateGround(name, width, height, width_subdivisions, height_subdivisions, scene, updatable).

Le code des deux fonctions est similaire, mais la seconde fonctionne avec un rectangle qui n'a pas la même résolution en largeur et longueur. Ici un lien vers la nouvelle fonction.

Voici des captures d'écrans de représentations en fils de fer (wireframe) des résultats.

Pas de 100m

Pas de 50m

Pas de 20m

Formations associées

Formations Outils et bases de données

Formation PostgreSQL

Nantes Du 29 au 31 janvier 2025

Voir la Formation PostgreSQL

Formations SIG / Cartographie

Formation QGIS

Nantes Du 2 au 4 avril 2025

Voir la Formation QGIS

Formations Front end

Formation Angular

Nantes Du 21 au 23 mai 2025

Voir la Formation Angular

Actualités en lien

Une randonnée en 3D grâce à BabylonJS ! Partie 2 : Le chemin

14/05/2014

Cet article est dédié à l'explication de certains points essentiels dans le code de la première démo de Geotrek 3D. La deuxième partie explique les étapes de la création du chemin.

Voir l'article
Image
 randonnée-3D-BabylonJS-Part1

Une randonnée en 3D grâce à BabylonJS ! Partie 1 : Le terrain

12/05/2014

Cet article est dédié à l'explication de certains points essentiels dans le code de la première démo de Geotrek 3D. La première partie est consacrée à la manière de créer un terrain tuilé en 3D grâce à BabylonJS.

Voir l'article
Image
 randonnée-3D-BabylonJS-Part1

Draper des lignes sur un MNT avec PostGIS

15/10/2013

Comment obtenir des géométries 3D en drapant des géométries sur un MNT avec PostGIS 2

Voir l'article
Image
MNT postGIS

Inscription à la newsletter

Nous vous avons convaincus