Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Globe - custom layers API and examples, globe dev guide #4577

Merged
merged 35 commits into from
Aug 27, 2024
Merged
Show file tree
Hide file tree
Changes from 31 commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
645e7f6
Port custom layer changes and globe docs
kubapelc Aug 19, 2024
77b1a46
Port transform changes
kubapelc Aug 19, 2024
0081170
Fix custom layer unit test
kubapelc Aug 19, 2024
2a068f3
Fix failing render tests
kubapelc Aug 19, 2024
51d0a58
Update build size
kubapelc Aug 19, 2024
0fbf464
Update globe custom layer example descritions, remove forgotten code
kubapelc Aug 19, 2024
bd10a02
Remove unused util function
kubapelc Aug 19, 2024
18f5d88
Incorporate globe docs feedback
kubapelc Aug 20, 2024
f8b46aa
Refactor and expose tile mesh generation
kubapelc Aug 20, 2024
d5d90f6
Refactor custom layers to get smaller args object and access map tran…
kubapelc Aug 20, 2024
f777b06
Simplify more of the custom layer API
kubapelc Aug 20, 2024
0148b2e
Clean up and adapt more examples
kubapelc Aug 20, 2024
11f2786
Fix mercator matrix precision
kubapelc Aug 20, 2024
4be9b35
Fix 3D model on terrain example
kubapelc Aug 20, 2024
3366971
Rename projectionDataForMercatorCoords to defaultProjectionData
kubapelc Aug 20, 2024
436ba55
Document ProjectionData type
kubapelc Aug 20, 2024
37cce84
Update build size
kubapelc Aug 20, 2024
705c8bf
Update developer-guides/globe.md
kubapelc Aug 20, 2024
588cad2
Decouple ProjectionData from rendering code
kubapelc Aug 26, 2024
e9a2730
Rename ProjectionData members
kubapelc Aug 26, 2024
5644e85
Fix mercator transform unit tests
kubapelc Aug 26, 2024
c101948
Add an example to createTileMesh
kubapelc Aug 26, 2024
082c3e9
Rename CustomRenderMethodInput.shader to shaderData
kubapelc Aug 26, 2024
0a0ec88
Add shaderData examples
kubapelc Aug 26, 2024
15b0d08
Document TileMesh and CreateTileMeshOptions types
kubapelc Aug 26, 2024
49fbf18
Fix custom layers in render tests
kubapelc Aug 26, 2024
18105dc
Update render tests
kubapelc Aug 26, 2024
ccad6e8
Add render test result from linux
kubapelc Aug 26, 2024
541f5f6
Update build size
kubapelc Aug 26, 2024
778b325
Update src/render/program/projection_program.ts
kubapelc Aug 26, 2024
02c2334
Rename createTileMeshInternal to createTileMeshWithBuffers
kubapelc Aug 26, 2024
3a2fffc
Merge remote-tracking branch 'upstream/globe' into globe-pr-custom
kubapelc Aug 26, 2024
8222f48
Update build size
kubapelc Aug 26, 2024
cf17bb6
Improve doc comments
kubapelc Aug 27, 2024
0870639
Merge remote-tracking branch 'upstream/globe' into globe-pr-custom
kubapelc Aug 27, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Binary file added developer-guides/assets/floats.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added developer-guides/assets/no_subdivision.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added developer-guides/assets/wireframe.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
195 changes: 195 additions & 0 deletions developer-guides/globe.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,195 @@
# Globe projection

This guide describes the inner workings of globe projection.
Globe draws the same vector polygons and lines as mercator projection,
ensuring a clear, unstretched image at all view angles and support for dynamic layers and geometry.

The actual projection is done in three steps:

- compute angular spherical coordinates from source web mercator tile data
- convert spherical coordinates to a 3D vector - a point on the surface of a unit sphere
- project the 3D vector using a common perspective projection matrix

So the globe is a unit sphere from the point of view of projection.
This also simplifies a lot of math, and is used extensively in the globe transform class.

Geometry is projected to the sphere in the vertex shader.

## Zoom behavior

To stay consistent with web mercator maps, globe is automatically enlarged when map center is nearing the poles.
This keeps the map center visually similar to a mercator map with the same x,y and zoom.
However, when panning the globe or performing camera animations,
we do not want the planet to get larger or smaller when changing latitudes.
Map movement thus compensates for the planet size change by also
changing zoom level along with latitude changes.

This behavior is completely automatic and transparent to the user.
The only case when the user needs to be aware of this is when
programmatically triggering animations such as `flyTo` and `easeTo`
and using them to both change the map center's latitude and *at the same time*
changing the map's zoom to an amount based on the map's starting zoom.
The example [globe-zoom-planet-size-function](https://maplibre.org/maplibre-gl-js/docs/examples/globe-zoom-planet-size-function/) demonstrates how to
compensate for planet size changes in this case.
All other camera animations (that either specify target zoom
that is not based on current zoom or do not specify zoom at all) will work as expected.

## Shaders

Most vertex shaders use the `projectTile` function, which
accepts a 2D vector of coordinates inside the currently drawn tile,
in range 0..EXTENT (8192), and returns its final projection that can
be directly passed to `gl_Position`.
When drawing a tile, proper uniforms must be set to convert from
these tile-local coordinates to web mercator.

The implementation of `projectTile` is automatically injected into the shader source code.
Different implementations can be injected, depending on the currently active projection.
Thanks to this many shaders use the exact same code for both mercator and globe,
although there are shaders that use `#ifdef GLOBE` for globe-specific code.

## Subdivision

If we were to draw mercator tiles with globe shaders directly, we would end up with a deformed sphere.
This is due to how polygons and lines are triangulated in MapLibre - the earcut algorithm
creates as few triangles as possible, which can sometimes result in huge triangles, for example in the oceans.
This behavior is desirable in mercator maps, but if we were to project the vertices of such large triangles to globe directly,
we would not get curved horizons, lines, etc.
For this reason, before a tile is finished loading, its geometry (both polygons and lines) is further subdivided.

The figure below demonstrates how globe would look without subdivision.
Note the deformed oceans, and the USA-Canada border that is not properly curved.

![](assets/no_subdivision.png)

It is critical that subdivision is as fast as possible, otherwise it would significantly slow down tile loading.
Currently the fastest approach seems to be taking the output geometry from `earcut` and subdividing that further.

When modifying subdivision, beware that it is very prone to subtle errors, resulting in single-pixel seams.
Subdivision should also split the geometry in consistent places,
so that polygons and lines match up correctly when projected.

We use subdivision that results in a square grid, visible in the figure below.

![](assets/wireframe.png)

Subdivision is configured in the Projection object.
Subdivision granularity is defined by the base tile granularity and minimal allowed granularity.
The tile for zoom level 0 will have base granularity, tile for zoom 1 will have half that, etc.,
but never less than minimal granularity.

The maximal subdivision granularity of 128 for fill layers is enough to get nicely curved horizons,
while also not generating too much new geometry and not overflowing the 16 bit vertex indices used throughout MapLibre.

Raster tiles in particular need a relative high base granularity, as otherwise they would exhibit
visible warping and deformations when changing zoom levels.

## Floating point precision & transitioning to mercator

Shaders work with 32 bit floating point numbers (64 bit are possible on some platforms, but very slow).
The 23 bits of mantissa and 1 sign bit can represent at most around 16 million values,
but the circumference of the earth is roughly 40 000 km, which works out to
about one float32 value per 2.5 meters, which is insufficient for a map.
Thus if we were to use globe projection at all zoom levels, we would unsurprisingly encounter precision issues.

![](assets/floats.png)

To combat this, globe projection automatically switches to mercator projection around zoom level 12.
This transition is smooth, animated and can only be noticed if you look very closely,
because globe and mercator projections converge at high zoom levels, and around level 12
they are already very close.

The transition animation is implemented in the shader's projection function,
and is controlled by a "globeness" parameter passed from the transform.

## GPU "atan" error correction

When implementing globe, we noticed that globe projection did not match mercator projection
after the automatic transition described in previous section.
This mismatch was very visible at certain latitudes, the globe map was shifted north/south by hundreds of meters,
but at other latitudes the shift was much smaller. This behavior was also inconsistent - one would
expect the shift to gradually increase or decrease with distance from equator, but that was not the case.

Eventually, we tracked this down to an issue in the projection shader, specifically the `atan` function.
On some GPU vendors, the function is inaccurate in a way that matches the observed projection shifts.

To combat this, every second we draw a 1x1 pixel framebuffer and store the `atan` value
for the current latitude, asynchronously download the pixel's value, compare it with `Math.atan`
reference, and shift the globe projection matrix to compensate.
This approach works, because the error is continuous and doesn't change too quickly with latitude.

This approach also has the advantage that it works regardless of the actual error of the `atan`,
so MapLibre should work fine even if it runs on some new GPU in the future with different
`atan` inaccuracies.

## Clipping

When drawing a planet, we need to somehow clip the geometry that is on its backfacing side.
Since MapLibre uses the Z-buffer for optimizing transparency drawing, filling it with custom
values, we cannot use it for this purpose.

Instead, we compute a plane that intersects the horizons, and for each vertex
we compute the distance from this plane and store it in `gl_Position.z`.
This forces the GPU's clipping hardware to clip geometry beyond the planet's horizon.
This does not affect MapLibre's custom Z values, since they are set later using
`glDepthRange`.

However this approach does not work on some phones due to what is likely a driver bug,
which applies `glDepthRange` and clipping in the wrong order.
So additionally, face culling is used for fill and raster layers
(earcut does not result in consistent winding order, this is ensured during subdivision)
and line layers (which have inconsistent winding order) discard beyond-horizon
pixels in the fragment shader.

## Raster tiles

Drawing raster tiles under globe is somewhat more complex than under mercator,
since under globe they are much more prone to having slight seams between tiles.
Tile are drawn as subdivided meshes instead of simple quads, and the curvature
near the edges can cause seams, especially in cases when two tiles of different
zoom levels are next to each other.

To make sure that there are both no seams and that every pixel is covered by
valid tile texture (as opposed to a stretched border of a neighboring tile),
we first draw all tiles *without* border, marking all drawn pixels in stencil.
Then, we draw all tiles *with* borders, but set stencil to discard all pixels
that were drawn in the first pass.

This ensures that no pixel is drawn twice, and that the stretched borders
are only drawn in regions between tiles.

## Symbols

Symbol rendering also had to be adapted for globe, as well as collision detection and placement.
MapLibre computed well-fitting bounding boxes even for curved symbols under globe projection
by computing the AABB from a projection of the symbol's box' corners and box edge midpoints.
This is an approximation, but works well in practice.

## Transformations and unprojection

Most projection and unprojection functions from the transform interface are adapted for globe,
with some caveats.
The `setLocationAtPoint`function may sometimes not find a valid solution
for the given parameters.
Globe transform currently does not support constraining the map's center.

## Controls

Globe uses slightly different controls than mercator map.
Panning, zooming, etc. is aware of the sphere and should work intuitively,
as well as camera animations such as `flyTo` and `easeTo`.

Specifically, when zooming, the location under the cursor stays under the cursor,
just like it does on a mercator map.
However this behavior has some limitations on the globe.
In some scenarios, such as zooming to the edge of the planet,
this way of zooming would result in rapid and unpleasant map panning.
Thus this behavior is slowly faded out at low zooms and replaced with an approximation.

There are also other edge cases, such as when looking at the planet's poles
and trying to zoom in to a location that is on the other hemisphere ("behind the pole").
MapLibre does not support moving the camera across poles, so instead we need to rotate around.
In this case, an approximation instead of exact zooming is used as well.

Globe controls also use panning inertia, just like mercator.
Special care was taken to keep the movement speed of inertia consistent.
Binary file added docs/assets/examples/globe-3d-model.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added docs/assets/examples/globe-custom-simple.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added docs/assets/examples/globe-custom-tiles.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
98 changes: 12 additions & 86 deletions src/geo/projection/globe.ts
Original file line number Diff line number Diff line change
@@ -1,21 +1,15 @@
import type {Context} from '../../gl/context';
import type {CanonicalTileID} from '../../source/tile_id';
import {PosArray, TriangleIndexArray} from '../../data/array_types.g';
import {Mesh} from '../../render/mesh';
import {EXTENT} from '../../data/extent';
import {SegmentVector} from '../../data/segment';
import posAttributes from '../../data/pos_attributes';
import {browser} from '../../util/browser';
import {easeCubicInOut, lerp} from '../../util/util';
import {mercatorYfromLat} from '../mercator_coordinate';
import {NORTH_POLE_Y, SOUTH_POLE_Y} from '../../render/subdivision';
import {SubdivisionGranularityExpression, SubdivisionGranularitySetting} from '../../render/subdivision_granularity_settings';
import type {Projection, ProjectionGPUContext, TileMeshUsage} from './projection';
import {PreparedShader, shaders} from '../../shaders/shaders';
import {MercatorProjection} from './mercator';
import {ProjectionErrorMeasurement} from './globe_projection_error_measurement';

const EXTENT_STENCIL_BORDER = EXTENT / 128;
import {createTileMeshWithBuffers, CreateTileMeshOptions} from '../../util/create_tile_mesh';

export const globeConstants = {
/**
Expand Down Expand Up @@ -170,8 +164,8 @@ export class GlobeProjection implements Projection {
this._errorCorrectionUsable = lerp(this._errorCorrectionPreviousValue, newCorrection, easeCubicInOut(mix));
}

private _getMeshKey(granularity: number, border: boolean, north: boolean, south: boolean): string {
return `${granularity.toString(36)}_${border ? 'b' : ''}${north ? 'n' : ''}${south ? 's' : ''}`;
private _getMeshKey(options: CreateTileMeshOptions): string {
return `${options.granularity.toString(36)}_${options.generateBorders ? 'b' : ''}${options.extendToNorthPole ? 'n' : ''}${options.extendToSouthPole ? 's' : ''}`;
}

public getMeshFromTileID(context: Context, canonical: CanonicalTileID, hasBorder: boolean, allowPoles: boolean, usage: TileMeshUsage): Mesh {
Expand All @@ -180,91 +174,23 @@ export class GlobeProjection implements Projection {
const granularity = granularityConfig.getGranularityForZoomLevel(canonical.z);
const north = (canonical.y === 0) && allowPoles;
const south = (canonical.y === (1 << canonical.z) - 1) && allowPoles;
return this._getMesh(context, granularity, hasBorder, north, south);
return this._getMesh(context, {
granularity,
generateBorders: hasBorder,
extendToNorthPole: north,
extendToSouthPole: south,
});
}

private _getMesh(context: Context, granularity: number, hasBorder: boolean, hasNorthEdge: boolean, hasSouthEdge: boolean): Mesh {
const key = this._getMeshKey(granularity, hasBorder, hasNorthEdge, hasSouthEdge);
private _getMesh(context: Context, options: CreateTileMeshOptions): Mesh {
const key = this._getMeshKey(options);

if (key in this._tileMeshCache) {
return this._tileMeshCache[key];
}

const mesh = this._createQuadMesh(context, granularity, hasBorder, hasNorthEdge, hasSouthEdge);
const mesh = createTileMeshWithBuffers(context, options);
this._tileMeshCache[key] = mesh;
return mesh;
}

/**
* Creates a quad mesh covering positions in range 0..EXTENT, for tile clipping.
* @param context - MapLibre's rendering context object.
* @param granularity - Mesh triangulation granularity: 1 for just a single quad, 3 for 3x3 quads.
* @returns
*/
private _createQuadMesh(context: Context, granularity: number, border: boolean, north: boolean, south: boolean): Mesh {
const vertexArray = new PosArray();
const indexArray = new TriangleIndexArray();

// We only want to generate the north/south border if the tile
// does NOT border the north/south edge of the mercator range.

const quadsPerAxisX = granularity + (border ? 2 : 0); // two extra quads for border
const quadsPerAxisY = granularity + ((north || border) ? 1 : 0) + (south || border ? 1 : 0);
const verticesPerAxisX = quadsPerAxisX + 1; // one more vertex than quads
const verticesPerAxisY = quadsPerAxisY + 1; // one more vertex than quads
const offsetX = border ? -1 : 0;
const offsetY = (border || north) ? -1 : 0;
const endX = granularity + (border ? 1 : 0);
const endY = granularity + ((border || south) ? 1 : 0);

if (verticesPerAxisX * verticesPerAxisY > (1 << 16)) {
throw new Error('Granularity is too large and meshes would not fit inside 16 bit vertex indices.');
}

const northY = NORTH_POLE_Y;
const southY = SOUTH_POLE_Y;

for (let y = offsetY; y <= endY; y++) {
for (let x = offsetX; x <= endX; x++) {
let vx = x / granularity * EXTENT;
if (x === -1) {
vx = -EXTENT_STENCIL_BORDER;
}
if (x === granularity + 1) {
vx = EXTENT + EXTENT_STENCIL_BORDER;
}
let vy = y / granularity * EXTENT;
if (y === -1) {
vy = north ? northY : (-EXTENT_STENCIL_BORDER);
}
if (y === granularity + 1) {
vy = south ? southY : EXTENT + EXTENT_STENCIL_BORDER;
}
vertexArray.emplaceBack(vx, vy);
}
}

for (let y = 0; y < quadsPerAxisY; y++) {
for (let x = 0; x < quadsPerAxisX; x++) {
const v0 = x + y * verticesPerAxisX;
const v1 = (x + 1) + y * verticesPerAxisX;
const v2 = x + (y + 1) * verticesPerAxisX;
const v3 = (x + 1) + (y + 1) * verticesPerAxisX;
// v0----v1
// | / |
// | / |
// v2----v3
indexArray.emplaceBack(v0, v2, v1);
indexArray.emplaceBack(v1, v2, v3);
}
}

const mesh = new Mesh(
context.createVertexBuffer(vertexArray, posAttributes.members),
context.createIndexBuffer(indexArray),
SegmentVector.simpleSegment(0, 0, vertexArray.length, indexArray.length)
);

return mesh;
}
}
Loading