diff --git a/examples/config.json b/examples/config.json
index 318b70e726..6b7648a23a 100644
--- a/examples/config.json
+++ b/examples/config.json
@@ -31,6 +31,7 @@
"Vector tiles": {
"vector_tile_raster_3d": "Raster on 3D map",
"vector_tile_raster_2d": "Raster on 2D map",
+ "vector_tile_mapbox_raster": "Mapbox Vector Tiles to raster",
"vector_tile_3d_mesh": "Vector tile to 3d objects",
"vector_tile_3d_mesh_mapbox": "Mapbox Vector tile to 3d objects",
"vector_tile_dragndrop": "Drag and drop a style"
diff --git a/examples/source_file_kml_raster_usgs.html b/examples/source_file_kml_raster_usgs.html
index db83222b2b..b755f0890e 100644
--- a/examples/source_file_kml_raster_usgs.html
+++ b/examples/source_file_kml_raster_usgs.html
@@ -69,7 +69,6 @@
}
var kmlStyle = {
- zoom: { min: 10, max: 20 },
text: {
field: '{name}',
haloColor: 'black',
diff --git a/examples/vector_tile_mapbox_raster.html b/examples/vector_tile_mapbox_raster.html
new file mode 100644
index 0000000000..01b704bd0f
--- /dev/null
+++ b/examples/vector_tile_mapbox_raster.html
@@ -0,0 +1,91 @@
+
+
+ Itowns - vector-tiles 2d
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/Converter/Feature2Texture.js b/src/Converter/Feature2Texture.js
index b197e7fe2b..82c13a50eb 100644
--- a/src/Converter/Feature2Texture.js
+++ b/src/Converter/Feature2Texture.js
@@ -70,6 +70,9 @@ function drawFeature(ctx, feature, extent, invCtxScale) {
for (const geometry of feature.geometries) {
if (Extent.intersectsExtent(geometry.extent, extent)) {
context.setGeometry(geometry);
+ if (style.zoom.min > style.context.zoom || style.zoom.max <= style.context.zoom) {
+ return;
+ }
if (
feature.type === FEATURE_TYPES.POINT && style.point
diff --git a/src/Core/Style.js b/src/Core/Style.js
index 3e716bd26a..f0df34dc9c 100644
--- a/src/Core/Style.js
+++ b/src/Core/Style.js
@@ -87,14 +87,18 @@ async function loadImage(source) {
return (await promise).image;
}
-function cropImage(img, cropValues = { width: img.naturalWidth, height: img.naturalHeight }) {
- canvas.width = cropValues.width;
- canvas.height = cropValues.height;
+function cropImage(img, cropValues) {
+ const x = cropValues.x || 0;
+ const y = cropValues.y || 0;
+ const width = cropValues.width || img.naturalWidth;
+ const height = cropValues.height || img.naturalHeight;
+ canvas.width = width;
+ canvas.height = height;
const ctx = canvas.getContext('2d', { willReadFrequently: true });
ctx.drawImage(img,
- cropValues.x || 0, cropValues.y || 0, cropValues.width, cropValues.height,
- 0, 0, cropValues.width, cropValues.height);
- return ctx.getImageData(0, 0, cropValues.width, cropValues.height);
+ x, y, width, height,
+ 0, 0, width, height);
+ return ctx.getImageData(0, 0, width, height);
}
function replaceWhitePxl(imgd, color, id) {
@@ -158,7 +162,7 @@ function defineStyleProperty(style, category, parameter, userValue, defaultValue
const dataValue = style.context.featureStyle?.[category]?.[parameter];
if (dataValue != undefined) { return readExpression(dataValue, style.context); }
if (defaultValue instanceof Function) {
- return defaultValue(style.context.properties, style.context);
+ return defaultValue(style.context.properties, style.context) ?? defaultValue;
}
return defaultValue;
},
@@ -298,15 +302,12 @@ function _addIcon(icon, domElement, opt) {
}
/**
- * An object that can contain any properties (order, zoom, fill, stroke, point,
+ * An object that can contain any properties (zoom, fill, stroke, point,
* text or/and icon) and sub properties of a Style.
* Used for the instanciation of a {@link Style}.
*
* @typedef {Object} StyleOptions
*
- * @property {Number} [order] - Order of the features that will be associated to
- * the style. It can helps sorting and prioritizing features if needed.
- *
* @property {Object} [zoom] - Level on which to display the feature
* @property {Number} [zoom.max] - max level
* @property {Number} [zoom.min] - min level
@@ -464,8 +465,6 @@ function _addIcon(icon, domElement, opt) {
* The first parameter of functions used to set `Style` properties is always an object containing
* the properties of the features displayed with the current `Style` instance.
*
- * @property {Number} order - Order of the features that will be associated to
- * the style. It can helps sorting and prioritizing features if needed.
* @property {Object} fill - Polygons and fillings style.
* @property {String|Function|THREE.Color} fill.color - Defines the main color of the filling. Can be
* any [valid color
@@ -615,15 +614,13 @@ function _addIcon(icon, domElement, opt) {
class Style {
/**
* @param {StyleOptions} [params={}] An object that contain any properties
- * (order, zoom, fill, stroke, point, text or/and icon)
+ * (zoom, fill, stroke, point, text or/and icon)
* and sub properties of a Style ({@link StyleOptions}).
*/
constructor(params = {}) {
this.isStyle = true;
this.context = new StyleContext();
- this.order = params.order || 0;
-
params.zoom = params.zoom || {};
params.fill = params.fill || {};
params.stroke = params.stroke || {};
@@ -763,12 +760,11 @@ class Style {
* set Style from vector tile layer properties.
* @param {Object} layer vector tile layer.
* @param {Object} sprites vector tile layer.
- * @param {Number} [order=0]
* @param {Boolean} [symbolToCircle=false]
*
* @returns {StyleOptions} containing all properties for itowns.Style
*/
- static setFromVectorTileLayer(layer, sprites, order = 0, symbolToCircle = false) {
+ static setFromVectorTileLayer(layer, sprites, symbolToCircle = false) {
const style = {
fill: {},
stroke: {},
@@ -780,8 +776,6 @@ class Style {
layer.layout = layer.layout || {};
layer.paint = layer.paint || {};
- style.order = order;
-
if (layer.type === 'fill') {
const { color, opacity } = rgba2rgb(readVectorProperty(layer.paint['fill-color'] || layer.paint['fill-pattern'], { type: 'color' }));
style.fill.color = color;
@@ -804,7 +798,8 @@ class Style {
style.stroke.color = color;
style.stroke.opacity = opacity;
style.stroke.width = 1.0;
- style.stroke.dasharray = [];
+ } else {
+ style.stroke.width = 0.0;
}
} else if (layer.type === 'line') {
const prepare = readVectorProperty(layer.paint['line-color'], { type: 'color' });
@@ -820,6 +815,8 @@ class Style {
style.point.opacity = opacity;
style.point.radius = readVectorProperty(layer.paint['circle-radius']);
} else if (layer.type === 'symbol') {
+ // if symbol we shouldn't draw stroke but defaut value is 1.
+ style.stroke.width = 0.0;
// overlapping order
style.text.zOrder = readVectorProperty(layer.layout['symbol-z-order']);
if (style.text.zOrder == 'auto') {
@@ -840,7 +837,7 @@ class Style {
// content
style.text.field = readVectorProperty(layer.layout['text-field']);
- style.text.wrap = readVectorProperty(layer.layout['text-max-width']);
+ style.text.wrap = readVectorProperty(layer.layout['text-max-width']);// Units ems
style.text.spacing = readVectorProperty(layer.layout['text-letter-spacing']);
style.text.transform = readVectorProperty(layer.layout['text-transform']);
style.text.justify = readVectorProperty(layer.layout['text-justify']);
@@ -861,6 +858,12 @@ class Style {
// additional icon
const iconImg = readVectorProperty(layer.layout['icon-image']);
if (iconImg) {
+ const cropValueDefault = {
+ x: 0,
+ y: 0,
+ width: 1,
+ height: 1,
+ };
try {
style.icon.id = iconImg;
if (iconImg.stops) {
@@ -871,9 +874,15 @@ class Style {
if (stop[1].includes('{')) {
cropValues = function _(p) {
const id = stop[1].replace(/\{(.+?)\}/g, (a, b) => (p[b] || '')).trim();
- cropValues = sprites[id];
+ if (cropValues === undefined) {
+ // const warning = `WARNING: "${id}" not found in sprite file`;
+ sprites[id] = cropValueDefault;// or return cropValueDefault;
+ }
return sprites[id];
};
+ } else if (cropValues === undefined) {
+ // const warning = `WARNING: "${stop[1]}" not found in sprite file`;
+ cropValues = cropValueDefault;
}
return [stop[0], cropValues];
}),
@@ -881,19 +890,28 @@ class Style {
style.icon.cropValues = iconCropValue;
} else {
style.icon.cropValues = sprites[iconImg];
- if (iconImg[0].includes('{')) {
+ if (iconImg.includes('{')) {
style.icon.cropValues = function _(p) {
const id = iconImg.replace(/\{(.+?)\}/g, (a, b) => (p[b] || '')).trim();
- style.icon.cropValues = sprites[id];
+ if (sprites[id] === undefined) {
+ // const warning = `WARNING: "${id}" not found in sprite file`;
+ sprites[id] = cropValueDefault;// or return cropValueDefault;
+ }
return sprites[id];
};
+ } else if (sprites[iconImg] === undefined) {
+ // const warning = `WARNING: "${iconImg}" not found in sprite file`;
+ style.icon.cropValues = cropValueDefault;
}
}
style.icon.source = sprites.source;
- style.icon.size = readVectorProperty(layer.layout['icon-size']) || 1;
+ style.icon.size = readVectorProperty(layer.layout['icon-size']) ?? 1;
const { color, opacity } = rgba2rgb(readVectorProperty(layer.paint['icon-color'], { type: 'color' }));
- style.icon.color = color;
- style.icon.opacity = readVectorProperty(layer.paint['icon-opacity']) || (opacity !== undefined && opacity);
+ // https://docs.mapbox.com/style-spec/reference/layers/#paint-symbol-icon-color
+ if (iconImg.sdf) {
+ style.icon.color = color;
+ }
+ style.icon.opacity = readVectorProperty(layer.paint['icon-opacity']) ?? (opacity !== undefined && opacity);
} catch (err) {
err.message = `VTlayer '${layer.id}': argument sprites must not be null when using layer.layout['icon-image']`;
throw err;
@@ -919,17 +937,16 @@ class Style {
* @param {Boolean} canBeFilled - true if feature.type == FEATURE_TYPES.POLYGON.
*/
applyToCanvasPolygon(txtrCtx, polygon, invCtxScale, canBeFilled) {
- const context = this.context;
// draw line or edge of polygon
- if (this.stroke) {
+ if (this.stroke.width > 0) {
// TO DO add possibility of using a pattern (https://github.com/iTowns/itowns/issues/2210)
- this._applyStrokeToPolygon(txtrCtx, invCtxScale, polygon, context);
+ this._applyStrokeToPolygon(txtrCtx, invCtxScale, polygon);
}
// fill inside of polygon
- if (canBeFilled && this.fill) {
+ if (canBeFilled && (this.fill.pattern || this.fill.color)) {
// canBeFilled can be move to StyleContext in the later PR
- this._applyFillToPolygon(txtrCtx, invCtxScale, polygon, context);
+ this._applyFillToPolygon(txtrCtx, invCtxScale, polygon);
}
}
@@ -937,11 +954,11 @@ class Style {
if (txtrCtx.strokeStyle !== this.stroke.color) {
txtrCtx.strokeStyle = this.stroke.color;
}
- const width = (this.stroke.width || 2.0) * invCtxScale;
+ const width = this.stroke.width * invCtxScale;
if (txtrCtx.lineWidth !== width) {
txtrCtx.lineWidth = width;
}
- const alpha = this.stroke.opacity == undefined ? 1.0 : this.stroke.opacity;
+ const alpha = this.stroke.opacity;
if (alpha !== txtrCtx.globalAlpha && typeof alpha == 'number') {
txtrCtx.globalAlpha = alpha;
}
@@ -957,7 +974,7 @@ class Style {
// need doc for the txtrCtx.fillStyle.src that seems to always be undefined
if (this.fill.pattern) {
let img = this.fill.pattern;
- const cropValues = this.fill.pattern.cropValues;
+ const cropValues = { ...this.fill.pattern.cropValues };
if (this.fill.pattern.source) {
img = await loadImage(this.fill.pattern.source);
}
@@ -1032,7 +1049,7 @@ class Style {
if (!this.icon.cropValues && !this.icon.color) {
icon.src = this.icon.source;
} else {
- const cropValues = this.icon.cropValues;
+ const cropValues = { ...this.icon.cropValues };
const color = this.icon.color;
const id = this.icon.id || this.icon.source;
const img = await loadImage(this.icon.source);
diff --git a/src/Layer/LabelLayer.js b/src/Layer/LabelLayer.js
index 1dcd649b00..ff2ffccc42 100644
--- a/src/Layer/LabelLayer.js
+++ b/src/Layer/LabelLayer.js
@@ -5,7 +5,6 @@ import GeometryLayer from 'Layer/GeometryLayer';
import Coordinates from 'Core/Geographic/Coordinates';
import Extent from 'Core/Geographic/Extent';
import Label from 'Core/Label';
-import { FEATURE_TYPES } from 'Core/Feature';
import { readExpression, StyleContext } from 'Core/Style';
import { ScreenGrid } from 'Renderer/Label2DRenderer';
@@ -254,10 +253,12 @@ class LabelLayer extends GeometryLayer {
context.setZoom(extentOrTile.zoom);
data.features.forEach((f) => {
- // TODO: add support for LINE and POLYGON
- if (f.type !== FEATURE_TYPES.POINT) {
- return;
+ if (f.style.text) {
+ if (Object.keys(f.style.text).length === 0) {
+ return;
+ }
}
+
context.setFeature(f);
const featureField = f.style?.text?.field;
@@ -270,19 +271,11 @@ class LabelLayer extends GeometryLayer {
labels.needsAltitude = labels.needsAltitude || this.forceClampToTerrain === true || (isDefaultElevationStyle && !f.hasRawElevationData);
f.geometries.forEach((g) => {
- // NOTE: this only works because only POINT is supported, it
- // needs more work for LINE and POLYGON
- coord.setFromArray(f.vertices, g.size * g.indices[0].offset);
- // Transform coordinate to data.crs projection
- coord.applyMatrix4(data.matrixWorld);
-
- if (!_extent.isPointInside(coord)) { return; }
- const geometryField = g.properties.style && g.properties.style.text && g.properties.style.text.field;
-
context.setGeometry(g);
- let content;
this.style.setContext(context);
const layerField = this.style.text && this.style.text.field;
+ const geometryField = g.properties.style && g.properties.style.text && g.properties.style.text.field;
+ let content;
if (this.labelDomelement) {
content = readExpression(this.labelDomelement, context);
} else if (!geometryField && !featureField && !layerField) {
@@ -294,12 +287,28 @@ class LabelLayer extends GeometryLayer {
}
}
- const label = new Label(content, coord.clone(), this.style);
+ if (this.style.zoom.min > this.style.context.zoom || this.style.zoom.max <= this.style.context.zoom) {
+ return;
+ }
+
+ // NOTE: this only works fine for POINT.
+ // It needs more work for LINE and POLYGON as we currently only use the first point of the entity
- label.layerId = this.id;
- label.padding = this.margin || label.padding;
+ g.indices.forEach((i) => {
+ coord.setFromArray(f.vertices, g.size * i.offset);
+ // Transform coordinate to data.crs projection
+ coord.applyMatrix4(data.matrixWorld);
- labels.push(label);
+ if (!_extent.isPointInside(coord)) { return; }
+
+ const label = new Label(content, coord.clone(), this.style);
+
+ label.layerId = this.id;
+ label.order = f.order;
+ label.padding = this.margin || label.padding;
+
+ labels.push(label);
+ });
});
});
diff --git a/src/Parser/VectorTileParser.js b/src/Parser/VectorTileParser.js
index bff52dc251..021b41035d 100644
--- a/src/Parser/VectorTileParser.js
+++ b/src/Parser/VectorTileParser.js
@@ -116,10 +116,11 @@ function vtFeatureToFeatureGeometry(vtFeature, feature, classify = false) {
function readPBF(file, options) {
options.out = options.out || {};
const vectorTile = new VectorTile(new Protobuf(file));
- const sourceLayers = Object.keys(vectorTile.layers);
+ const vtLayerNames = Object.keys(vectorTile.layers);
- if (sourceLayers.length < 1) {
- return;
+ const collection = new FeatureCollection(options.out);
+ if (vtLayerNames.length < 1) {
+ return Promise.resolve(collection);
}
// x,y,z tile coordinates
@@ -130,28 +131,40 @@ function readPBF(file, options) {
// Only if the layer.origin is top
const y = options.in.isInverted ? options.extent.row : (1 << z) - options.extent.row - 1;
- const collection = new FeatureCollection(options.out);
-
- const vFeature = vectorTile.layers[sourceLayers[0]];
- // TODO: verify if size is correct because is computed with only one feature (vFeature).
- const size = vFeature.extent * 2 ** z;
+ const vFeature0 = vectorTile.layers[vtLayerNames[0]];
+ // TODO: verify if size is correct because is computed with only one feature (vFeature0).
+ const size = vFeature0.extent * 2 ** z;
const center = -0.5 * size;
collection.scale.set(globalExtent.x / size, -globalExtent.y / size, 1);
- collection.position.set(vFeature.extent * x + center, vFeature.extent * y + center, 0).multiply(collection.scale);
+ collection.position.set(vFeature0.extent * x + center, vFeature0.extent * y + center, 0).multiply(collection.scale);
collection.updateMatrixWorld();
- sourceLayers.forEach((layer_id) => {
- if (!options.in.layers[layer_id]) { return; }
+ vtLayerNames.forEach((vtLayerName) => {
+ if (!options.in.layers[vtLayerName]) { return Promise.resolve(collection); }
- const sourceLayer = vectorTile.layers[layer_id];
+ const vectorTileLayer = vectorTile.layers[vtLayerName];
- for (let i = sourceLayer.length - 1; i >= 0; i--) {
- const vtFeature = sourceLayer.feature(i);
+ for (let i = vectorTileLayer.length - 1; i >= 0; i--) {
+ const vtFeature = vectorTileLayer.feature(i);
vtFeature.tileNumbers = { x, y: options.extent.row, z };
- const layers = options.in.layers[layer_id].filter(l => l.filterExpression.filter({ zoom: z }, vtFeature) && z >= l.zoom.min && z < l.zoom.max);
- let feature;
+ // Find layers where this vtFeature is used
+ const layers = options.in.layers[vtLayerName]
+ .filter(l => l.filterExpression.filter({ zoom: z }, vtFeature));
+ for (const layer of layers) {
+ const feature = collection.requestFeatureById(layer.id, vtFeature.type - 1);
+ feature.id = layer.id;
+ feature.order = layer.order;
+ feature.style = options.in.styles[feature.id];
+ vtFeatureToFeatureGeometry(vtFeature, feature);
+ }
+
+
+ /*
+ // This optimization is not fully working and need to be reassessed
+ // (see https://github.com/iTowns/itowns/pull/2469/files#r1861802136)
+ let feature;
for (const layer of layers) {
if (!feature) {
feature = collection.requestFeatureById(layer.id, vtFeature.type - 1);
@@ -166,6 +179,7 @@ function readPBF(file, options) {
feature.style = options.in.styles[feature.id];
}
}
+ */
}
});
diff --git a/src/Renderer/Label2DRenderer.js b/src/Renderer/Label2DRenderer.js
index f9ac17ab60..6e927844df 100644
--- a/src/Renderer/Label2DRenderer.js
+++ b/src/Renderer/Label2DRenderer.js
@@ -186,13 +186,15 @@ class Label2DRenderer {
if (!frustum.containsPoint(worldPosition.applyMatrix4(camera.matrixWorldInverse)) ||
// Check if globe horizon culls the label
// Do some horizon culling (if possible) if the tiles level is small enough.
- label.horizonCullingPoint && GlobeLayer.horizonCulling(label.horizonCullingPoint) ||
- // Check if content isn't present in visible labels
- this.grid.visible.some((l) => {
- // TODO for icon without text filter by position
- const textContent = label.content.textContent;
- return textContent !== '' && l.content.textContent.toLowerCase() == textContent.toLowerCase();
- })) {
+ label.horizonCullingPoint && GlobeLayer.horizonCulling(label.horizonCullingPoint)
+ // Why do we might need this part ?
+ // || // Check if content isn't present in visible labels
+ // this.grid.visible.some((l) => {
+ // // TODO for icon without text filter by position
+ // const textContent = label.content.textContent;
+ // return textContent !== '' && l.content.textContent.toLowerCase() == textContent.toLowerCase();
+ // })
+ ) {
label.visible = false;
} else {
// projecting world position label
diff --git a/src/Source/VectorTilesSource.js b/src/Source/VectorTilesSource.js
index ed5b9562ed..4d70ae39e9 100644
--- a/src/Source/VectorTilesSource.js
+++ b/src/Source/VectorTilesSource.js
@@ -20,6 +20,16 @@ function mergeCollections(collections) {
return collection;
}
+// A deprecated (but still in use) Mapbox spec allows using 'ref' as a propertie to reference an other layer
+// instead of duplicating the following properties: 'type', 'source', 'source-layer', 'minzoom', 'maxzoom', 'filter', 'layout'
+function getPropertiesFromRefLayer(layers, layer) {
+ const refProperties = ['type', 'source', 'source-layer', 'minzoom', 'maxzoom', 'filter', 'layout'];
+ const refLayer = layers.filter(l => l.id === layer.ref)[0];
+ refProperties.forEach((prop) => {
+ layer[prop] = refLayer[prop];
+ });
+}
+
/**
* VectorTilesSource are object containing informations on how to fetch vector
* tiles resources.
@@ -71,10 +81,11 @@ class VectorTilesSource extends TMSSource {
this.accessToken = source.accessToken;
+ let mvtStyleUrl;
if (source.style) {
if (typeof source.style == 'string') {
- const styleUrl = urlParser.normalizeStyleURL(source.style, this.accessToken);
- promise = Fetcher.json(styleUrl, this.networkOptions);
+ mvtStyleUrl = urlParser.normalizeStyleURL(source.style, this.accessToken);
+ promise = Fetcher.json(mvtStyleUrl, this.networkOptions);
} else {
promise = Promise.resolve(source.style);
}
@@ -82,27 +93,31 @@ class VectorTilesSource extends TMSSource {
throw new Error('New VectorTilesSource: style is required');
}
- this.whenReady = promise.then((style) => {
- this.jsonStyle = style;
- const baseurl = source.sprite || style.sprite;
+ this.whenReady = promise.then((mvtStyle) => {
+ this.jsonStyle = mvtStyle;
+ let baseurl = source.sprite || mvtStyle.sprite;
if (baseurl) {
+ baseurl = new URL(baseurl, mvtStyleUrl).toString();
const spriteUrl = urlParser.normalizeSpriteURL(baseurl, '', '.json', this.accessToken);
return Fetcher.json(spriteUrl, this.networkOptions).then((sprites) => {
this.sprites = sprites;
const imgUrl = urlParser.normalizeSpriteURL(baseurl, '', '.png', this.accessToken);
this.sprites.source = imgUrl;
- return style;
+ return mvtStyle;
});
}
- return style;
- }).then((style) => {
- style.layers.forEach((layer, order) => {
+ return mvtStyle;
+ }).then((mvtStyle) => {
+ mvtStyle.layers.forEach((layer, order) => {
layer.sourceUid = this.uid;
if (layer.type === 'background') {
this.backgroundLayer = layer;
} else if (ffilter(layer)) {
- const style = Style.setFromVectorTileLayer(layer, this.sprites, order, this.symbolToCircle);
+ if (layer['source-layer'] === undefined) {
+ getPropertiesFromRefLayer(mvtStyle.layers, layer);
+ }
+ const style = Style.setFromVectorTileLayer(layer, this.sprites, this.symbolToCircle);
this.styles[layer.id] = style;
if (!this.layers[layer['source-layer']]) {
@@ -112,20 +127,18 @@ class VectorTilesSource extends TMSSource {
id: layer.id,
order,
filterExpression: featureFilter(layer.filter),
- zoom: {
- min: layer.minzoom || 0,
- max: layer.maxzoom || 24,
- },
});
}
});
if (this.url == '.') {
- const TMSUrlList = Object.values(style.sources).map((sourceVT) => {
+ const TMSUrlList = Object.values(mvtStyle.sources).map((sourceVT) => {
if (sourceVT.url) {
+ sourceVT.url = new URL(sourceVT.url, mvtStyleUrl).toString();
const urlSource = urlParser.normalizeSourceURL(sourceVT.url, this.accessToken);
return Fetcher.json(urlSource, this.networkOptions).then((tileJSON) => {
if (tileJSON.tiles[0]) {
+ tileJSON.tiles[0] = decodeURIComponent(new URL(tileJSON.tiles[0], urlSource).toString());
return toTMSUrl(tileJSON.tiles[0]);
}
});
@@ -136,7 +149,7 @@ class VectorTilesSource extends TMSSource {
});
return Promise.all(TMSUrlList);
}
- return (Promise.resolve([this.url]));
+ return (Promise.resolve([toTMSUrl(this.url)]));
}).then((TMSUrlList) => {
this.urls = Array.from(new Set(TMSUrlList));
});
diff --git a/test/data/vectortiles/style.json b/test/data/vectortiles/style.json
index 1af58000a3..ffc6a8c707 100644
--- a/test/data/vectortiles/style.json
+++ b/test/data/vectortiles/style.json
@@ -19,7 +19,8 @@
"maxzoom": 13,
"paint": {
"fill-color": "rgb(255, 0, 0)"
- }
+ },
+ "source-layer": "source_layer"
}
]
}
\ No newline at end of file
diff --git a/test/unit/vectortiles.js b/test/unit/vectortiles.js
index fa93cfdca4..610c752053 100644
--- a/test/unit/vectortiles.js
+++ b/test/unit/vectortiles.js
@@ -13,7 +13,7 @@ import sprite from '../data/vectortiles/sprite.json';
import mapboxStyle from '../data/mapboxMulti.json';
const resources = {
- 'test/data/vectortiles/style.json': style,
+ 'https://test/data/vectortiles/style.json': style,
'https://test/tilejson.json': tilejson,
'https://test/sprite.json': sprite,
'https://api.mapbox.com/v4/mapbox.mapbox-terrain-v2,mapbox.mapbox-streets-v7.json': mapboxStyle,
@@ -67,15 +67,17 @@ describe('Vector tiles', function () {
}).catch(done);
});
- it('returns nothing', (done) => {
+ it('returns an empty collection', (done) => {
parse(null).then((collection) => {
- assert.equal(collection, undefined);
+ assert.ok(collection.isFeatureCollection);
+ assert.equal(collection.features.length, 0);
done();
}).catch(done);
});
it('filters all features out', (done) => {
parse(multipolygon, {}).then((collection) => {
+ assert.ok(collection.isFeatureCollection);
assert.equal(collection.features.length, 0);
done();
}).catch(done);
@@ -174,6 +176,7 @@ describe('VectorTilesSource', function () {
paint: {
'fill-color': 'rgb(255, 0, 0)',
},
+ 'source-layer': 'source_layer',
}],
},
});
@@ -187,7 +190,7 @@ describe('VectorTilesSource', function () {
it('loads the style from a file', function _it(done) {
const source = new VectorTilesSource({
- style: 'test/data/vectortiles/style.json',
+ style: 'https://test/data/vectortiles/style.json',
});
source.whenReady
.then(() => {
@@ -199,6 +202,40 @@ describe('VectorTilesSource', function () {
}).catch(done);
});
+ it('loads the style from a file with ref layer', function _it(done) {
+ const source = new VectorTilesSource({
+ url: 'fakeurl',
+ style: {
+ sources: { tilejson: {} },
+ layers: [
+ {
+ id: 'land',
+ type: 'fill',
+ paint: {
+ 'fill-color': 'rgb(255, 0, 0)',
+ },
+ 'source-layer': 'source_layer',
+ },
+ {
+ id: 'land-secondary',
+ paint: {
+ 'fill-color': 'rgb(0, 255, 0)',
+ },
+ ref: 'land',
+ },
+ ],
+ },
+ });
+ source.whenReady.then(() => {
+ assert.ok(source.styles.land);
+ assert.equal(source.styles.land.fill.color, 'rgb(255,0,0)');
+ assert.ok(source.styles['land-secondary']);
+ assert.equal(source.styles['land-secondary'].fill.color, 'rgb(0,255,0)');
+ done();
+ })
+ .catch(done);
+ });
+
it('sets the correct Style#zoom.min', (done) => {
const source = new VectorTilesSource({
url: 'fakeurl',
@@ -211,6 +248,7 @@ describe('VectorTilesSource', function () {
paint: {
'fill-color': 'rgb(255, 0, 0)',
},
+ 'source-layer': 'source_layer',
}, {
// minzoom is 5 (specified)
id: 'second',
@@ -219,6 +257,7 @@ describe('VectorTilesSource', function () {
'fill-color': 'rgb(255, 0, 0)',
},
minzoom: 5,
+ 'source-layer': 'source_layer',
}, {
// minzoom is 4 (first stop)
// If a style have `stops` expression, should it be used to determine the min zoom?
@@ -228,6 +267,7 @@ describe('VectorTilesSource', function () {
'fill-color': 'rgb(255, 0, 0)',
'fill-opacity': { stops: [[4, 1], [7, 0.5]] },
},
+ 'source-layer': 'source_layer',
}, {
// minzoom is 1 (first stop and no specified minzoom)
id: 'fourth',
@@ -236,6 +276,7 @@ describe('VectorTilesSource', function () {
'fill-color': 'rgb(255, 0, 0)',
'fill-opacity': { stops: [[1, 1], [7, 0.5]] },
},
+ 'source-layer': 'source_layer',
}, {
// minzoom is 4 (first stop is higher than specified)
id: 'fifth',
@@ -245,6 +286,7 @@ describe('VectorTilesSource', function () {
'fill-opacity': { stops: [[4, 1], [7, 0.5]] },
},
minzoom: 3,
+ 'source-layer': 'source_layer',
}],
},
});