3D View¶
QWC offers a full 3D view based on THREE.js and Giro3D. It allows displaying 3D objects, mainly served as 3D Tiles, on a terrain provided by a COG (Cloud Optimised GeoTIFF). Layers from the 2D view will be rendered as draped textures on the 3D terrain.

Configuring the 3D View¶
To enable the 3D view, you need to configure the plugin in config.json. A typical View3D plugin configuration may look as follows:
{
  "name": "View3D",
  "cfg": {
    "pluginOptions": {
      "Identify3D": {
        "tileInfoServiceUrl": "/api/v1/tileinfo/objinfo?tileset={tileset}&objectid={objectid}"
      },
      "LayerTree3D": {
        "importedTilesBaseUrl": "/assets/"
      },
      "TopBar3D": {
        "menuItems": [
          {"key": "ThemeSwitcher", "icon": "themes"},
          {"key": "LayerTree3D", "icon": "layers"},
          {"key": "Draw3D", "icon": "draw"},
          {"key": "Measure3D", "icon": "measure"},
          {"key": "Compare3D", "icon": "compare"},
          {"key": "HideObjects3D", "icon": "eye"},
          {"key": "MapLight3D", "icon": "light"},
          {"key": "MapExport3D", "icon": "rasterexport"},
          {"key": "ExportObjects3D", "icon": "export"},
          {"key": "Share", "icon": "share"},
          {"key": "Bookmark", "icon": "bookmark"},
          {"key": "Help", "icon": "info"},
          {"key": "Settings3D", "icon": "cog"},
          {"key": "Settings", "icon": "cog"}
        ],
        "toolbarItems": [
          {"key": "Measure3D", "icon": "measure"}
        ]
      }
    }
  }
}
NOTE:
- The menuItemsandtoolbarItemscan contain both 3D tools as well as generic tools (which are not specifically 2D map tools). If you run the 3D View in split-screen 3D+2D mode, only 3D tools will be displayed in the 3D window toolbar and menu. If you run the 3D View in fullscreen, all configured entries will be displayed.
- Consult the View3D plugin reference as well as well as the 3D tool plugins reference for additional configuation options.
Next, to add a 3D View to a theme, add a map3d configuration to the desired theme item in themesConfig.json:
"map3d": {
    "extent": {
      "bounds": [<xmin>,<ymin>,<xmax>,<ymax>],
      "crs": "<epsg_code>"
    },
    "initialView": {
      "camera": [x, y, z],
      "target": [x, y, z],
      "personHeight": h
    },
    "dtm": {"url": "<url_to_dtm.tif>", "crs": "<dtm_epsg_code>},
    "basemaps": [
         {"name": "<name_of_background_layer>", "visibility": true, "overview": true},
         {"name": "<name_of_background_layer>"},
         ...
    ],
    "tiles3d": [
         {
             "name": "<name>",
             "url": "<url_to_tileset.json>",
             "title": "<title>",
             "baseColor": "<css RGB(A) color string>",
             "idAttr": "<tile_feature_attr>",
             "styles": {"<styleName>", "<url_to_tilesetStyle.json>", ...},
             "style": "<styleName>",
             "colorAttr": "<tile_feature_attr>",
             "alphaAttr": "<tile_feature_attr>",
             "labelAttr": "<tile_feature_attr>",
         }
    ],
    "objects3d": [
        {
             "name": "<name>",
             "url": "<url_to_file.gltf>",
             "title": "<title>"
        }
    ]
}
Where:
- extentis optional and allows setting the full extent of the 3D view. If not set, the extent of the 2D view is used.
- initialViewis optional and allows to define the initial view when opening the 3D view. If- personHeightis specified and greater than 0, the first-person view is activated. If not specified, the 2D view is synchronized.
- The dtmURL should point to a cloud optimized GeoTIFF.
- 
The background layer names refer to the names of the entries defined in backgroundLayersin thethemesConfig.json. Additionally:- visibilitycontrols the initially visibile background layer
- overview: truecontrols the name of background layer to display in the overview map. If no background layer is marked with- overview: true, the currently visibile background layer id dipslayed in the overview map.
 
- 
The tiles3dentry contains an optional list of 3d tiles to add to the scene, with:- idAttr: feature properties table attribute which stores the object id, used for styling and passed to- tileInfoServiceUrlof the- Identify3Dplugin. Default:- id.
- styles: optional, available tileset styles. Takes precedente over- colorAttr,- alphaAttr,- labelAttr.
- style: optional, tileset style enabled by default.
- baseColor: the fallback color for the tile objects, defaults to- white.
- colorAttr: optional, feature properties table attribute which stores the feature color, as a- 0xRRGGBBinteger.
- alphaAttr: optional, feature properties table attribute which stores the feature alpha (transparency), as an integer between- 0and- 255.
- labelAttr: optional, feature properties table attribute which stores the feature label, displayed above the geometry.
 
- 
The objects3dentry contains an optional list of GLTF objects to add to the scene.
You can control whether a theme is loaded by default in 2D, 3D or splitscreen 2D/3D view via startupView in the theme item configuration.
NOTE: If a theme contains a map3d configuration, a view switcher map button will automatically be displayed.
Permissions ¶
To restrict 3D tiles configured in tiles3d of a map3d theme configuration, create a 3D Tiles Tileset resource named according to the name of the tiles3d entry and create permissions assigning the desired roles to the resources.
Styling 3D tiles¶
The tileset style JSON is a 3D Tiles stylesheet,
of which currently the color section is supported, and which may in addition also contain a featureStyles section as follows:
{
    "color": {
       ...
    },
    "featureStyles": {
      "<object_id>": {
          "label": "<label>",
          "labelOffset": <offset>,
          "color": "<css RGB(A) color string>"
      }
   }
}
Where:
- labelis an optional string with which to label the object.
- labelOffsetis an optional number which represents the vertical offset between the object top and the label. Defaults to- 80.
- coloris an optional CSS color string which defines the object color.
Note:
- The color declarations in the featureStylessection override any color resulting from a color expression in thecolorsection.
- You must ensure that your 3D tiles properties table contains all attributes which are referenced as variables in a color expression!
3D Tile Info Service¶
The qwc-3d-tile-info-service serves two purposes:
- It provides additional info attributes for 3D Tiles via /objinfo?tileset=<tileset_name>&objectid=<feature_id>.
- It provides 3D Tiles stylesheets, converted from 2D SLDstyles via/tileinfo/stylesheet?tileset=<tileset_name>&stylename=<style_name>.
To set up the qwc-3d-tile-info-service, add the following to docker-compose.yml:
  qwc-3d-tile-info-service:
    image: docker.io/sourcepole/qwc-3d-tile-info-service:<tag>
    environment:
      <<: *qwc-service-variables
      SERVICE_MOUNTPOINT: '/api/v1/tileinfo'
    volumes:
      - ./pg_service.conf:/srv/pg_service.conf:ro
      - ./volumes/config:/srv/qwc_service/config:ro
      - ./volumes/qgs-resources:/data:ro
and configure the route in the api-gateway/nginx.conf:
location /api/v1/tileinfo {
    proxy_pass http://qwc-3d-tile-info-service:9090;
}
Then, assuming a buildings 3D Tiles dataset configured as
"tiles3d":[
  {
    "name": "buildings",
    "url":" :/assets/3dtiles/buildings/tileset.json",
    "title":"Buildings",
    "idAttr":"gml_id",
    "styles":{"default":"/api/v1/tileinfo/stylesheet?tileset=buildings&stylename=default"},
    "style": "default"
  }
]
a possible tileinfo configuration in tenantConfig.json may look as follows:
{
      "name": "tileinfo",
      "config": {
        "info_datasets": {
          "buildings" : {
            "dataset": "/data/buildings2d.gpkg",
            "layername": "buildings2d",
            "idfield": "gml_id",
            "type": "gpkg",
            "attribute_aliases": {
              "function": "Building function",
              "description": "Description"
            },
            "attribute_blacklist": [
              "gml_id"
            ],
            "styles": {
              "default": {
                "filename": "/data/buildings.sld"
                // or
                "query": "SELECT styleSLD FROM layer_styles WHERE f_table_name = 'buildings'"
              }
            }
          }
        }
      }
    }
NOTE:
- The idAttrin thetiles3dentry of the themes configuration specifies the tile feature properties attribute which contains the feature id passed to/objinfo.
- The idfieldin theinfo_datasetsentry of the service configuration specifies the feature id attribute of thebuildings2ddataset to use for looking up the feature id passed to/objinfo.
Import¶
To import scene objects in formats other than GLTF, a ogcProcessesUrl in config.json needs to point to a BBOX OGC processes server.
Generating 3D tiles¶
QWC is specialized on 3D tiles in local coordinate reference systems. The following workflow shows all steps to a import CityGML model in EPSG:25832 into a 3DCity PostgreSQL database and generating 3D tiles with attributes.
Create docker network:
docker network create citydb-net
Set PostgreSQL password:
 export CITYDB_PASSWORD="citydb"
Start PostgreSQL database container:
docker run -t -d --rm --name citydb --network=citydb-net \
  -e POSTGRES_USER=citydb -e POSTGRES_DB=citydb -e POSTGRES_PASSWORD=$CITYDB_PASSWORD \
  -e SRID=25832 -e SRS_NAME="urn:ogc:def:crs,crs:EPSG::25832,crs:EPSG::5783" \
  -p 127.0.0.1:5439:5432 -v $PWD/db:/var/lib/postgresql/data \
  3dcitydb/3dcitydb-pg:15-3.4-5
Import CityGML file (./data/LoD2_Flughafen.gml) into 3DCityDB:
docker run -t --rm --user=$UID --network=citydb-net -v $PWD/data:/data:ro \
  3dcitydb/citydb-tool:1.0 import citygml LoD2_Flughafen.gml -H citydb -d citydb -u citydb -p $CITYDB_PASSWORD
Create a view with a filter on PolyhedralSurface geometries and joined attributes:
docker exec -it citydb psql -U citydb -d citydb -c "CREATE VIEW geometry_data_ext AS
      SELECT g.*, coalesce(p1.feature_id, g.feature_id) AS parent_feature_id, p2.val_uri AS gml_id, split_part(p3.val_string, '_', 2)::integer AS function, p4.val_string::integer AS roof_type
          FROM geometry_data g
          LEFT JOIN property p1 ON (p1.name = 'buildingPart' and p1.val_feature_id = g.feature_id)
          LEFT JOIN property p2 ON (p2.name = 'externalReference' and p2.feature_id IN (p1.feature_id, g.feature_id))
          LEFT JOIN property p3 ON (p3.name = 'function' and p3.feature_id IN (p1.feature_id, g.feature_id))
          LEFT JOIN property p4 ON (p4.name = 'roofType' and p4.feature_id = g.feature_id)
      WHERE ST_GeometryType(geometry) = 'ST_PolyhedralSurface'
      "
Create a spatial index:
docker exec -it citydb psql -U citydb -d citydb -c "CREATE INDEX ON geometry_data USING gist(st_centroid(st_envelope(geometry)))"
Generate 3D tiles with pg2b3dm:
docker run -t --rm --user $(id -u):$(id -g) -v $PWD/dataout:/app/output --network=citydb-net -e PGPASSWORD=$DB_PASSWORD geodan/pg2b3dm:2.20.2 -h citydb -U citydb -d citydb -t geometry_data_ext -c geometry --keep_projection true --attributecolumns feature_id,gml_id,function,roof_type
For an overview of 3D tile generation see this talk from FOSS4G Europe 2025.
More tools and information can be found in the Awesome 3D Tiles collection.