diff --git a/examples/reprojection-by-code.html b/examples/reprojection-by-code.html
new file mode 100644
index 0000000000..ab784c4607
--- /dev/null
+++ b/examples/reprojection-by-code.html
@@ -0,0 +1,26 @@
+---
+template: example.html
+title: Reprojection with EPSG.io database search
+shortdesc: Demonstrates client-side raster reprojection of MapQuest OSM to arbitrary projection
+docs: >
+ This example shows client-side raster reprojection capabilities from
+ MapQuest OSM (EPSG:3857) to arbitrary projection by searching
+ in EPSG.io database.
+tags: "reprojection, projection, proj4js, mapquest, epsg.io"
+resources:
+ - http://cdnjs.cloudflare.com/ajax/libs/proj4js/2.3.6/proj4.js
+---
+
diff --git a/examples/reprojection-by-code.js b/examples/reprojection-by-code.js
new file mode 100644
index 0000000000..96203a4177
--- /dev/null
+++ b/examples/reprojection-by-code.js
@@ -0,0 +1,116 @@
+goog.require('ol.Map');
+goog.require('ol.View');
+goog.require('ol.extent');
+goog.require('ol.layer.Tile');
+goog.require('ol.proj');
+goog.require('ol.source.MapQuest');
+goog.require('ol.source.TileImage');
+
+
+
+var map = new ol.Map({
+ layers: [
+ new ol.layer.Tile({
+ source: new ol.source.MapQuest({layer: 'osm'})
+ })
+ ],
+ renderer: common.getRendererFromQueryString(),
+ target: 'map',
+ view: new ol.View({
+ projection: 'EPSG:3857',
+ center: [0, 0],
+ zoom: 1
+ })
+});
+
+
+var queryInput = document.getElementById('epsg-query');
+var searchButton = document.getElementById('epsg-search');
+var resultSpan = document.getElementById('epsg-result');
+var renderEdgesCheckbox = document.getElementById('render-edges');
+
+function setProjection(code, name, proj4def, bbox) {
+ if (code === null || name === null || proj4def === null || bbox === null) {
+ resultSpan.innerHTML = 'Nothing usable found, using EPSG:3857...';
+ map.setView(new ol.View({
+ projection: 'EPSG:3857',
+ center: [0, 0],
+ zoom: 1
+ }));
+ return;
+ }
+
+ resultSpan.innerHTML = '(' + code + ') ' + name;
+
+ var newProjCode = 'EPSG:' + code;
+ proj4.defs(newProjCode, proj4def);
+ var newProj = ol.proj.get(newProjCode);
+ var fromLonLat = ol.proj.getTransform('EPSG:4326', newProj);
+
+ // very approximate calculation of projection extent
+ var extent = ol.extent.applyTransform(
+ [bbox[1], bbox[2], bbox[3], bbox[0]], fromLonLat);
+ newProj.setExtent(extent);
+ var newView = new ol.View({
+ projection: newProj
+ });
+ map.setView(newView);
+
+ var size = map.getSize();
+ if (size) {
+ newView.fit(extent, size);
+ }
+}
+
+
+function search(query) {
+ resultSpan.innerHTML = 'Searching...';
+ $.ajax({
+ url: 'http://epsg.io/?format=json&q=' + query,
+ dataType: 'jsonp',
+ success: function(response) {
+ if (response) {
+ var results = response['results'];
+ if (results && results.length > 0) {
+ for (var i = 0, ii = results.length; i < ii; i++) {
+ var result = results[i];
+ if (result) {
+ var code = result['code'], name = result['name'],
+ proj4def = result['proj4'], bbox = result['bbox'];
+ if (code && code.length > 0 && proj4def && proj4def.length > 0 &&
+ bbox && bbox.length == 4) {
+ setProjection(code, name, proj4def, bbox);
+ return;
+ }
+ }
+ }
+ }
+ }
+ setProjection(null, null, null, null);
+ }
+ });
+}
+
+
+/**
+ * @param {Event} e Change event.
+ */
+searchButton.onclick = function(e) {
+ search(queryInput.value);
+ e.preventDefault();
+};
+
+
+/**
+ * @param {Event} e Change event.
+ */
+renderEdgesCheckbox.onchange = function(e) {
+ map.getLayers().forEach(function(layer) {
+ if (layer instanceof ol.layer.Tile) {
+ var source = layer.getSource();
+ if (source instanceof ol.source.TileImage) {
+ source.setRenderReprojectionEdges(renderEdgesCheckbox.checked);
+ }
+ }
+ });
+};
diff --git a/examples/reprojection-image.html b/examples/reprojection-image.html
new file mode 100644
index 0000000000..4b505605ad
--- /dev/null
+++ b/examples/reprojection-image.html
@@ -0,0 +1,15 @@
+---
+template: example.html
+title: Image reprojection example
+shortdesc: Demonstrates client-side reprojection of single image source.
+docs: >
+ This example shows client-side reprojection of single image source.
+tags: "reprojection, projection, proj4js, mapquest, image, imagestatic"
+resources:
+ - http://cdnjs.cloudflare.com/ajax/libs/proj4js/2.3.6/proj4.js
+---
+
diff --git a/examples/reprojection-image.js b/examples/reprojection-image.js
new file mode 100644
index 0000000000..3688097980
--- /dev/null
+++ b/examples/reprojection-image.js
@@ -0,0 +1,39 @@
+goog.require('ol.Map');
+goog.require('ol.View');
+goog.require('ol.extent');
+goog.require('ol.layer.Image');
+goog.require('ol.layer.Tile');
+goog.require('ol.proj');
+goog.require('ol.source.ImageStatic');
+goog.require('ol.source.MapQuest');
+
+
+proj4.defs('EPSG:27700', '+proj=tmerc +lat_0=49 +lon_0=-2 +k=0.9996012717 ' +
+ '+x_0=400000 +y_0=-100000 +ellps=airy ' +
+ '+towgs84=446.448,-125.157,542.06,0.15,0.247,0.842,-20.489 ' +
+ '+units=m +no_defs');
+var imageExtent = [0, 0, 700000, 1300000];
+
+var map = new ol.Map({
+ layers: [
+ new ol.layer.Tile({
+ source: new ol.source.MapQuest({layer: 'osm'})
+ }),
+ new ol.layer.Image({
+ source: new ol.source.ImageStatic({
+ url: 'http://upload.wikimedia.org/wikipedia/commons/thumb/1/18/' +
+ 'British_National_Grid.svg/2000px-British_National_Grid.svg.png',
+ crossOrigin: '',
+ projection: 'EPSG:27700',
+ imageExtent: imageExtent
+ })
+ })
+ ],
+ renderer: common.getRendererFromQueryString(),
+ target: 'map',
+ view: new ol.View({
+ center: ol.proj.transform(ol.extent.getCenter(imageExtent),
+ 'EPSG:27700', 'EPSG:3857'),
+ zoom: 4
+ })
+});
diff --git a/examples/reprojection-wgs84.html b/examples/reprojection-wgs84.html
new file mode 100644
index 0000000000..f6f1ec3754
--- /dev/null
+++ b/examples/reprojection-wgs84.html
@@ -0,0 +1,13 @@
+---
+template: example.html
+title: OpenStreetMap reprojection example
+shortdesc: Demonstrates client-side reprojection of OpenStreetMap in WGS84.
+docs: >
+ This example shows client-side reprojection of OpenStreetMap in WGS84.
+tags: "reprojection, projection, openstreetmap, wgs84, tile"
+---
+
diff --git a/examples/reprojection-wgs84.js b/examples/reprojection-wgs84.js
new file mode 100644
index 0000000000..6f85d7d219
--- /dev/null
+++ b/examples/reprojection-wgs84.js
@@ -0,0 +1,19 @@
+goog.require('ol.Map');
+goog.require('ol.View');
+goog.require('ol.layer.Tile');
+goog.require('ol.source.OSM');
+
+var map = new ol.Map({
+ layers: [
+ new ol.layer.Tile({
+ source: new ol.source.OSM()
+ })
+ ],
+ renderer: common.getRendererFromQueryString(),
+ target: 'map',
+ view: new ol.View({
+ projection: 'EPSG:4326',
+ center: [0, 0],
+ zoom: 1
+ })
+});
diff --git a/examples/reprojection.html b/examples/reprojection.html
new file mode 100644
index 0000000000..365c612945
--- /dev/null
+++ b/examples/reprojection.html
@@ -0,0 +1,48 @@
+---
+template: example.html
+title: Raster reprojection example
+shortdesc: Demonstrates client-side raster reprojection between various projections.
+docs: >
+ This example shows client-side raster reprojection between various projections.
+tags: "reprojection, projection, proj4js, mapquest, wms, wmts, hidpi"
+resources:
+ - http://cdnjs.cloudflare.com/ajax/libs/proj4js/2.3.6/proj4.js
+---
+
diff --git a/examples/reprojection.js b/examples/reprojection.js
new file mode 100644
index 0000000000..dec799b8eb
--- /dev/null
+++ b/examples/reprojection.js
@@ -0,0 +1,244 @@
+goog.require('ol.Attribution');
+goog.require('ol.Map');
+goog.require('ol.View');
+goog.require('ol.extent');
+goog.require('ol.format.WMTSCapabilities');
+goog.require('ol.layer.Tile');
+goog.require('ol.proj');
+goog.require('ol.source.MapQuest');
+goog.require('ol.source.TileImage');
+goog.require('ol.source.TileWMS');
+goog.require('ol.source.WMTS');
+goog.require('ol.source.XYZ');
+goog.require('ol.tilegrid.TileGrid');
+
+
+proj4.defs('EPSG:27700', '+proj=tmerc +lat_0=49 +lon_0=-2 +k=0.9996012717 ' +
+ '+x_0=400000 +y_0=-100000 +ellps=airy ' +
+ '+towgs84=446.448,-125.157,542.06,0.15,0.247,0.842,-20.489 ' +
+ '+units=m +no_defs');
+var proj27700 = ol.proj.get('EPSG:27700');
+proj27700.setExtent([0, 0, 700000, 1300000]);
+
+proj4.defs('EPSG:23032', '+proj=utm +zone=32 +ellps=intl ' +
+ '+towgs84=-87,-98,-121,0,0,0,0 +units=m +no_defs');
+var proj23032 = ol.proj.get('EPSG:23032');
+proj23032.setExtent([-1206118.71, 4021309.92, 1295389.00, 8051813.28]);
+
+proj4.defs('EPSG:5479', '+proj=lcc +lat_1=-76.66666666666667 +lat_2=' +
+ '-79.33333333333333 +lat_0=-78 +lon_0=163 +x_0=7000000 +y_0=5000000 ' +
+ '+ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs');
+var proj5479 = ol.proj.get('EPSG:5479');
+proj5479.setExtent([6825737.53, 4189159.80, 9633741.96, 5782472.71]);
+
+proj4.defs('EPSG:21781', '+proj=somerc +lat_0=46.95240555555556 ' +
+ '+lon_0=7.439583333333333 +k_0=1 +x_0=600000 +y_0=200000 +ellps=bessel ' +
+ '+towgs84=674.4,15.1,405.3,0,0,0,0 +units=m +no_defs');
+var proj21781 = ol.proj.get('EPSG:21781');
+proj21781.setExtent([485071.54, 75346.36, 828515.78, 299941.84]);
+
+proj4.defs('EPSG:3413', '+proj=stere +lat_0=90 +lat_ts=70 +lon_0=-45 +k=1 ' +
+ '+x_0=0 +y_0=0 +datum=WGS84 +units=m +no_defs');
+var proj3413 = ol.proj.get('EPSG:3413');
+proj3413.setExtent([-4194304, -4194304, 4194304, 4194304]);
+
+proj4.defs('EPSG:2163', '+proj=laea +lat_0=45 +lon_0=-100 +x_0=0 +y_0=0 ' +
+ '+a=6370997 +b=6370997 +units=m +no_defs');
+var proj2163 = ol.proj.get('EPSG:2163');
+proj2163.setExtent([-8040784.5135, -2577524.9210, 3668901.4484, 4785105.1096]);
+
+proj4.defs('ESRI:54009', '+proj=moll +lon_0=0 +x_0=0 +y_0=0 +datum=WGS84 ' +
+ '+units=m +no_defs');
+var proj54009 = ol.proj.get('ESRI:54009');
+proj54009.setExtent([-18e6, -9e6, 18e6, 9e6]);
+
+
+var layers = [];
+
+layers['bng'] = new ol.layer.Tile({
+ source: new ol.source.XYZ({
+ projection: 'EPSG:27700',
+ url: 'http://tileserver.maptiler.com/miniscale/{z}/{x}/{y}.png',
+ crossOrigin: '',
+ maxZoom: 6
+ })
+});
+
+layers['mapquest'] = new ol.layer.Tile({
+ source: new ol.source.MapQuest({layer: 'osm'})
+});
+
+layers['wms4326'] = new ol.layer.Tile({
+ source: new ol.source.TileWMS({
+ url: 'http://demo.boundlessgeo.com/geoserver/wms',
+ crossOrigin: '',
+ params: {
+ 'LAYERS': 'ne:NE1_HR_LC_SR_W_DR'
+ },
+ projection: 'EPSG:4326'
+ })
+});
+
+layers['wms21781'] = new ol.layer.Tile({
+ source: new ol.source.TileWMS({
+ attributions: [new ol.Attribution({
+ html: '© ' +
+ '' +
+ 'Pixelmap 1:1000000 / geo.admin.ch'
+ })],
+ crossOrigin: 'anonymous',
+ params: {
+ 'LAYERS': 'ch.swisstopo.pixelkarte-farbe-pk1000.noscale',
+ 'FORMAT': 'image/jpeg'
+ },
+ url: 'http://wms.geo.admin.ch/',
+ projection: 'EPSG:21781'
+ })
+});
+
+var parser = new ol.format.WMTSCapabilities();
+$.ajax('http://map1.vis.earthdata.nasa.gov/wmts-arctic/' +
+ 'wmts.cgi?SERVICE=WMTS&request=GetCapabilities').then(function(response) {
+ var result = parser.read(response);
+ var options = ol.source.WMTS.optionsFromCapabilities(result,
+ {layer: 'OSM_Land_Mask', matrixSet: 'EPSG3413_250m'});
+ options.crossOrigin = '';
+ options.projection = 'EPSG:3413';
+ options.wrapX = false;
+ layers['wmts3413'] = new ol.layer.Tile({
+ source: new ol.source.WMTS(options)
+ });
+});
+
+layers['grandcanyon'] = new ol.layer.Tile({
+ source: new ol.source.XYZ({
+ url: 'http://tileserver.maptiler.com/grandcanyon@2x/{z}/{x}/{y}.png',
+ crossOrigin: '',
+ tilePixelRatio: 2,
+ maxZoom: 15,
+ attributions: [new ol.Attribution({
+ html: 'Tiles © USGS, rendered with ' +
+ 'MapTiler'
+ })]
+ })
+});
+
+var startResolution =
+ ol.extent.getWidth(ol.proj.get('EPSG:3857').getExtent()) / 256;
+var resolutions = new Array(22);
+for (var i = 0, ii = resolutions.length; i < ii; ++i) {
+ resolutions[i] = startResolution / Math.pow(2, i);
+}
+
+layers['states'] = new ol.layer.Tile({
+ source: new ol.source.TileWMS({
+ url: 'http://demo.boundlessgeo.com/geoserver/wms',
+ crossOrigin: '',
+ params: {'LAYERS': 'topp:states', 'TILED': true},
+ serverType: 'geoserver',
+ tileGrid: new ol.tilegrid.TileGrid({
+ extent: [-13884991, 2870341, -7455066, 6338219],
+ resolutions: resolutions,
+ tileSize: [512, 256]
+ }),
+ projection: 'EPSG:3857'
+ })
+});
+
+
+var map = new ol.Map({
+ layers: [
+ layers['mapquest'],
+ layers['bng']
+ ],
+ renderer: common.getRendererFromQueryString(),
+ target: 'map',
+ view: new ol.View({
+ projection: 'EPSG:3857',
+ center: [0, 0],
+ zoom: 2
+ })
+});
+
+
+var baseLayerSelect = document.getElementById('base-layer');
+var overlayLayerSelect = document.getElementById('overlay-layer');
+var viewProjSelect = document.getElementById('view-projection');
+var renderEdgesCheckbox = document.getElementById('render-edges');
+var renderEdges = false;
+
+function updateViewProjection() {
+ var newProj = ol.proj.get(viewProjSelect.value);
+ var newProjExtent = newProj.getExtent();
+ var newView = new ol.View({
+ projection: newProj,
+ center: ol.extent.getCenter(newProjExtent || [0, 0, 0, 0]),
+ zoom: 0,
+ extent: newProjExtent || undefined
+ });
+ map.setView(newView);
+
+ // Example how to prevent double occurence of map by limiting layer extent
+ if (newProj == ol.proj.get('EPSG:3857')) {
+ layers['bng'].setExtent([-1057216, 6405988, 404315, 8759696]);
+ } else {
+ layers['bng'].setExtent(undefined);
+ }
+}
+
+
+/**
+ * @param {Event} e Change event.
+ */
+viewProjSelect.onchange = function(e) {
+ updateViewProjection();
+};
+
+updateViewProjection();
+
+var updateRenderEdgesOnLayer = function(layer) {
+ if (layer instanceof ol.layer.Tile) {
+ var source = layer.getSource();
+ if (source instanceof ol.source.TileImage) {
+ source.setRenderReprojectionEdges(renderEdges);
+ }
+ }
+};
+
+
+/**
+ * @param {Event} e Change event.
+ */
+baseLayerSelect.onchange = function(e) {
+ var layer = layers[baseLayerSelect.value];
+ if (layer) {
+ layer.setOpacity(1);
+ updateRenderEdgesOnLayer(layer);
+ map.getLayers().setAt(0, layer);
+ }
+};
+
+
+/**
+ * @param {Event} e Change event.
+ */
+overlayLayerSelect.onchange = function(e) {
+ var layer = layers[overlayLayerSelect.value];
+ if (layer) {
+ layer.setOpacity(0.7);
+ updateRenderEdgesOnLayer(layer);
+ map.getLayers().setAt(1, layer);
+ }
+};
+
+
+/**
+ * @param {Event} e Change event.
+ */
+renderEdgesCheckbox.onchange = function(e) {
+ renderEdges = renderEdgesCheckbox.checked;
+ map.getLayers().forEach(function(layer) {
+ updateRenderEdgesOnLayer(layer);
+ });
+};
diff --git a/externs/olx.js b/externs/olx.js
index b7abb24f37..60da9abb15 100644
--- a/externs/olx.js
+++ b/externs/olx.js
@@ -3602,6 +3602,7 @@ olx.source;
* key: string,
* imagerySet: string,
* maxZoom: (number|undefined),
+ * reprojectionErrorThreshold: (number|undefined),
* tileLoadFunction: (ol.TileLoadFunctionType|undefined),
* wrapX: (boolean|undefined)}}
* @api
@@ -3642,6 +3643,15 @@ olx.source.BingMapsOptions.prototype.imagerySet;
olx.source.BingMapsOptions.prototype.maxZoom;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.BingMapsOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* Optional function to load a tile given a URL.
* @type {ol.TileLoadFunctionType|undefined}
@@ -3761,6 +3771,7 @@ olx.source.TileUTFGridOptions.prototype.url;
* logo: (string|olx.LogoOptions|undefined),
* opaque: (boolean|undefined),
* projection: ol.proj.ProjectionLike,
+ * reprojectionErrorThreshold: (number|undefined),
* state: (ol.source.State|string|undefined),
* tileClass: (function(new: ol.ImageTile, ol.TileCoord,
* ol.TileState, string, ?string,
@@ -3819,6 +3830,15 @@ olx.source.TileImageOptions.prototype.opaque;
olx.source.TileImageOptions.prototype.projection;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.TileImageOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* Source state.
* @type {ol.source.State|string|undefined}
@@ -4076,6 +4096,7 @@ olx.source.ImageMapGuideOptions.prototype.params;
/**
* @typedef {{layer: string,
+ * reprojectionErrorThreshold: (number|undefined),
* tileLoadFunction: (ol.TileLoadFunctionType|undefined),
* url: (string|undefined)}}
* @api
@@ -4091,6 +4112,15 @@ olx.source.MapQuestOptions;
olx.source.MapQuestOptions.prototype.layer;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.MapQuestOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* Optional function to load a tile given a URL.
* @type {ol.TileLoadFunctionType|undefined}
@@ -4144,6 +4174,7 @@ olx.source.TileDebugOptions.prototype.wrapX;
* @typedef {{attributions: (Array.|undefined),
* crossOrigin: (null|string|undefined),
* maxZoom: (number|undefined),
+ * reprojectionErrorThreshold: (number|undefined),
* tileLoadFunction: (ol.TileLoadFunctionType|undefined),
* url: (string|undefined),
* wrapX: (boolean|undefined)}}
@@ -4182,6 +4213,15 @@ olx.source.OSMOptions.prototype.crossOrigin;
olx.source.OSMOptions.prototype.maxZoom;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.OSMOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* Optional function to load a tile given a URL.
* @type {ol.TileLoadFunctionType|undefined}
@@ -4538,6 +4578,7 @@ olx.source.ImageWMSOptions.prototype.url;
* minZoom: (number|undefined),
* maxZoom: (number|undefined),
* opaque: (boolean|undefined),
+ * reprojectionErrorThreshold: (number|undefined),
* tileLoadFunction: (ol.TileLoadFunctionType|undefined),
* url: (string|undefined)}}
* @api
@@ -4577,6 +4618,15 @@ olx.source.StamenOptions.prototype.maxZoom;
olx.source.StamenOptions.prototype.opaque;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.StamenOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* Optional function to load a tile given a URL.
* @type {ol.TileLoadFunctionType|undefined}
@@ -4683,6 +4733,7 @@ olx.source.ImageStaticOptions.prototype.url;
* logo: (string|olx.LogoOptions|undefined),
* tileGrid: (ol.tilegrid.TileGrid|undefined),
* projection: ol.proj.ProjectionLike,
+ * reprojectionErrorThreshold: (number|undefined),
* tileLoadFunction: (ol.TileLoadFunctionType|undefined),
* url: (string|undefined),
* urls: (Array.|undefined),
@@ -4754,6 +4805,15 @@ olx.source.TileArcGISRestOptions.prototype.tileGrid;
olx.source.TileArcGISRestOptions.prototype.projection;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.TileArcGISRestOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* Optional function to load a tile given a URL.
* @type {ol.TileLoadFunctionType|undefined}
@@ -4791,6 +4851,7 @@ olx.source.TileArcGISRestOptions.prototype.urls;
/**
* @typedef {{attributions: (Array.|undefined),
* crossOrigin: (null|string|undefined),
+ * reprojectionErrorThreshold: (number|undefined),
* tileLoadFunction: (ol.TileLoadFunctionType|undefined),
* url: string,
* wrapX: (boolean|undefined)}}
@@ -4821,6 +4882,15 @@ olx.source.TileJSONOptions.prototype.attributions;
olx.source.TileJSONOptions.prototype.crossOrigin;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.TileJSONOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* Optional function to load a tile given a URL.
* @type {ol.TileLoadFunctionType|undefined}
@@ -4855,6 +4925,7 @@ olx.source.TileJSONOptions.prototype.wrapX;
* tileGrid: (ol.tilegrid.TileGrid|undefined),
* maxZoom: (number|undefined),
* projection: ol.proj.ProjectionLike,
+ * reprojectionErrorThreshold: (number|undefined),
* serverType: (ol.source.wms.ServerType|string|undefined),
* tileLoadFunction: (ol.TileLoadFunctionType|undefined),
* url: (string|undefined),
@@ -4955,6 +5026,15 @@ olx.source.TileWMSOptions.prototype.maxZoom;
olx.source.TileWMSOptions.prototype.projection;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.TileWMSOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* The type of the remote WMS server. Currently only used when `hidpi` is
* `true`. Default is `undefined`.
@@ -5120,6 +5200,7 @@ olx.source.VectorOptions.prototype.wrapX;
* logo: (string|olx.LogoOptions|undefined),
* tileGrid: ol.tilegrid.WMTS,
* projection: ol.proj.ProjectionLike,
+ * reprojectionErrorThreshold: (number|undefined),
* requestEncoding: (ol.source.WMTSRequestEncoding|string|undefined),
* layer: string,
* style: string,
@@ -5185,6 +5266,15 @@ olx.source.WMTSOptions.prototype.tileGrid;
olx.source.WMTSOptions.prototype.projection;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.WMTSOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* Request encoding. Default is `KVP`.
* @type {ol.source.WMTSRequestEncoding|string|undefined}
@@ -5311,6 +5401,7 @@ olx.source.WMTSOptions.prototype.wrapX;
* crossOrigin: (null|string|undefined),
* logo: (string|olx.LogoOptions|undefined),
* projection: ol.proj.ProjectionLike,
+ * reprojectionErrorThreshold: (number|undefined),
* maxZoom: (number|undefined),
* minZoom: (number|undefined),
* tileGrid: (ol.tilegrid.TileGrid|undefined),
@@ -5362,6 +5453,15 @@ olx.source.XYZOptions.prototype.logo;
olx.source.XYZOptions.prototype.projection;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.XYZOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* Optional max zoom level. Default is `18`.
* @type {number|undefined}
@@ -5452,6 +5552,7 @@ olx.source.XYZOptions.prototype.wrapX;
* @typedef {{attributions: (Array.|undefined),
* crossOrigin: (null|string|undefined),
* logo: (string|olx.LogoOptions|undefined),
+ * reprojectionErrorThreshold: (number|undefined),
* url: !string,
* tierSizeCalculation: (string|undefined),
* size: ol.Size}}
@@ -5488,6 +5589,15 @@ olx.source.ZoomifyOptions.prototype.crossOrigin;
olx.source.ZoomifyOptions.prototype.logo;
+/**
+ * Maximum allowed reprojection error (in pixels). Default is `0.5`.
+ * Higher values can increase reprojection performance, but decrease precision.
+ * @type {number|undefined}
+ * @api
+ */
+olx.source.ZoomifyOptions.prototype.reprojectionErrorThreshold;
+
+
/**
* Prefix of URL template.
* @type {!string}
diff --git a/src/ol/math.js b/src/ol/math.js
index c0244bda01..6884b965eb 100644
--- a/src/ol/math.js
+++ b/src/ol/math.js
@@ -78,6 +78,69 @@ ol.math.squaredDistance = function(x1, y1, x2, y2) {
};
+/**
+ * Solves system of linear equations using Gaussian elimination method.
+ *
+ * @param {Array.>} mat Augmented matrix (n x n + 1 column)
+ * in row-major order.
+ * @return {Array.} The resulting vector.
+ */
+ol.math.solveLinearSystem = function(mat) {
+ var n = mat.length;
+
+ if (goog.asserts.ENABLE_ASSERTS) {
+ for (var row = 0; row < n; row++) {
+ goog.asserts.assert(mat[row].length == n + 1,
+ 'every row should have correct number of columns');
+ }
+ }
+
+ for (var i = 0; i < n; i++) {
+ // Find max in the i-th column (ignoring i - 1 first rows)
+ var maxRow = i;
+ var maxEl = Math.abs(mat[i][i]);
+ for (var r = i + 1; r < n; r++) {
+ var absValue = Math.abs(mat[r][i]);
+ if (absValue > maxEl) {
+ maxEl = absValue;
+ maxRow = r;
+ }
+ }
+
+ if (maxEl === 0) {
+ return null; // matrix is singular
+ }
+
+ // Swap max row with i-th (current) row
+ var tmp = mat[maxRow];
+ mat[maxRow] = mat[i];
+ mat[i] = tmp;
+
+ // Subtract the i-th row to make all the remaining rows 0 in the i-th column
+ for (var j = i + 1; j < n; j++) {
+ var coef = -mat[j][i] / mat[i][i];
+ for (var k = i; k < n + 1; k++) {
+ if (i == k) {
+ mat[j][k] = 0;
+ } else {
+ mat[j][k] += coef * mat[i][k];
+ }
+ }
+ }
+ }
+
+ // Solve Ax=b for upper triangular matrix A (mat)
+ var x = new Array(n);
+ for (var l = n - 1; l >= 0; l--) {
+ x[l] = mat[l][n] / mat[l][l];
+ for (var m = l - 1; m >= 0; m--) {
+ mat[m][n] -= mat[m][l] * x[l];
+ }
+ }
+ return x;
+};
+
+
/**
* Converts radians to to degrees.
*
diff --git a/src/ol/ol.js b/src/ol/ol.js
index cffdecbc29..23c754f49a 100644
--- a/src/ol/ol.js
+++ b/src/ol/ol.js
@@ -28,6 +28,13 @@ ol.DEFAULT_MAX_ZOOM = 42;
ol.DEFAULT_MIN_ZOOM = 0;
+/**
+ * @define {number} Default maximum allowed threshold (in pixels) for
+ * reprojection triangulation. Default is `0.5`.
+ */
+ol.DEFAULT_RASTER_REPROJECTION_ERROR_THRESHOLD = 0.5;
+
+
/**
* @define {number} Default high water mark.
*/
@@ -94,6 +101,13 @@ ol.ENABLE_NAMED_COLORS = false;
ol.ENABLE_PROJ4JS = true;
+/**
+ * @define {boolean} Enable automatic reprojection of raster sources. Default is
+ * `true`.
+ */
+ol.ENABLE_RASTER_REPROJECTION = true;
+
+
/**
* @define {boolean} Enable rendering of ol.layer.Tile based layers. Default is
* `true`. Setting this to false at compile time in advanced mode removes
@@ -159,6 +173,41 @@ ol.OVERVIEWMAP_MAX_RATIO = 0.75;
ol.OVERVIEWMAP_MIN_RATIO = 0.1;
+/**
+ * @define {number} Maximum number of source tiles for raster reprojection of
+ * a single tile.
+ * If too many source tiles are determined to be loaded to create a single
+ * reprojected tile the browser can become unresponsive or even crash.
+ * This can happen if the developer defines projections improperly and/or
+ * with unlimited extents.
+ * If too many tiles are required, no tiles are loaded and
+ * `ol.TileState.ERROR` state is set. Default is `100`.
+ */
+ol.RASTER_REPROJECTION_MAX_SOURCE_TILES = 100;
+
+
+/**
+ * @define {number} Maximum number of subdivision steps during raster
+ * reprojection triangulation. Prevents high memory usage and large
+ * number of proj4 calls (for certain transformations and areas).
+ * At most `2*(2^this)` triangles are created for each triangulated
+ * extent (tile/image). Default is `10`.
+ */
+ol.RASTER_REPROJECTION_MAX_SUBDIVISION = 10;
+
+
+/**
+ * @define {number} Maximum allowed size of triangle relative to world width.
+ * When transforming corners of world extent between certain projections,
+ * the resulting triangulation seems to have zero error and no subdivision
+ * is performed.
+ * If the triangle width is more than this (relative to world width; 0-1),
+ * subdivison is forced (up to `ol.RASTER_REPROJECTION_MAX_SUBDIVISION`).
+ * Default is `0.25`.
+ */
+ol.RASTER_REPROJECTION_MAX_TRIANGLE_WIDTH = 0.25;
+
+
/**
* @define {number} Tolerance for geometry simplification in device pixels.
*/
diff --git a/src/ol/renderer/canvas/canvasimagelayerrenderer.js b/src/ol/renderer/canvas/canvasimagelayerrenderer.js
index c752dd4630..22fae30287 100644
--- a/src/ol/renderer/canvas/canvasimagelayerrenderer.js
+++ b/src/ol/renderer/canvas/canvasimagelayerrenderer.js
@@ -170,11 +170,13 @@ ol.renderer.canvas.ImageLayer.prototype.prepareFrame =
if (!hints[ol.ViewHint.ANIMATING] && !hints[ol.ViewHint.INTERACTING] &&
!ol.extent.isEmpty(renderedExtent)) {
var projection = viewState.projection;
- var sourceProjection = imageSource.getProjection();
- if (sourceProjection) {
- goog.asserts.assert(ol.proj.equivalent(projection, sourceProjection),
- 'projection and sourceProjection are equivalent');
- projection = sourceProjection;
+ if (!ol.ENABLE_RASTER_REPROJECTION) {
+ var sourceProjection = imageSource.getProjection();
+ if (sourceProjection) {
+ goog.asserts.assert(ol.proj.equivalent(projection, sourceProjection),
+ 'projection and sourceProjection are equivalent');
+ projection = sourceProjection;
+ }
}
image = imageSource.getImage(
renderedExtent, viewResolution, pixelRatio, projection);
diff --git a/src/ol/renderer/canvas/canvastilelayerrenderer.js b/src/ol/renderer/canvas/canvastilelayerrenderer.js
index e6ce046ab9..14ada48772 100644
--- a/src/ol/renderer/canvas/canvastilelayerrenderer.js
+++ b/src/ol/renderer/canvas/canvastilelayerrenderer.js
@@ -309,7 +309,8 @@ ol.renderer.canvas.TileLayer.prototype.prepareFrame =
/** @type {Array.} */
var tilesToClear = [];
- var findLoadedTiles = this.createLoadedTileFinder(tileSource, tilesToDrawByZ);
+ var findLoadedTiles = this.createLoadedTileFinder(
+ tileSource, projection, tilesToDrawByZ);
var useInterimTilesOnError = tileLayer.getUseInterimTilesOnError();
diff --git a/src/ol/renderer/dom/domimagelayerrenderer.js b/src/ol/renderer/dom/domimagelayerrenderer.js
index 29308d3ee2..033de8614b 100644
--- a/src/ol/renderer/dom/domimagelayerrenderer.js
+++ b/src/ol/renderer/dom/domimagelayerrenderer.js
@@ -101,11 +101,13 @@ ol.renderer.dom.ImageLayer.prototype.prepareFrame =
if (!hints[ol.ViewHint.ANIMATING] && !hints[ol.ViewHint.INTERACTING] &&
!ol.extent.isEmpty(renderedExtent)) {
var projection = viewState.projection;
- var sourceProjection = imageSource.getProjection();
- if (sourceProjection) {
- goog.asserts.assert(ol.proj.equivalent(projection, sourceProjection),
- 'projection and sourceProjection are equivalent');
- projection = sourceProjection;
+ if (!ol.ENABLE_RASTER_REPROJECTION) {
+ var sourceProjection = imageSource.getProjection();
+ if (sourceProjection) {
+ goog.asserts.assert(ol.proj.equivalent(projection, sourceProjection),
+ 'projection and sourceProjection are equivalent');
+ projection = sourceProjection;
+ }
}
var image_ = imageSource.getImage(renderedExtent, viewResolution,
frameState.pixelRatio, projection);
diff --git a/src/ol/renderer/dom/domtilelayerrenderer.js b/src/ol/renderer/dom/domtilelayerrenderer.js
index eb93e39538..d9a66b5f7e 100644
--- a/src/ol/renderer/dom/domtilelayerrenderer.js
+++ b/src/ol/renderer/dom/domtilelayerrenderer.js
@@ -121,7 +121,8 @@ ol.renderer.dom.TileLayer.prototype.prepareFrame =
var tilesToDrawByZ = {};
tilesToDrawByZ[z] = {};
- var findLoadedTiles = this.createLoadedTileFinder(tileSource, tilesToDrawByZ);
+ var findLoadedTiles = this.createLoadedTileFinder(
+ tileSource, projection, tilesToDrawByZ);
var useInterimTilesOnError = tileLayer.getUseInterimTilesOnError();
diff --git a/src/ol/renderer/layerrenderer.js b/src/ol/renderer/layerrenderer.js
index f6dd738106..cd0dba0093 100644
--- a/src/ol/renderer/layerrenderer.js
+++ b/src/ol/renderer/layerrenderer.js
@@ -87,6 +87,7 @@ ol.renderer.Layer.prototype.hasFeatureAtCoordinate = goog.functions.FALSE;
/**
* Create a function that adds loaded tiles to the tile lookup.
* @param {ol.source.Tile} source Tile source.
+ * @param {ol.proj.Projection} projection Projection of the tiles.
* @param {Object.>} tiles Lookup of loaded
* tiles by zoom level.
* @return {function(number, ol.TileRange):boolean} A function that can be
@@ -94,7 +95,8 @@ ol.renderer.Layer.prototype.hasFeatureAtCoordinate = goog.functions.FALSE;
* lookup.
* @protected
*/
-ol.renderer.Layer.prototype.createLoadedTileFinder = function(source, tiles) {
+ol.renderer.Layer.prototype.createLoadedTileFinder =
+ function(source, projection, tiles) {
return (
/**
* @param {number} zoom Zoom level.
@@ -102,12 +104,13 @@ ol.renderer.Layer.prototype.createLoadedTileFinder = function(source, tiles) {
* @return {boolean} The tile range is fully loaded.
*/
function(zoom, tileRange) {
- return source.forEachLoadedTile(zoom, tileRange, function(tile) {
- if (!tiles[zoom]) {
- tiles[zoom] = {};
- }
- tiles[zoom][tile.tileCoord.toString()] = tile;
- });
+ return source.forEachLoadedTile(projection, zoom,
+ tileRange, function(tile) {
+ if (!tiles[zoom]) {
+ tiles[zoom] = {};
+ }
+ tiles[zoom][tile.tileCoord.toString()] = tile;
+ });
});
};
@@ -156,8 +159,9 @@ ol.renderer.Layer.prototype.loadImage = function(image) {
if (imageState == ol.ImageState.IDLE) {
image.load();
imageState = image.getState();
- goog.asserts.assert(imageState == ol.ImageState.LOADING,
- 'imageState is "loading"');
+ goog.asserts.assert(imageState == ol.ImageState.LOADING ||
+ imageState == ol.ImageState.LOADED,
+ 'imageState is "loading" or "loaded"');
}
return imageState == ol.ImageState.LOADED;
};
@@ -191,7 +195,8 @@ ol.renderer.Layer.prototype.scheduleExpireCache =
*/
function(tileSource, map, frameState) {
var tileSourceKey = goog.getUid(tileSource).toString();
- tileSource.expireCache(frameState.usedTiles[tileSourceKey]);
+ tileSource.expireCache(frameState.viewState.projection,
+ frameState.usedTiles[tileSourceKey]);
}, tileSource));
}
};
@@ -324,7 +329,7 @@ ol.renderer.Layer.prototype.manageTilePyramid = function(
opt_tileCallback.call(opt_this, tile);
}
} else {
- tileSource.useTile(z, x, y);
+ tileSource.useTile(z, x, y, projection);
}
}
}
diff --git a/src/ol/renderer/webgl/webglimagelayerrenderer.js b/src/ol/renderer/webgl/webglimagelayerrenderer.js
index 317544fc66..8aad156335 100644
--- a/src/ol/renderer/webgl/webglimagelayerrenderer.js
+++ b/src/ol/renderer/webgl/webglimagelayerrenderer.js
@@ -125,11 +125,13 @@ ol.renderer.webgl.ImageLayer.prototype.prepareFrame =
if (!hints[ol.ViewHint.ANIMATING] && !hints[ol.ViewHint.INTERACTING] &&
!ol.extent.isEmpty(renderedExtent)) {
var projection = viewState.projection;
- var sourceProjection = imageSource.getProjection();
- if (sourceProjection) {
- goog.asserts.assert(ol.proj.equivalent(projection, sourceProjection),
- 'projection and sourceProjection are equivalent');
- projection = sourceProjection;
+ if (!ol.ENABLE_RASTER_REPROJECTION) {
+ var sourceProjection = imageSource.getProjection();
+ if (sourceProjection) {
+ goog.asserts.assert(ol.proj.equivalent(projection, sourceProjection),
+ 'projection and sourceProjection are equivalent');
+ projection = sourceProjection;
+ }
}
var image_ = imageSource.getImage(renderedExtent, viewResolution,
pixelRatio, projection);
diff --git a/src/ol/renderer/webgl/webgltilelayerrenderer.js b/src/ol/renderer/webgl/webgltilelayerrenderer.js
index 6752062548..675aac21ae 100644
--- a/src/ol/renderer/webgl/webgltilelayerrenderer.js
+++ b/src/ol/renderer/webgl/webgltilelayerrenderer.js
@@ -105,6 +105,7 @@ ol.renderer.webgl.TileLayer.prototype.disposeInternal = function() {
/**
* Create a function that adds loaded tiles to the tile lookup.
* @param {ol.source.Tile} source Tile source.
+ * @param {ol.proj.Projection} projection Projection of the tiles.
* @param {Object.>} tiles Lookup of loaded
* tiles by zoom level.
* @return {function(number, ol.TileRange):boolean} A function that can be
@@ -113,7 +114,7 @@ ol.renderer.webgl.TileLayer.prototype.disposeInternal = function() {
* @protected
*/
ol.renderer.webgl.TileLayer.prototype.createLoadedTileFinder =
- function(source, tiles) {
+ function(source, projection, tiles) {
var mapRenderer = this.mapRenderer;
return (
@@ -123,16 +124,17 @@ ol.renderer.webgl.TileLayer.prototype.createLoadedTileFinder =
* @return {boolean} The tile range is fully loaded.
*/
function(zoom, tileRange) {
- return source.forEachLoadedTile(zoom, tileRange, function(tile) {
- var loaded = mapRenderer.isTileTextureLoaded(tile);
- if (loaded) {
- if (!tiles[zoom]) {
- tiles[zoom] = {};
- }
- tiles[zoom][tile.tileCoord.toString()] = tile;
- }
- return loaded;
- });
+ return source.forEachLoadedTile(projection, zoom,
+ tileRange, function(tile) {
+ var loaded = mapRenderer.isTileTextureLoaded(tile);
+ if (loaded) {
+ if (!tiles[zoom]) {
+ tiles[zoom] = {};
+ }
+ tiles[zoom][tile.tileCoord.toString()] = tile;
+ }
+ return loaded;
+ });
});
};
@@ -239,7 +241,7 @@ ol.renderer.webgl.TileLayer.prototype.prepareFrame =
tilesToDrawByZ[z] = {};
var findLoadedTiles = this.createLoadedTileFinder(
- tileSource, tilesToDrawByZ);
+ tileSource, projection, tilesToDrawByZ);
var useInterimTilesOnError = tileLayer.getUseInterimTilesOnError();
var allTilesLoaded = true;
diff --git a/src/ol/reproj/image.js b/src/ol/reproj/image.js
new file mode 100644
index 0000000000..0292d7168e
--- /dev/null
+++ b/src/ol/reproj/image.js
@@ -0,0 +1,210 @@
+goog.provide('ol.reproj.Image');
+goog.provide('ol.reproj.ImageFunctionType');
+
+goog.require('goog.asserts');
+goog.require('goog.events');
+goog.require('goog.events.EventType');
+goog.require('ol.ImageBase');
+goog.require('ol.ImageState');
+goog.require('ol.extent');
+goog.require('ol.proj');
+goog.require('ol.reproj');
+goog.require('ol.reproj.Triangulation');
+
+
+/**
+ * @typedef {function(ol.Extent, number, number) : ol.ImageBase}
+ */
+ol.reproj.ImageFunctionType;
+
+
+
+/**
+ * @classdesc
+ * Class encapsulating single reprojected image.
+ * See {@link ol.source.Image}.
+ *
+ * @constructor
+ * @extends {ol.ImageBase}
+ * @param {ol.proj.Projection} sourceProj Source projection (of the data).
+ * @param {ol.proj.Projection} targetProj Target projection.
+ * @param {ol.Extent} targetExtent Target extent.
+ * @param {number} targetResolution Target resolution.
+ * @param {number} pixelRatio Pixel ratio.
+ * @param {ol.reproj.ImageFunctionType} getImageFunction
+ * Function returning source images (extent, resolution, pixelRatio).
+ */
+ol.reproj.Image = function(sourceProj, targetProj,
+ targetExtent, targetResolution, pixelRatio, getImageFunction) {
+
+ /**
+ * @private
+ * @type {ol.proj.Projection}
+ */
+ this.targetProj_ = targetProj;
+
+ /**
+ * @private
+ * @type {ol.Extent}
+ */
+ this.maxSourceExtent_ = sourceProj.getExtent();
+ var maxTargetExtent = targetProj.getExtent();
+
+ var limitedTargetExtent = maxTargetExtent ?
+ ol.extent.getIntersection(targetExtent, maxTargetExtent) : targetExtent;
+
+ var targetCenter = ol.extent.getCenter(limitedTargetExtent);
+ var sourceResolution = ol.reproj.calculateSourceResolution(
+ sourceProj, targetProj, targetCenter, targetResolution);
+
+ var errorThresholdInPixels = ol.DEFAULT_RASTER_REPROJECTION_ERROR_THRESHOLD;
+
+ /**
+ * @private
+ * @type {!ol.reproj.Triangulation}
+ */
+ this.triangulation_ = new ol.reproj.Triangulation(
+ sourceProj, targetProj, limitedTargetExtent, this.maxSourceExtent_,
+ sourceResolution * errorThresholdInPixels);
+
+ /**
+ * @private
+ * @type {number}
+ */
+ this.targetResolution_ = targetResolution;
+
+ /**
+ * @private
+ * @type {ol.Extent}
+ */
+ this.targetExtent_ = targetExtent;
+
+ var sourceExtent = this.triangulation_.calculateSourceExtent();
+
+ /**
+ * @private
+ * @type {ol.ImageBase}
+ */
+ this.sourceImage_ =
+ getImageFunction(sourceExtent, sourceResolution, pixelRatio);
+
+ /**
+ * @private
+ * @type {number}
+ */
+ this.sourcePixelRatio_ =
+ this.sourceImage_ ? this.sourceImage_.getPixelRatio() : 1;
+
+ /**
+ * @private
+ * @type {HTMLCanvasElement}
+ */
+ this.canvas_ = null;
+
+ /**
+ * @private
+ * @type {goog.events.Key}
+ */
+ this.sourceListenerKey_ = null;
+
+
+ var state = ol.ImageState.LOADED;
+ var attributions = [];
+
+ if (this.sourceImage_) {
+ state = ol.ImageState.IDLE;
+ attributions = this.sourceImage_.getAttributions();
+ }
+
+ goog.base(this, targetExtent, targetResolution, this.sourcePixelRatio_,
+ state, attributions);
+};
+goog.inherits(ol.reproj.Image, ol.ImageBase);
+
+
+/**
+ * @inheritDoc
+ */
+ol.reproj.Image.prototype.disposeInternal = function() {
+ if (this.state == ol.ImageState.LOADING) {
+ this.unlistenSource_();
+ }
+ goog.base(this, 'disposeInternal');
+};
+
+
+/**
+ * @inheritDoc
+ */
+ol.reproj.Image.prototype.getImage = function(opt_context) {
+ return this.canvas_;
+};
+
+
+/**
+ * @return {ol.proj.Projection} Projection.
+ */
+ol.reproj.Image.prototype.getProjection = function() {
+ return this.targetProj_;
+};
+
+
+/**
+ * @private
+ */
+ol.reproj.Image.prototype.reproject_ = function() {
+ var sourceState = this.sourceImage_.getState();
+ if (sourceState == ol.ImageState.LOADED) {
+ var width = ol.extent.getWidth(this.targetExtent_) / this.targetResolution_;
+ var height =
+ ol.extent.getHeight(this.targetExtent_) / this.targetResolution_;
+
+ this.canvas_ = ol.reproj.render(width, height, this.sourcePixelRatio_,
+ this.sourceImage_.getResolution(), this.maxSourceExtent_,
+ this.targetResolution_, this.targetExtent_, this.triangulation_, [{
+ extent: this.sourceImage_.getExtent(),
+ image: this.sourceImage_.getImage()
+ }]);
+ }
+ this.state = sourceState;
+ this.changed();
+};
+
+
+/**
+ * @inheritDoc
+ */
+ol.reproj.Image.prototype.load = function() {
+ if (this.state == ol.ImageState.IDLE) {
+ this.state = ol.ImageState.LOADING;
+ this.changed();
+
+ var sourceState = this.sourceImage_.getState();
+ if (sourceState == ol.ImageState.LOADED ||
+ sourceState == ol.ImageState.ERROR) {
+ this.reproject_();
+ } else {
+ this.sourceListenerKey_ = this.sourceImage_.listen(
+ goog.events.EventType.CHANGE, function(e) {
+ var sourceState = this.sourceImage_.getState();
+ if (sourceState == ol.ImageState.LOADED ||
+ sourceState == ol.ImageState.ERROR) {
+ this.unlistenSource_();
+ this.reproject_();
+ }
+ }, false, this);
+ this.sourceImage_.load();
+ }
+ }
+};
+
+
+/**
+ * @private
+ */
+ol.reproj.Image.prototype.unlistenSource_ = function() {
+ goog.asserts.assert(this.sourceListenerKey_,
+ 'this.sourceListenerKey_ should not be null');
+ goog.events.unlistenByKey(this.sourceListenerKey_);
+ this.sourceListenerKey_ = null;
+};
diff --git a/src/ol/reproj/reproj.js b/src/ol/reproj/reproj.js
new file mode 100644
index 0000000000..fbca15fa90
--- /dev/null
+++ b/src/ol/reproj/reproj.js
@@ -0,0 +1,258 @@
+goog.provide('ol.reproj');
+
+goog.require('goog.labs.userAgent.browser');
+goog.require('goog.labs.userAgent.platform');
+goog.require('goog.math');
+goog.require('ol.dom');
+goog.require('ol.extent');
+goog.require('ol.math');
+goog.require('ol.proj');
+
+
+/**
+ * We need to employ more sophisticated solution
+ * if the web browser antialiases clipping edges on canvas.
+ *
+ * Currently only Chrome does not antialias the edges, but this is probably
+ * going to be "fixed" in the future: http://crbug.com/424291
+ *
+ * @type {boolean}
+ * @private
+ */
+ol.reproj.browserAntialiasesClip_ = !goog.labs.userAgent.browser.isChrome() ||
+ goog.labs.userAgent.platform.isIos();
+
+
+/**
+ * Calculates ideal resolution to use from the source in order to achieve
+ * pixel mapping as close as possible to 1:1 during reprojection.
+ * The resolution is calculated regardless of what resolutions
+ * are actually available in the dataset (TileGrid, Image, ...).
+ *
+ * @param {ol.proj.Projection} sourceProj Source projection.
+ * @param {ol.proj.Projection} targetProj Target projection.
+ * @param {ol.Coordinate} targetCenter Target center.
+ * @param {number} targetResolution Target resolution.
+ * @return {number} The best resolution to use. Can be +-Infinity, NaN or 0.
+ */
+ol.reproj.calculateSourceResolution = function(sourceProj, targetProj,
+ targetCenter, targetResolution) {
+
+ var sourceCenter = ol.proj.transform(targetCenter, targetProj, sourceProj);
+
+ // calculate the ideal resolution of the source data
+ var sourceResolution =
+ targetProj.getPointResolution(targetResolution, targetCenter);
+
+ var targetMetersPerUnit = targetProj.getMetersPerUnit();
+ if (targetMetersPerUnit !== undefined) {
+ sourceResolution *= targetMetersPerUnit;
+ }
+ var sourceMetersPerUnit = sourceProj.getMetersPerUnit();
+ if (sourceMetersPerUnit !== undefined) {
+ sourceResolution /= sourceMetersPerUnit;
+ }
+
+ // Based on the projection properties, the point resolution at the specified
+ // coordinates may be slightly different. We need to reverse-compensate this
+ // in order to achieve optimal results.
+
+ var compensationFactor =
+ sourceProj.getPointResolution(sourceResolution, sourceCenter) /
+ sourceResolution;
+
+ if (goog.math.isFiniteNumber(compensationFactor) && compensationFactor > 0) {
+ sourceResolution /= compensationFactor;
+ }
+
+ return sourceResolution;
+};
+
+
+/**
+ * Enlarge the clipping triangle point by 1 pixel to ensure the edges overlap
+ * in order to mask gaps caused by antialiasing.
+ *
+ * @param {number} centroidX Centroid of the triangle (x coordinate in pixels).
+ * @param {number} centroidY Centroid of the triangle (y coordinate in pixels).
+ * @param {number} x X coordinate of the point (in pixels).
+ * @param {number} y Y coordinate of the point (in pixels).
+ * @return {ol.Coordinate} New point 1 px farther from the centroid.
+ * @private
+ */
+ol.reproj.enlargeClipPoint_ = function(centroidX, centroidY, x, y) {
+ var dX = x - centroidX, dY = y - centroidY;
+ var distance = Math.sqrt(dX * dX + dY * dY);
+ return [Math.round(x + dX / distance), Math.round(y + dY / distance)];
+};
+
+
+/**
+ * Renders the source data into new canvas based on the triangulation.
+ *
+ * @param {number} width Width of the canvas.
+ * @param {number} height Height of the canvas.
+ * @param {number} pixelRatio Pixel ratio.
+ * @param {number} sourceResolution Source resolution.
+ * @param {ol.Extent} sourceExtent Extent of the data source.
+ * @param {number} targetResolution Target resolution.
+ * @param {ol.Extent} targetExtent Target extent.
+ * @param {ol.reproj.Triangulation} triangulation Calculated triangulation.
+ * @param {Array.<{extent: ol.Extent,
+ * image: (HTMLCanvasElement|Image|HTMLVideoElement)}>} sources
+ * Array of sources.
+ * @param {boolean=} opt_renderEdges Render reprojection edges.
+ * @return {HTMLCanvasElement} Canvas with reprojected data.
+ */
+ol.reproj.render = function(width, height, pixelRatio,
+ sourceResolution, sourceExtent, targetResolution, targetExtent,
+ triangulation, sources, opt_renderEdges) {
+
+ var context = ol.dom.createCanvasContext2D(Math.round(pixelRatio * width),
+ Math.round(pixelRatio * height));
+
+ if (sources.length === 0) {
+ return context.canvas;
+ }
+
+ context.scale(pixelRatio, pixelRatio);
+
+ var sourceDataExtent = ol.extent.createEmpty();
+ sources.forEach(function(src, i, arr) {
+ ol.extent.extend(sourceDataExtent, src.extent);
+ });
+
+ var canvasWidthInUnits = ol.extent.getWidth(sourceDataExtent);
+ var canvasHeightInUnits = ol.extent.getHeight(sourceDataExtent);
+ var stitchContext = ol.dom.createCanvasContext2D(
+ Math.round(pixelRatio * canvasWidthInUnits / sourceResolution),
+ Math.round(pixelRatio * canvasHeightInUnits / sourceResolution));
+
+ stitchContext.scale(pixelRatio / sourceResolution,
+ pixelRatio / sourceResolution);
+ stitchContext.translate(-sourceDataExtent[0], sourceDataExtent[3]);
+
+ sources.forEach(function(src, i, arr) {
+ var xPos = src.extent[0];
+ var yPos = -src.extent[3];
+ var srcWidth = ol.extent.getWidth(src.extent);
+ var srcHeight = ol.extent.getHeight(src.extent);
+
+ stitchContext.drawImage(src.image, xPos, yPos, srcWidth, srcHeight);
+ });
+
+ var targetTopLeft = ol.extent.getTopLeft(targetExtent);
+
+ triangulation.getTriangles().forEach(function(triangle, i, arr) {
+ /* Calculate affine transform (src -> dst)
+ * Resulting matrix can be used to transform coordinate
+ * from `sourceProjection` to destination pixels.
+ *
+ * To optimize number of context calls and increase numerical stability,
+ * we also do the following operations:
+ * trans(-topLeftExtentCorner), scale(1 / targetResolution), scale(1, -1)
+ * here before solving the linear system so [ui, vi] are pixel coordinates.
+ *
+ * Src points: xi, yi
+ * Dst points: ui, vi
+ * Affine coefficients: aij
+ *
+ * | x0 y0 1 0 0 0 | |a00| |u0|
+ * | x1 y1 1 0 0 0 | |a01| |u1|
+ * | x2 y2 1 0 0 0 | x |a02| = |u2|
+ * | 0 0 0 x0 y0 1 | |a10| |v0|
+ * | 0 0 0 x1 y1 1 | |a11| |v1|
+ * | 0 0 0 x2 y2 1 | |a12| |v2|
+ */
+ var source = triangle.source, target = triangle.target;
+ var x0 = source[0][0], y0 = source[0][1],
+ x1 = source[1][0], y1 = source[1][1],
+ x2 = source[2][0], y2 = source[2][1];
+ var u0 = (target[0][0] - targetTopLeft[0]) / targetResolution,
+ v0 = -(target[0][1] - targetTopLeft[1]) / targetResolution;
+ var u1 = (target[1][0] - targetTopLeft[0]) / targetResolution,
+ v1 = -(target[1][1] - targetTopLeft[1]) / targetResolution;
+ var u2 = (target[2][0] - targetTopLeft[0]) / targetResolution,
+ v2 = -(target[2][1] - targetTopLeft[1]) / targetResolution;
+
+ // Shift all the source points to improve numerical stability
+ // of all the subsequent calculations. The [x0, y0] is used here.
+ // This is also used to simplify the linear system.
+ var sourceNumericalShiftX = x0, sourceNumericalShiftY = y0;
+ x0 = 0;
+ y0 = 0;
+ x1 -= sourceNumericalShiftX;
+ y1 -= sourceNumericalShiftY;
+ x2 -= sourceNumericalShiftX;
+ y2 -= sourceNumericalShiftY;
+
+ var augmentedMatrix = [
+ [x1, y1, 0, 0, u1 - u0],
+ [x2, y2, 0, 0, u2 - u0],
+ [0, 0, x1, y1, v1 - v0],
+ [0, 0, x2, y2, v2 - v0]
+ ];
+ var affineCoefs = ol.math.solveLinearSystem(augmentedMatrix);
+ if (!affineCoefs) {
+ return;
+ }
+
+ context.save();
+ context.beginPath();
+ if (ol.reproj.browserAntialiasesClip_) {
+ var centroidX = (u0 + u1 + u2) / 3, centroidY = (v0 + v1 + v2) / 3;
+ var p0 = ol.reproj.enlargeClipPoint_(centroidX, centroidY, u0, v0);
+ var p1 = ol.reproj.enlargeClipPoint_(centroidX, centroidY, u1, v1);
+ var p2 = ol.reproj.enlargeClipPoint_(centroidX, centroidY, u2, v2);
+
+ context.moveTo(p0[0], p0[1]);
+ context.lineTo(p1[0], p1[1]);
+ context.lineTo(p2[0], p2[1]);
+ } else {
+ context.moveTo(u0, v0);
+ context.lineTo(u1, v1);
+ context.lineTo(u2, v2);
+ }
+ context.closePath();
+ context.clip();
+
+ context.transform(
+ affineCoefs[0], affineCoefs[2], affineCoefs[1], affineCoefs[3], u0, v0);
+
+ context.translate(sourceDataExtent[0] - sourceNumericalShiftX,
+ sourceDataExtent[3] - sourceNumericalShiftY);
+
+ context.scale(sourceResolution / pixelRatio,
+ -sourceResolution / pixelRatio);
+
+ context.drawImage(stitchContext.canvas, 0, 0);
+ context.restore();
+ });
+
+ if (opt_renderEdges) {
+ context.save();
+
+ context.strokeStyle = 'black';
+ context.lineWidth = 1;
+
+ triangulation.getTriangles().forEach(function(triangle, i, arr) {
+ var target = triangle.target;
+ var u0 = (target[0][0] - targetTopLeft[0]) / targetResolution,
+ v0 = -(target[0][1] - targetTopLeft[1]) / targetResolution;
+ var u1 = (target[1][0] - targetTopLeft[0]) / targetResolution,
+ v1 = -(target[1][1] - targetTopLeft[1]) / targetResolution;
+ var u2 = (target[2][0] - targetTopLeft[0]) / targetResolution,
+ v2 = -(target[2][1] - targetTopLeft[1]) / targetResolution;
+
+ context.beginPath();
+ context.moveTo(u0, v0);
+ context.lineTo(u1, v1);
+ context.lineTo(u2, v2);
+ context.closePath();
+ context.stroke();
+ });
+
+ context.restore();
+ }
+ return context.canvas;
+};
diff --git a/src/ol/reproj/tile.js b/src/ol/reproj/tile.js
new file mode 100644
index 0000000000..a1c819d620
--- /dev/null
+++ b/src/ol/reproj/tile.js
@@ -0,0 +1,332 @@
+goog.provide('ol.reproj.Tile');
+goog.provide('ol.reproj.TileFunctionType');
+
+goog.require('goog.asserts');
+goog.require('goog.events');
+goog.require('goog.events.EventType');
+goog.require('goog.math');
+goog.require('goog.object');
+goog.require('ol.Tile');
+goog.require('ol.TileState');
+goog.require('ol.extent');
+goog.require('ol.math');
+goog.require('ol.proj');
+goog.require('ol.reproj');
+goog.require('ol.reproj.Triangulation');
+
+
+/**
+ * @typedef {function(number, number, number, number) : ol.Tile}
+ */
+ol.reproj.TileFunctionType;
+
+
+
+/**
+ * @classdesc
+ * Class encapsulating single reprojected tile.
+ * See {@link ol.source.TileImage}.
+ *
+ * @constructor
+ * @extends {ol.Tile}
+ * @param {ol.proj.Projection} sourceProj Source projection.
+ * @param {ol.tilegrid.TileGrid} sourceTileGrid Source tile grid.
+ * @param {ol.proj.Projection} targetProj Target projection.
+ * @param {ol.tilegrid.TileGrid} targetTileGrid Target tile grid.
+ * @param {number} z Zoom level.
+ * @param {number} x X.
+ * @param {number} y Y.
+ * @param {number} pixelRatio Pixel ratio.
+ * @param {ol.reproj.TileFunctionType} getTileFunction
+ * Function returning source tiles (z, x, y, pixelRatio).
+ * @param {number=} opt_errorThreshold Acceptable reprojection error (in px).
+ * @param {boolean=} opt_renderEdges Render reprojection edges.
+ */
+ol.reproj.Tile = function(sourceProj, sourceTileGrid,
+ targetProj, targetTileGrid, z, x, y, pixelRatio, getTileFunction,
+ opt_errorThreshold,
+ opt_renderEdges) {
+ goog.base(this, [z, x, y], ol.TileState.IDLE);
+
+ /**
+ * @private
+ * @type {boolean}
+ */
+ this.renderEdges_ = opt_renderEdges !== undefined ? opt_renderEdges : false;
+
+ /**
+ * @private
+ * @type {number}
+ */
+ this.pixelRatio_ = pixelRatio;
+
+ /**
+ * @private
+ * @type {HTMLCanvasElement}
+ */
+ this.canvas_ = null;
+
+ /**
+ * @private
+ * @type {Object.}
+ */
+ this.canvasByContext_ = {};
+
+ /**
+ * @private
+ * @type {ol.tilegrid.TileGrid}
+ */
+ this.sourceTileGrid_ = sourceTileGrid;
+
+ /**
+ * @private
+ * @type {ol.tilegrid.TileGrid}
+ */
+ this.targetTileGrid_ = targetTileGrid;
+
+ /**
+ * @private
+ * @type {!Array.}
+ */
+ this.sourceTiles_ = [];
+
+ /**
+ * @private
+ * @type {Array.}
+ */
+ this.sourcesListenerKeys_ = null;
+
+ /**
+ * @private
+ * @type {number}
+ */
+ this.sourceZ_ = 0;
+
+ var targetExtent = targetTileGrid.getTileCoordExtent(this.getTileCoord());
+ var maxTargetExtent = this.targetTileGrid_.getExtent();
+ var maxSourceExtent = this.sourceTileGrid_.getExtent();
+
+ var limitedTargetExtent = maxTargetExtent ?
+ ol.extent.getIntersection(targetExtent, maxTargetExtent) : targetExtent;
+
+ if (ol.extent.getArea(limitedTargetExtent) === 0) {
+ // Tile is completely outside range -> EMPTY
+ // TODO: is it actually correct that the source even creates the tile ?
+ this.state = ol.TileState.EMPTY;
+ return;
+ }
+
+ var sourceProjExtent = sourceProj.getExtent();
+ if (sourceProjExtent) {
+ if (!maxSourceExtent) {
+ maxSourceExtent = sourceProjExtent;
+ } else {
+ maxSourceExtent = ol.extent.getIntersection(
+ maxSourceExtent, sourceProjExtent);
+ }
+ }
+
+ var targetResolution = targetTileGrid.getResolution(z);
+
+ var targetCenter = ol.extent.getCenter(limitedTargetExtent);
+ var sourceResolution = ol.reproj.calculateSourceResolution(
+ sourceProj, targetProj, targetCenter, targetResolution);
+
+ if (!goog.math.isFiniteNumber(sourceResolution) || sourceResolution <= 0) {
+ // invalid sourceResolution -> EMPTY
+ // probably edges of the projections when no extent is defined
+ this.state = ol.TileState.EMPTY;
+ return;
+ }
+
+ var errorThresholdInPixels = opt_errorThreshold !== undefined ?
+ opt_errorThreshold : ol.DEFAULT_RASTER_REPROJECTION_ERROR_THRESHOLD;
+
+ /**
+ * @private
+ * @type {!ol.reproj.Triangulation}
+ */
+ this.triangulation_ = new ol.reproj.Triangulation(
+ sourceProj, targetProj, limitedTargetExtent, maxSourceExtent,
+ sourceResolution * errorThresholdInPixels);
+
+ if (this.triangulation_.getTriangles().length === 0) {
+ // no valid triangles -> EMPTY
+ this.state = ol.TileState.EMPTY;
+ return;
+ }
+
+ this.sourceZ_ = sourceTileGrid.getZForResolution(sourceResolution);
+ var sourceExtent = this.triangulation_.calculateSourceExtent();
+
+ if (maxSourceExtent) {
+ if (sourceProj.canWrapX()) {
+ sourceExtent[1] = ol.math.clamp(
+ sourceExtent[1], maxSourceExtent[1], maxSourceExtent[3]);
+ sourceExtent[3] = ol.math.clamp(
+ sourceExtent[3], maxSourceExtent[1], maxSourceExtent[3]);
+ } else {
+ sourceExtent = ol.extent.getIntersection(sourceExtent, maxSourceExtent);
+ }
+ }
+
+ if (!ol.extent.getArea(sourceExtent)) {
+ this.state = ol.TileState.EMPTY;
+ } else {
+ var sourceRange = sourceTileGrid.getTileRangeForExtentAndZ(
+ sourceExtent, this.sourceZ_);
+
+ var tilesRequired = sourceRange.getWidth() * sourceRange.getHeight();
+ if (!goog.asserts.assert(
+ tilesRequired < ol.RASTER_REPROJECTION_MAX_SOURCE_TILES,
+ 'reasonable number of tiles is required')) {
+ this.state = ol.TileState.ERROR;
+ return;
+ }
+ for (var srcX = sourceRange.minX; srcX <= sourceRange.maxX; srcX++) {
+ for (var srcY = sourceRange.minY; srcY <= sourceRange.maxY; srcY++) {
+ var tile = getTileFunction(this.sourceZ_, srcX, srcY, pixelRatio);
+ if (tile) {
+ this.sourceTiles_.push(tile);
+ }
+ }
+ }
+
+ if (this.sourceTiles_.length === 0) {
+ this.state = ol.TileState.EMPTY;
+ }
+ }
+};
+goog.inherits(ol.reproj.Tile, ol.Tile);
+
+
+/**
+ * @inheritDoc
+ */
+ol.reproj.Tile.prototype.disposeInternal = function() {
+ if (this.state == ol.TileState.LOADING) {
+ this.unlistenSources_();
+ }
+ goog.base(this, 'disposeInternal');
+};
+
+
+/**
+ * @inheritDoc
+ */
+ol.reproj.Tile.prototype.getImage = function(opt_context) {
+ if (opt_context !== undefined) {
+ var image;
+ var key = goog.getUid(opt_context);
+ if (key in this.canvasByContext_) {
+ return this.canvasByContext_[key];
+ } else if (goog.object.isEmpty(this.canvasByContext_)) {
+ image = this.canvas_;
+ } else {
+ image = /** @type {HTMLCanvasElement} */ (this.canvas_.cloneNode(false));
+ }
+ this.canvasByContext_[key] = image;
+ return image;
+ } else {
+ return this.canvas_;
+ }
+};
+
+
+/**
+ * @private
+ */
+ol.reproj.Tile.prototype.reproject_ = function() {
+ var sources = [];
+ this.sourceTiles_.forEach(function(tile, i, arr) {
+ if (tile && tile.getState() == ol.TileState.LOADED) {
+ sources.push({
+ extent: this.sourceTileGrid_.getTileCoordExtent(tile.tileCoord),
+ image: tile.getImage()
+ });
+ }
+ }, this);
+ this.sourceTiles_.length = 0;
+
+ var tileCoord = this.getTileCoord();
+ var z = tileCoord[0];
+ var size = this.targetTileGrid_.getTileSize(z);
+ var width = goog.isNumber(size) ? size : size[0];
+ var height = goog.isNumber(size) ? size : size[1];
+ var targetResolution = this.targetTileGrid_.getResolution(z);
+ var sourceResolution = this.sourceTileGrid_.getResolution(this.sourceZ_);
+
+ var targetExtent = this.targetTileGrid_.getTileCoordExtent(tileCoord);
+ this.canvas_ = ol.reproj.render(width, height, this.pixelRatio_,
+ sourceResolution, this.sourceTileGrid_.getExtent(),
+ targetResolution, targetExtent, this.triangulation_, sources,
+ this.renderEdges_);
+
+ this.state = ol.TileState.LOADED;
+ this.changed();
+};
+
+
+/**
+ * @inheritDoc
+ */
+ol.reproj.Tile.prototype.load = function() {
+ if (this.state == ol.TileState.IDLE) {
+ this.state = ol.TileState.LOADING;
+ this.changed();
+
+ var leftToLoad = 0;
+
+ goog.asserts.assert(!this.sourcesListenerKeys_,
+ 'this.sourcesListenerKeys_ should be null');
+
+ this.sourcesListenerKeys_ = [];
+ this.sourceTiles_.forEach(function(tile, i, arr) {
+ var state = tile.getState();
+ if (state == ol.TileState.IDLE || state == ol.TileState.LOADING) {
+ leftToLoad++;
+
+ var sourceListenKey;
+ sourceListenKey = tile.listen(goog.events.EventType.CHANGE,
+ function(e) {
+ var state = tile.getState();
+ if (state == ol.TileState.LOADED ||
+ state == ol.TileState.ERROR ||
+ state == ol.TileState.EMPTY) {
+ goog.events.unlistenByKey(sourceListenKey);
+ leftToLoad--;
+ goog.asserts.assert(leftToLoad >= 0,
+ 'leftToLoad should not be negative');
+ if (leftToLoad === 0) {
+ this.unlistenSources_();
+ this.reproject_();
+ }
+ }
+ }, false, this);
+ this.sourcesListenerKeys_.push(sourceListenKey);
+ }
+ }, this);
+
+ this.sourceTiles_.forEach(function(tile, i, arr) {
+ var state = tile.getState();
+ if (state == ol.TileState.IDLE) {
+ tile.load();
+ }
+ });
+
+ if (leftToLoad === 0) {
+ this.reproject_();
+ }
+ }
+};
+
+
+/**
+ * @private
+ */
+ol.reproj.Tile.prototype.unlistenSources_ = function() {
+ goog.asserts.assert(this.sourcesListenerKeys_,
+ 'this.sourcesListenerKeys_ should not be null');
+ this.sourcesListenerKeys_.forEach(goog.events.unlistenByKey);
+ this.sourcesListenerKeys_ = null;
+};
diff --git a/src/ol/reproj/triangulation.js b/src/ol/reproj/triangulation.js
new file mode 100644
index 0000000000..571c676c6a
--- /dev/null
+++ b/src/ol/reproj/triangulation.js
@@ -0,0 +1,341 @@
+goog.provide('ol.reproj.Triangulation');
+
+goog.require('goog.asserts');
+goog.require('goog.math');
+goog.require('ol.extent');
+goog.require('ol.proj');
+
+
+/**
+ * Single triangle; consists of 3 source points and 3 target points.
+ *
+ * @typedef {{source: Array.,
+ * target: Array.}}
+ */
+ol.reproj.Triangle;
+
+
+
+/**
+ * @classdesc
+ * Class containing triangulation of the given target extent.
+ * Used for determining source data and the reprojection itself.
+ *
+ * @param {ol.proj.Projection} sourceProj Source projection.
+ * @param {ol.proj.Projection} targetProj Target projection.
+ * @param {ol.Extent} targetExtent Target extent to triangulate.
+ * @param {ol.Extent} maxSourceExtent Maximal source extent that can be used.
+ * @param {number} errorThreshold Acceptable error (in source units).
+ * @constructor
+ */
+ol.reproj.Triangulation = function(sourceProj, targetProj, targetExtent,
+ maxSourceExtent, errorThreshold) {
+
+ /**
+ * @type {ol.proj.Projection}
+ * @private
+ */
+ this.sourceProj_ = sourceProj;
+
+ /**
+ * @type {ol.proj.Projection}
+ * @private
+ */
+ this.targetProj_ = targetProj;
+
+ /** @type {!Object.} */
+ var transformInvCache = {};
+ var transformInv = ol.proj.getTransform(this.targetProj_, this.sourceProj_);
+
+ /**
+ * @param {ol.Coordinate} c
+ * @return {ol.Coordinate}
+ * @private
+ */
+ this.transformInv_ = function(c) {
+ var key = c[0] + '/' + c[1];
+ if (!transformInvCache[key]) {
+ transformInvCache[key] = transformInv(c);
+ }
+ return transformInvCache[key];
+ };
+
+ /**
+ * @type {ol.Extent}
+ * @private
+ */
+ this.maxSourceExtent_ = maxSourceExtent;
+
+ /**
+ * @type {number}
+ * @private
+ */
+ this.errorThresholdSquared_ = errorThreshold * errorThreshold;
+
+ /**
+ * @type {Array.}
+ * @private
+ */
+ this.triangles_ = [];
+
+ /**
+ * Indicates that the triangulation crosses edge of the source projection.
+ * @type {boolean}
+ * @private
+ */
+ this.wrapsXInSource_ = false;
+
+ /**
+ * @type {boolean}
+ * @private
+ */
+ this.canWrapXInSource_ = this.sourceProj_.canWrapX() &&
+ !!maxSourceExtent &&
+ !!this.sourceProj_.getExtent() &&
+ (ol.extent.getWidth(maxSourceExtent) ==
+ ol.extent.getWidth(this.sourceProj_.getExtent()));
+
+ /**
+ * @type {?number}
+ * @private
+ */
+ this.sourceWorldWidth_ = this.sourceProj_.getExtent() ?
+ ol.extent.getWidth(this.sourceProj_.getExtent()) : null;
+
+ /**
+ * @type {?number}
+ * @private
+ */
+ this.targetWorldWidth_ = this.targetProj_.getExtent() ?
+ ol.extent.getWidth(this.targetProj_.getExtent()) : null;
+
+ var destinationTopLeft = ol.extent.getTopLeft(targetExtent);
+ var destinationTopRight = ol.extent.getTopRight(targetExtent);
+ var destinationBottomRight = ol.extent.getBottomRight(targetExtent);
+ var destinationBottomLeft = ol.extent.getBottomLeft(targetExtent);
+ var sourceTopLeft = this.transformInv_(destinationTopLeft);
+ var sourceTopRight = this.transformInv_(destinationTopRight);
+ var sourceBottomRight = this.transformInv_(destinationBottomRight);
+ var sourceBottomLeft = this.transformInv_(destinationBottomLeft);
+
+ this.addQuad_(
+ destinationTopLeft, destinationTopRight,
+ destinationBottomRight, destinationBottomLeft,
+ sourceTopLeft, sourceTopRight, sourceBottomRight, sourceBottomLeft,
+ ol.RASTER_REPROJECTION_MAX_SUBDIVISION);
+
+ if (this.wrapsXInSource_) {
+ // Fix coordinates (ol.proj returns wrapped coordinates, "unwrap" here).
+ // This significantly simplifies the rest of the reprojection process.
+
+ goog.asserts.assert(this.sourceWorldWidth_ !== null);
+ var leftBound = Infinity;
+ this.triangles_.forEach(function(triangle, i, arr) {
+ leftBound = Math.min(leftBound,
+ triangle.source[0][0], triangle.source[1][0], triangle.source[2][0]);
+ });
+
+ // Shift triangles to be as close to `leftBound` as possible
+ // (if the distance is more than `worldWidth / 2` it can be closer.
+ this.triangles_.forEach(function(triangle) {
+ if (Math.max(triangle.source[0][0], triangle.source[1][0],
+ triangle.source[2][0]) - leftBound > this.sourceWorldWidth_ / 2) {
+ var newTriangle = [[triangle.source[0][0], triangle.source[0][1]],
+ [triangle.source[1][0], triangle.source[1][1]],
+ [triangle.source[2][0], triangle.source[2][1]]];
+ if ((newTriangle[0][0] - leftBound) > this.sourceWorldWidth_ / 2) {
+ newTriangle[0][0] -= this.sourceWorldWidth_;
+ }
+ if ((newTriangle[1][0] - leftBound) > this.sourceWorldWidth_ / 2) {
+ newTriangle[1][0] -= this.sourceWorldWidth_;
+ }
+ if ((newTriangle[2][0] - leftBound) > this.sourceWorldWidth_ / 2) {
+ newTriangle[2][0] -= this.sourceWorldWidth_;
+ }
+
+ // Rarely (if the extent contains both the dateline and prime meridian)
+ // the shift can in turn break some triangles.
+ // Detect this here and don't shift in such cases.
+ var minX = Math.min(
+ newTriangle[0][0], newTriangle[1][0], newTriangle[2][0]);
+ var maxX = Math.max(
+ newTriangle[0][0], newTriangle[1][0], newTriangle[2][0]);
+ if ((maxX - minX) < this.sourceWorldWidth_ / 2) {
+ triangle.source = newTriangle;
+ }
+ }
+ }, this);
+ }
+
+ transformInvCache = {};
+};
+
+
+/**
+ * Adds triangle to the triangulation.
+ * @param {ol.Coordinate} a
+ * @param {ol.Coordinate} b
+ * @param {ol.Coordinate} c
+ * @param {ol.Coordinate} aSrc
+ * @param {ol.Coordinate} bSrc
+ * @param {ol.Coordinate} cSrc
+ * @private
+ */
+ol.reproj.Triangulation.prototype.addTriangle_ = function(a, b, c,
+ aSrc, bSrc, cSrc) {
+ this.triangles_.push({
+ source: [aSrc, bSrc, cSrc],
+ target: [a, b, c]
+ });
+};
+
+
+/**
+ * Adds quad (points in clock-wise order) to the triangulation
+ * (and reprojects the vertices) if valid.
+ * Performs quad subdivision if needed to increase precision.
+ *
+ * @param {ol.Coordinate} a
+ * @param {ol.Coordinate} b
+ * @param {ol.Coordinate} c
+ * @param {ol.Coordinate} d
+ * @param {ol.Coordinate} aSrc
+ * @param {ol.Coordinate} bSrc
+ * @param {ol.Coordinate} cSrc
+ * @param {ol.Coordinate} dSrc
+ * @param {number} maxSubdivision Maximal allowed subdivision of the quad.
+ * @private
+ */
+ol.reproj.Triangulation.prototype.addQuad_ = function(a, b, c, d,
+ aSrc, bSrc, cSrc, dSrc, maxSubdivision) {
+
+ var sourceQuadExtent = ol.extent.boundingExtent([aSrc, bSrc, cSrc, dSrc]);
+ var sourceCoverageX = this.sourceWorldWidth_ ?
+ ol.extent.getWidth(sourceQuadExtent) / this.sourceWorldWidth_ : null;
+
+ // when the quad is wrapped in the source projection
+ // it covers most of the projection extent, but not fully
+ var wrapsX = this.sourceProj_.canWrapX() &&
+ sourceCoverageX > 0.5 && sourceCoverageX < 1;
+
+ var needsSubdivision = false;
+
+ if (maxSubdivision > 0) {
+ if (this.targetProj_.isGlobal() && this.targetWorldWidth_) {
+ var targetQuadExtent = ol.extent.boundingExtent([a, b, c, d]);
+ var targetCoverageX =
+ ol.extent.getWidth(targetQuadExtent) / this.targetWorldWidth_;
+ needsSubdivision |=
+ targetCoverageX > ol.RASTER_REPROJECTION_MAX_TRIANGLE_WIDTH;
+ }
+ if (!wrapsX && this.sourceProj_.isGlobal() && sourceCoverageX) {
+ needsSubdivision |=
+ sourceCoverageX > ol.RASTER_REPROJECTION_MAX_TRIANGLE_WIDTH;
+ }
+ }
+
+ if (!needsSubdivision && this.maxSourceExtent_) {
+ if (!ol.extent.intersects(sourceQuadExtent, this.maxSourceExtent_)) {
+ // whole quad outside source projection extent -> ignore
+ return;
+ }
+ }
+
+ if (!needsSubdivision) {
+ if (!isFinite(aSrc[0]) || !isFinite(aSrc[1]) ||
+ !isFinite(bSrc[0]) || !isFinite(bSrc[1]) ||
+ !isFinite(cSrc[0]) || !isFinite(cSrc[1]) ||
+ !isFinite(dSrc[0]) || !isFinite(dSrc[1])) {
+ if (maxSubdivision > 0) {
+ needsSubdivision = true;
+ } else {
+ return;
+ }
+ }
+ }
+
+ if (maxSubdivision > 0) {
+ if (!needsSubdivision) {
+ var center = [(a[0] + c[0]) / 2, (a[1] + c[1]) / 2];
+ var centerSrc = this.transformInv_(center);
+
+ var dx;
+ if (wrapsX) {
+ goog.asserts.assert(this.sourceWorldWidth_);
+ var centerSrcEstimX =
+ (goog.math.modulo(aSrc[0], this.sourceWorldWidth_) +
+ goog.math.modulo(cSrc[0], this.sourceWorldWidth_)) / 2;
+ dx = centerSrcEstimX -
+ goog.math.modulo(centerSrc[0], this.sourceWorldWidth_);
+ } else {
+ dx = (aSrc[0] + cSrc[0]) / 2 - centerSrc[0];
+ }
+ var dy = (aSrc[1] + cSrc[1]) / 2 - centerSrc[1];
+ var centerSrcErrorSquared = dx * dx + dy * dy;
+ needsSubdivision = centerSrcErrorSquared > this.errorThresholdSquared_;
+ }
+ if (needsSubdivision) {
+ if (Math.abs(a[0] - c[0]) <= Math.abs(a[1] - c[1])) {
+ // split horizontally (top & bottom)
+ var bc = [(b[0] + c[0]) / 2, (b[1] + c[1]) / 2];
+ var bcSrc = this.transformInv_(bc);
+ var da = [(d[0] + a[0]) / 2, (d[1] + a[1]) / 2];
+ var daSrc = this.transformInv_(da);
+
+ this.addQuad_(
+ a, b, bc, da, aSrc, bSrc, bcSrc, daSrc, maxSubdivision - 1);
+ this.addQuad_(
+ da, bc, c, d, daSrc, bcSrc, cSrc, dSrc, maxSubdivision - 1);
+ } else {
+ // split vertically (left & right)
+ var ab = [(a[0] + b[0]) / 2, (a[1] + b[1]) / 2];
+ var abSrc = this.transformInv_(ab);
+ var cd = [(c[0] + d[0]) / 2, (c[1] + d[1]) / 2];
+ var cdSrc = this.transformInv_(cd);
+
+ this.addQuad_(
+ a, ab, cd, d, aSrc, abSrc, cdSrc, dSrc, maxSubdivision - 1);
+ this.addQuad_(
+ ab, b, c, cd, abSrc, bSrc, cSrc, cdSrc, maxSubdivision - 1);
+ }
+ return;
+ }
+ }
+
+ if (wrapsX) {
+ if (!this.canWrapXInSource_) {
+ return;
+ }
+ this.wrapsXInSource_ = true;
+ }
+
+ this.addTriangle_(a, c, d, aSrc, cSrc, dSrc);
+ this.addTriangle_(a, b, c, aSrc, bSrc, cSrc);
+};
+
+
+/**
+ * Calculates extent of the 'source' coordinates from all the triangles.
+ *
+ * @return {ol.Extent} Calculated extent.
+ */
+ol.reproj.Triangulation.prototype.calculateSourceExtent = function() {
+ var extent = ol.extent.createEmpty();
+
+ this.triangles_.forEach(function(triangle, i, arr) {
+ var src = triangle.source;
+ ol.extent.extendCoordinate(extent, src[0]);
+ ol.extent.extendCoordinate(extent, src[1]);
+ ol.extent.extendCoordinate(extent, src[2]);
+ });
+
+ return extent;
+};
+
+
+/**
+ * @return {Array.} Array of the calculated triangles.
+ */
+ol.reproj.Triangulation.prototype.getTriangles = function() {
+ return this.triangles_;
+};
diff --git a/src/ol/source/bingmapssource.js b/src/ol/source/bingmapssource.js
index 998d439442..8c42d1b152 100644
--- a/src/ol/source/bingmapssource.js
+++ b/src/ol/source/bingmapssource.js
@@ -29,6 +29,7 @@ ol.source.BingMaps = function(options) {
crossOrigin: 'anonymous',
opaque: true,
projection: ol.proj.get('EPSG:3857'),
+ reprojectionErrorThreshold: options.reprojectionErrorThreshold,
state: ol.source.State.LOADING,
tileLoadFunction: options.tileLoadFunction,
wrapX: options.wrapX !== undefined ? options.wrapX : true
diff --git a/src/ol/source/imagecanvassource.js b/src/ol/source/imagecanvassource.js
index 8532e8b313..57f3d8b5c3 100644
--- a/src/ol/source/imagecanvassource.js
+++ b/src/ol/source/imagecanvassource.js
@@ -59,7 +59,7 @@ goog.inherits(ol.source.ImageCanvas, ol.source.Image);
/**
* @inheritDoc
*/
-ol.source.ImageCanvas.prototype.getImage =
+ol.source.ImageCanvas.prototype.getImageInternal =
function(extent, resolution, pixelRatio, projection) {
resolution = this.findNearestResolution(resolution);
diff --git a/src/ol/source/imagemapguidesource.js b/src/ol/source/imagemapguidesource.js
index 031becffb9..2ff6a92701 100644
--- a/src/ol/source/imagemapguidesource.js
+++ b/src/ol/source/imagemapguidesource.js
@@ -126,7 +126,7 @@ ol.source.ImageMapGuide.prototype.getParams = function() {
/**
* @inheritDoc
*/
-ol.source.ImageMapGuide.prototype.getImage =
+ol.source.ImageMapGuide.prototype.getImageInternal =
function(extent, resolution, pixelRatio, projection) {
resolution = this.findNearestResolution(resolution);
pixelRatio = this.hidpi_ ? pixelRatio : 1;
diff --git a/src/ol/source/imagesource.js b/src/ol/source/imagesource.js
index 3bd9afe517..a0a4b4cb09 100644
--- a/src/ol/source/imagesource.js
+++ b/src/ol/source/imagesource.js
@@ -5,9 +5,11 @@ goog.require('goog.array');
goog.require('goog.asserts');
goog.require('goog.events.Event');
goog.require('ol.Attribution');
-goog.require('ol.Extent');
goog.require('ol.ImageState');
goog.require('ol.array');
+goog.require('ol.extent');
+goog.require('ol.proj');
+goog.require('ol.reproj.Image');
goog.require('ol.source.Source');
@@ -56,6 +58,20 @@ ol.source.Image = function(options) {
return b - a;
}, true), 'resolutions must be null or sorted in descending order');
+
+ /**
+ * @private
+ * @type {ol.reproj.Image}
+ */
+ this.reprojectedImage_ = null;
+
+
+ /**
+ * @private
+ * @type {number}
+ */
+ this.reprojectedRevision_ = 0;
+
};
goog.inherits(ol.source.Image, ol.source.Source);
@@ -90,7 +106,53 @@ ol.source.Image.prototype.findNearestResolution =
* @param {ol.proj.Projection} projection Projection.
* @return {ol.ImageBase} Single image.
*/
-ol.source.Image.prototype.getImage = goog.abstractMethod;
+ol.source.Image.prototype.getImage =
+ function(extent, resolution, pixelRatio, projection) {
+ var sourceProjection = this.getProjection();
+ if (!ol.ENABLE_RASTER_REPROJECTION ||
+ !sourceProjection ||
+ !projection ||
+ ol.proj.equivalent(sourceProjection, projection)) {
+ if (sourceProjection) {
+ projection = sourceProjection;
+ }
+ return this.getImageInternal(extent, resolution, pixelRatio, projection);
+ } else {
+ if (this.reprojectedImage_) {
+ if (this.reprojectedRevision_ == this.getRevision() &&
+ ol.proj.equivalent(
+ this.reprojectedImage_.getProjection(), projection) &&
+ this.reprojectedImage_.getResolution() == resolution &&
+ this.reprojectedImage_.getPixelRatio() == pixelRatio &&
+ ol.extent.equals(this.reprojectedImage_.getExtent(), extent)) {
+ return this.reprojectedImage_;
+ }
+ this.reprojectedImage_.dispose();
+ this.reprojectedImage_ = null;
+ }
+
+ this.reprojectedImage_ = new ol.reproj.Image(
+ sourceProjection, projection, extent, resolution, pixelRatio,
+ goog.bind(function(extent, resolution, pixelRatio) {
+ return this.getImageInternal(extent, resolution,
+ pixelRatio, sourceProjection);
+ }, this));
+ this.reprojectedRevision_ = this.getRevision();
+
+ return this.reprojectedImage_;
+ }
+};
+
+
+/**
+ * @param {ol.Extent} extent Extent.
+ * @param {number} resolution Resolution.
+ * @param {number} pixelRatio Pixel ratio.
+ * @param {ol.proj.Projection} projection Projection.
+ * @return {ol.ImageBase} Single image.
+ * @protected
+ */
+ol.source.Image.prototype.getImageInternal = goog.abstractMethod;
/**
diff --git a/src/ol/source/imagestaticsource.js b/src/ol/source/imagestaticsource.js
index 8bffee68fe..f6e57ce82d 100644
--- a/src/ol/source/imagestaticsource.js
+++ b/src/ol/source/imagestaticsource.js
@@ -62,7 +62,7 @@ goog.inherits(ol.source.ImageStatic, ol.source.Image);
/**
* @inheritDoc
*/
-ol.source.ImageStatic.prototype.getImage =
+ol.source.ImageStatic.prototype.getImageInternal =
function(extent, resolution, pixelRatio, projection) {
if (ol.extent.intersects(extent, this.image_.getExtent())) {
return this.image_;
diff --git a/src/ol/source/imagewmssource.js b/src/ol/source/imagewmssource.js
index e9b49b95fd..72eabf98fb 100644
--- a/src/ol/source/imagewmssource.js
+++ b/src/ol/source/imagewmssource.js
@@ -185,7 +185,7 @@ ol.source.ImageWMS.prototype.getParams = function() {
/**
* @inheritDoc
*/
-ol.source.ImageWMS.prototype.getImage =
+ol.source.ImageWMS.prototype.getImageInternal =
function(extent, resolution, pixelRatio, projection) {
if (this.url_ === undefined) {
diff --git a/src/ol/source/mapquestsource.js b/src/ol/source/mapquestsource.js
index 69ecc336b7..49e5241b24 100644
--- a/src/ol/source/mapquestsource.js
+++ b/src/ol/source/mapquestsource.js
@@ -40,6 +40,7 @@ ol.source.MapQuest = function(opt_options) {
crossOrigin: 'anonymous',
logo: 'https://developer.mapquest.com/content/osm/mq_logo.png',
maxZoom: layerConfig.maxZoom,
+ reprojectionErrorThreshold: options.reprojectionErrorThreshold,
opaque: true,
tileLoadFunction: options.tileLoadFunction,
url: url
diff --git a/src/ol/source/osmsource.js b/src/ol/source/osmsource.js
index af17595c5e..f00859485e 100644
--- a/src/ol/source/osmsource.js
+++ b/src/ol/source/osmsource.js
@@ -36,6 +36,7 @@ ol.source.OSM = function(opt_options) {
crossOrigin: crossOrigin,
opaque: true,
maxZoom: options.maxZoom !== undefined ? options.maxZoom : 19,
+ reprojectionErrorThreshold: options.reprojectionErrorThreshold,
tileLoadFunction: options.tileLoadFunction,
url: url,
wrapX: options.wrapX
diff --git a/src/ol/source/stamensource.js b/src/ol/source/stamensource.js
index f49f2d8b98..733542835e 100644
--- a/src/ol/source/stamensource.js
+++ b/src/ol/source/stamensource.js
@@ -109,6 +109,7 @@ ol.source.Stamen = function(options) {
// FIXME uncomment the following when tilegrid supports minZoom
//minZoom: providerConfig.minZoom,
opaque: layerConfig.opaque,
+ reprojectionErrorThreshold: options.reprojectionErrorThreshold,
tileLoadFunction: options.tileLoadFunction,
url: url
});
diff --git a/src/ol/source/tilearcgisrestsource.js b/src/ol/source/tilearcgisrestsource.js
index 3a105fe646..76ba2c336e 100644
--- a/src/ol/source/tilearcgisrestsource.js
+++ b/src/ol/source/tilearcgisrestsource.js
@@ -41,6 +41,7 @@ ol.source.TileArcGISRest = function(opt_options) {
crossOrigin: options.crossOrigin,
logo: options.logo,
projection: options.projection,
+ reprojectionErrorThreshold: options.reprojectionErrorThreshold,
tileGrid: options.tileGrid,
tileLoadFunction: options.tileLoadFunction,
tileUrlFunction: goog.bind(this.tileUrlFunction_, this),
diff --git a/src/ol/source/tileimagesource.js b/src/ol/source/tileimagesource.js
index fa92f0671d..964e7d7690 100644
--- a/src/ol/source/tileimagesource.js
+++ b/src/ol/source/tileimagesource.js
@@ -3,12 +3,16 @@ goog.provide('ol.source.TileImage');
goog.require('goog.asserts');
goog.require('goog.events');
goog.require('goog.events.EventType');
+goog.require('goog.object');
goog.require('ol.ImageTile');
+goog.require('ol.TileCache');
goog.require('ol.TileCoord');
goog.require('ol.TileLoadFunctionType');
goog.require('ol.TileState');
goog.require('ol.TileUrlFunction');
goog.require('ol.TileUrlFunctionType');
+goog.require('ol.proj');
+goog.require('ol.reproj.Tile');
goog.require('ol.source.Tile');
goog.require('ol.source.TileEvent');
@@ -69,6 +73,29 @@ ol.source.TileImage = function(options) {
this.tileClass = options.tileClass !== undefined ?
options.tileClass : ol.ImageTile;
+ /**
+ * @protected
+ * @type {Object.}
+ */
+ this.tileCacheForProjection = {};
+
+ /**
+ * @protected
+ * @type {Object.}
+ */
+ this.tileGridForProjection = {};
+
+ /**
+ * @private
+ * @type {number|undefined}
+ */
+ this.reprojectionErrorThreshold_ = options.reprojectionErrorThreshold;
+
+ /**
+ * @private
+ * @type {boolean}
+ */
+ this.renderReprojectionEdges_ = false;
};
goog.inherits(ol.source.TileImage, ol.source.Tile);
@@ -82,11 +109,129 @@ ol.source.TileImage.defaultTileLoadFunction = function(imageTile, src) {
};
+/**
+ * @inheritDoc
+ */
+ol.source.TileImage.prototype.canExpireCache = function() {
+ if (!ol.ENABLE_RASTER_REPROJECTION) {
+ return goog.base(this, 'canExpireCache');
+ }
+ var canExpire = this.tileCache.canExpireCache();
+ if (canExpire) {
+ return true;
+ } else {
+ return goog.object.some(this.tileCacheForProjection, function(tileCache) {
+ return tileCache.canExpireCache();
+ });
+ }
+};
+
+
+/**
+ * @inheritDoc
+ */
+ol.source.TileImage.prototype.expireCache = function(projection, usedTiles) {
+ if (!ol.ENABLE_RASTER_REPROJECTION) {
+ goog.base(this, 'expireCache', projection, usedTiles);
+ return;
+ }
+ var usedTileCache = this.getTileCacheForProjection(projection);
+
+ this.tileCache.expireCache(this.tileCache == usedTileCache ? usedTiles : {});
+ goog.object.forEach(this.tileCacheForProjection, function(tileCache) {
+ tileCache.expireCache(tileCache == usedTileCache ? usedTiles : {});
+ });
+};
+
+
+/**
+ * @inheritDoc
+ */
+ol.source.TileImage.prototype.getTileGridForProjection = function(projection) {
+ if (!ol.ENABLE_RASTER_REPROJECTION) {
+ return goog.base(this, 'getTileGridForProjection', projection);
+ }
+ var thisProj = this.getProjection();
+ if (this.tileGrid &&
+ (!thisProj || ol.proj.equivalent(thisProj, projection))) {
+ return this.tileGrid;
+ } else {
+ var projKey = goog.getUid(projection).toString();
+ if (!(projKey in this.tileGridForProjection)) {
+ this.tileGridForProjection[projKey] =
+ ol.tilegrid.getForProjection(projection);
+ }
+ return this.tileGridForProjection[projKey];
+ }
+};
+
+
+/**
+ * @inheritDoc
+ */
+ol.source.TileImage.prototype.getTileCacheForProjection = function(projection) {
+ if (!ol.ENABLE_RASTER_REPROJECTION) {
+ return goog.base(this, 'getTileCacheForProjection', projection);
+ }
+ var thisProj = this.getProjection();
+ if (!thisProj || ol.proj.equivalent(thisProj, projection)) {
+ return this.tileCache;
+ } else {
+ var projKey = goog.getUid(projection).toString();
+ if (!(projKey in this.tileCacheForProjection)) {
+ this.tileCacheForProjection[projKey] = new ol.TileCache();
+ }
+ return this.tileCacheForProjection[projKey];
+ }
+};
+
+
/**
* @inheritDoc
*/
ol.source.TileImage.prototype.getTile =
function(z, x, y, pixelRatio, projection) {
+ if (!ol.ENABLE_RASTER_REPROJECTION ||
+ !this.getProjection() ||
+ !projection ||
+ ol.proj.equivalent(this.getProjection(), projection)) {
+ return this.getTileInternal(z, x, y, pixelRatio, projection);
+ } else {
+ var cache = this.getTileCacheForProjection(projection);
+ var tileCoordKey = this.getKeyZXY(z, x, y);
+ if (cache.containsKey(tileCoordKey)) {
+ return /** @type {!ol.Tile} */(cache.get(tileCoordKey));
+ } else {
+ var sourceProjection = this.getProjection();
+ var sourceTileGrid = this.getTileGridForProjection(sourceProjection);
+ var targetTileGrid = this.getTileGridForProjection(projection);
+ var tile = new ol.reproj.Tile(
+ sourceProjection, sourceTileGrid,
+ projection, targetTileGrid,
+ z, x, y, this.getTilePixelRatio(),
+ goog.bind(function(z, x, y, pixelRatio) {
+ return this.getTileInternal(z, x, y, pixelRatio, sourceProjection);
+ }, this), this.reprojectionErrorThreshold_,
+ this.renderReprojectionEdges_);
+
+ cache.set(tileCoordKey, tile);
+ return tile;
+ }
+ }
+};
+
+
+/**
+ * @param {number} z Tile coordinate z.
+ * @param {number} x Tile coordinate x.
+ * @param {number} y Tile coordinate y.
+ * @param {number} pixelRatio Pixel ratio.
+ * @param {ol.proj.Projection} projection Projection.
+ * @return {!ol.Tile} Tile.
+ * @protected
+ */
+ol.source.TileImage.prototype.getTileInternal =
+ function(z, x, y, pixelRatio, projection) {
var tileCoordKey = this.getKeyZXY(z, x, y);
if (this.tileCache.containsKey(tileCoordKey)) {
return /** @type {!ol.Tile} */ (this.tileCache.get(tileCoordKey));
@@ -156,6 +301,50 @@ ol.source.TileImage.prototype.handleTileChange_ = function(event) {
};
+/**
+ * Sets whether to render reprojection edges or not (usually for debugging).
+ * @param {boolean} render Render the edges.
+ * @api
+ */
+ol.source.TileImage.prototype.setRenderReprojectionEdges = function(render) {
+ if (!ol.ENABLE_RASTER_REPROJECTION ||
+ this.renderReprojectionEdges_ == render) {
+ return;
+ }
+ this.renderReprojectionEdges_ = render;
+ goog.object.forEach(this.tileCacheForProjection, function(tileCache) {
+ tileCache.clear();
+ });
+ this.changed();
+};
+
+
+/**
+ * Sets the tile grid to use when reprojecting the tiles to the given
+ * projection instead of the default tile grid for the projection.
+ *
+ * This can be useful when the default tile grid cannot be created
+ * (e.g. projection has no extent defined) or
+ * for optimization reasons (custom tile size, resolutions, ...).
+ *
+ * @param {ol.proj.ProjectionLike} projection Projection.
+ * @param {ol.tilegrid.TileGrid} tilegrid Tile grid to use for the projection.
+ * @api
+ */
+ol.source.TileImage.prototype.setTileGridForProjection =
+ function(projection, tilegrid) {
+ if (ol.ENABLE_RASTER_REPROJECTION) {
+ var proj = ol.proj.get(projection);
+ if (proj) {
+ var projKey = goog.getUid(proj).toString();
+ if (!(projKey in this.tileGridForProjection)) {
+ this.tileGridForProjection[projKey] = tilegrid;
+ }
+ }
+ }
+};
+
+
/**
* Set the tile load function of the source.
* @param {ol.TileLoadFunctionType} tileLoadFunction Tile load function.
@@ -163,6 +352,7 @@ ol.source.TileImage.prototype.handleTileChange_ = function(event) {
*/
ol.source.TileImage.prototype.setTileLoadFunction = function(tileLoadFunction) {
this.tileCache.clear();
+ this.tileCacheForProjection = {};
this.tileLoadFunction = tileLoadFunction;
this.changed();
};
@@ -178,6 +368,7 @@ ol.source.TileImage.prototype.setTileUrlFunction = function(tileUrlFunction) {
// FIXME cache. The tile URL function would need to be incorporated into the
// FIXME cache key somehow.
this.tileCache.clear();
+ this.tileCacheForProjection = {};
this.tileUrlFunction = tileUrlFunction;
this.changed();
};
@@ -186,9 +377,10 @@ ol.source.TileImage.prototype.setTileUrlFunction = function(tileUrlFunction) {
/**
* @inheritDoc
*/
-ol.source.TileImage.prototype.useTile = function(z, x, y) {
+ol.source.TileImage.prototype.useTile = function(z, x, y, projection) {
+ var tileCache = this.getTileCacheForProjection(projection);
var tileCoordKey = this.getKeyZXY(z, x, y);
- if (this.tileCache.containsKey(tileCoordKey)) {
- this.tileCache.get(tileCoordKey);
+ if (tileCache && tileCache.containsKey(tileCoordKey)) {
+ tileCache.get(tileCoordKey);
}
};
diff --git a/src/ol/source/tilejsonsource.js b/src/ol/source/tilejsonsource.js
index e1c3f854fc..bf82741004 100644
--- a/src/ol/source/tilejsonsource.js
+++ b/src/ol/source/tilejsonsource.js
@@ -34,6 +34,7 @@ ol.source.TileJSON = function(options) {
attributions: options.attributions,
crossOrigin: options.crossOrigin,
projection: ol.proj.get('EPSG:3857'),
+ reprojectionErrorThreshold: options.reprojectionErrorThreshold,
state: ol.source.State.LOADING,
tileLoadFunction: options.tileLoadFunction,
wrapX: options.wrapX !== undefined ? options.wrapX : true
diff --git a/src/ol/source/tilesource.js b/src/ol/source/tilesource.js
index 111fb6cf18..e47afe0d3f 100644
--- a/src/ol/source/tilesource.js
+++ b/src/ol/source/tilesource.js
@@ -10,6 +10,7 @@ goog.require('ol.Extent');
goog.require('ol.TileCache');
goog.require('ol.TileRange');
goog.require('ol.TileState');
+goog.require('ol.proj');
goog.require('ol.size');
goog.require('ol.source.Source');
goog.require('ol.tilecoord');
@@ -97,14 +98,19 @@ ol.source.Tile.prototype.canExpireCache = function() {
/**
+ * @param {ol.proj.Projection} projection Projection.
* @param {Object.} usedTiles Used tiles.
*/
-ol.source.Tile.prototype.expireCache = function(usedTiles) {
- this.tileCache.expireCache(usedTiles);
+ol.source.Tile.prototype.expireCache = function(projection, usedTiles) {
+ var tileCache = this.getTileCacheForProjection(projection);
+ if (tileCache) {
+ tileCache.expireCache(usedTiles);
+ }
};
/**
+ * @param {ol.proj.Projection} projection Projection.
* @param {number} z Zoom level.
* @param {ol.TileRange} tileRange Tile range.
* @param {function(ol.Tile):(boolean|undefined)} callback Called with each
@@ -112,15 +118,21 @@ ol.source.Tile.prototype.expireCache = function(usedTiles) {
* considered loaded.
* @return {boolean} The tile range is fully covered with loaded tiles.
*/
-ol.source.Tile.prototype.forEachLoadedTile = function(z, tileRange, callback) {
+ol.source.Tile.prototype.forEachLoadedTile =
+ function(projection, z, tileRange, callback) {
+ var tileCache = this.getTileCacheForProjection(projection);
+ if (!tileCache) {
+ return false;
+ }
+
var covered = true;
var tile, tileCoordKey, loaded;
for (var x = tileRange.minX; x <= tileRange.maxX; ++x) {
for (var y = tileRange.minY; y <= tileRange.maxY; ++y) {
tileCoordKey = this.getKeyZXY(z, x, y);
loaded = false;
- if (this.tileCache.containsKey(tileCoordKey)) {
- tile = /** @type {!ol.Tile} */ (this.tileCache.get(tileCoordKey));
+ if (tileCache.containsKey(tileCoordKey)) {
+ tile = /** @type {!ol.Tile} */ (tileCache.get(tileCoordKey));
loaded = tile.getState() === ol.TileState.LOADED;
if (loaded) {
loaded = (callback(tile) !== false);
@@ -174,7 +186,7 @@ ol.source.Tile.prototype.getResolutions = function() {
* @param {number} x Tile coordinate x.
* @param {number} y Tile coordinate y.
* @param {number} pixelRatio Pixel ratio.
- * @param {ol.proj.Projection=} opt_projection Projection.
+ * @param {ol.proj.Projection} projection Projection.
* @return {!ol.Tile} Tile.
*/
ol.source.Tile.prototype.getTile = goog.abstractMethod;
@@ -203,6 +215,29 @@ ol.source.Tile.prototype.getTileGridForProjection = function(projection) {
};
+/**
+ * @param {ol.proj.Projection} projection Projection.
+ * @return {ol.TileCache} Tile cache.
+ * @protected
+ */
+ol.source.Tile.prototype.getTileCacheForProjection = function(projection) {
+ var thisProj = this.getProjection();
+ if (thisProj && !ol.proj.equivalent(thisProj, projection)) {
+ return null;
+ } else {
+ return this.tileCache;
+ }
+};
+
+
+/**
+ * @return {number} Tile pixel ratio.
+ */
+ol.source.Tile.prototype.getTilePixelRatio = function() {
+ return this.tilePixelRatio_;
+};
+
+
/**
* @param {number} z Z.
* @param {number} pixelRatio Pixel ratio.
@@ -244,6 +279,7 @@ ol.source.Tile.prototype.getTileCoordForTileUrlFunction =
* @param {number} z Tile coordinate z.
* @param {number} x Tile coordinate x.
* @param {number} y Tile coordinate y.
+ * @param {ol.proj.Projection} projection Projection.
*/
ol.source.Tile.prototype.useTile = ol.nullFunction;
diff --git a/src/ol/source/tilewmssource.js b/src/ol/source/tilewmssource.js
index e3ad0ed0d3..265b607a2d 100644
--- a/src/ol/source/tilewmssource.js
+++ b/src/ol/source/tilewmssource.js
@@ -45,6 +45,7 @@ ol.source.TileWMS = function(opt_options) {
logo: options.logo,
opaque: !transparent,
projection: options.projection,
+ reprojectionErrorThreshold: options.reprojectionErrorThreshold,
tileGrid: options.tileGrid,
tileLoadFunction: options.tileLoadFunction,
tileUrlFunction: goog.bind(this.tileUrlFunction_, this),
diff --git a/src/ol/source/wmtssource.js b/src/ol/source/wmtssource.js
index c80e745752..854f1062d5 100644
--- a/src/ol/source/wmtssource.js
+++ b/src/ol/source/wmtssource.js
@@ -185,6 +185,7 @@ ol.source.WMTS = function(options) {
crossOrigin: options.crossOrigin,
logo: options.logo,
projection: options.projection,
+ reprojectionErrorThreshold: options.reprojectionErrorThreshold,
tileClass: options.tileClass,
tileGrid: tileGrid,
tileLoadFunction: options.tileLoadFunction,
diff --git a/src/ol/source/xyzsource.js b/src/ol/source/xyzsource.js
index 1863cdb574..17c1f46ed7 100644
--- a/src/ol/source/xyzsource.js
+++ b/src/ol/source/xyzsource.js
@@ -49,6 +49,7 @@ ol.source.XYZ = function(options) {
crossOrigin: options.crossOrigin,
logo: options.logo,
projection: projection,
+ reprojectionErrorThreshold: options.reprojectionErrorThreshold,
tileGrid: tileGrid,
tileLoadFunction: options.tileLoadFunction,
tilePixelRatio: options.tilePixelRatio,
diff --git a/src/ol/source/zoomifysource.js b/src/ol/source/zoomifysource.js
index f9533a74b0..bcddc08af1 100644
--- a/src/ol/source/zoomifysource.js
+++ b/src/ol/source/zoomifysource.js
@@ -124,6 +124,7 @@ ol.source.Zoomify = function(opt_options) {
attributions: options.attributions,
crossOrigin: options.crossOrigin,
logo: options.logo,
+ reprojectionErrorThreshold: options.reprojectionErrorThreshold,
tileClass: ol.source.ZoomifyTile_,
tileGrid: tileGrid,
tileUrlFunction: tileUrlFunction
diff --git a/test/spec/ol/math.test.js b/test/spec/ol/math.test.js
index f299028b8d..ba781f75da 100644
--- a/test/spec/ol/math.test.js
+++ b/test/spec/ol/math.test.js
@@ -91,6 +91,45 @@ describe('ol.math.roundUpToPowerOfTwo', function() {
});
+describe('ol.math.solveLinearSystem', function() {
+ it('calculates correctly', function() {
+ var result = ol.math.solveLinearSystem([
+ [2, 1, 3, 1],
+ [2, 6, 8, 3],
+ [6, 8, 18, 5]
+ ]);
+ expect(result[0]).to.roughlyEqual(0.3, 1e-9);
+ expect(result[1]).to.roughlyEqual(0.4, 1e-9);
+ expect(result[2]).to.roughlyEqual(0, 1e-9);
+ });
+ it('can handle singular matrix', function() {
+ var result = ol.math.solveLinearSystem([
+ [2, 1, 3, 1],
+ [2, 6, 8, 3],
+ [2, 1, 3, 1]
+ ]);
+ expect(result).to.be(null);
+ });
+ it('raises an exception when the matrix is malformed', function() {
+ expect(function() {
+ ol.math.solveLinearSystem([
+ [2, 1, 3, 1],
+ [2, 6, 8, 3],
+ [6, 8, 18]
+ ]);
+ }).to.throwException();
+
+ expect(function() {
+ ol.math.solveLinearSystem([
+ [2, 1, 3, 1],
+ [2, 6, 8, 3],
+ [6, 8, 18, 5, 0]
+ ]);
+ }).to.throwException();
+ });
+});
+
+
describe('ol.math.toDegrees', function() {
it('returns the correct value at -π', function() {
expect(ol.math.toDegrees(-Math.PI)).to.be(-180);
diff --git a/test/spec/ol/reproj/image.test.js b/test/spec/ol/reproj/image.test.js
new file mode 100644
index 0000000000..13d8a8627a
--- /dev/null
+++ b/test/spec/ol/reproj/image.test.js
@@ -0,0 +1,60 @@
+goog.provide('ol.test.reproj.Image');
+
+describe('ol.reproj.Image', function() {
+ function createImage(pixelRatio) {
+ return new ol.reproj.Image(
+ ol.proj.get('EPSG:3857'), ol.proj.get('EPSG:4326'),
+ [-180, -85, 180, 85], 10, pixelRatio,
+ function(extent, resolution, pixelRatio) {
+ return new ol.Image(extent, resolution, pixelRatio, [],
+ 'data:image/gif;base64,' +
+ 'R0lGODlhAQABAIAAAP///wAAACwAAAAAAQABAAACAkQBADs=', null,
+ function(image, src) {
+ image.getImage().src = src;
+ });
+ });
+ }
+
+ it('changes state as expected', function(done) {
+ var image = createImage(1);
+ expect(image.getState()).to.be(ol.ImageState.IDLE);
+ image.listen('change', function() {
+ if (image.getState() == ol.ImageState.LOADED) {
+ done();
+ }
+ });
+ image.load();
+ });
+
+ it('returns correct canvas size', function(done) {
+ var image = createImage(1);
+ image.listen('change', function() {
+ if (image.getState() == ol.ImageState.LOADED) {
+ var canvas = image.getImage();
+ expect(canvas.width).to.be(36);
+ expect(canvas.height).to.be(17);
+ done();
+ }
+ });
+ image.load();
+ });
+
+ it('respects pixelRatio', function(done) {
+ var image = createImage(2);
+ image.listen('change', function() {
+ if (image.getState() == ol.ImageState.LOADED) {
+ var canvas = image.getImage();
+ expect(canvas.width).to.be(72);
+ expect(canvas.height).to.be(34);
+ done();
+ }
+ });
+ image.load();
+ });
+});
+
+
+goog.require('ol.Image');
+goog.require('ol.ImageState');
+goog.require('ol.proj');
+goog.require('ol.reproj.Image');
diff --git a/test/spec/ol/reproj/reproj.test.js b/test/spec/ol/reproj/reproj.test.js
new file mode 100644
index 0000000000..44283a86c1
--- /dev/null
+++ b/test/spec/ol/reproj/reproj.test.js
@@ -0,0 +1,45 @@
+goog.provide('ol.test.reproj');
+
+describe('ol.reproj', function() {
+
+ describe('#calculateSourceResolution', function() {
+ var proj3857 = ol.proj.get('EPSG:3857');
+ var proj4326 = ol.proj.get('EPSG:4326');
+ var origin = [0, 0];
+ var point3857 = [50, 40];
+ var point4326 = ol.proj.transform(point3857, proj3857, proj4326);
+
+ it('is identity for identical projection', function() {
+ var result;
+ var resolution = 500;
+ result = ol.reproj.calculateSourceResolution(
+ proj3857, proj3857, origin, resolution);
+ expect(result).to.be(resolution);
+
+ result = ol.reproj.calculateSourceResolution(
+ proj3857, proj3857, point3857, resolution);
+ expect(result).to.be(resolution);
+
+ result = ol.reproj.calculateSourceResolution(
+ proj4326, proj4326, point4326, resolution);
+ expect(result).to.be(resolution);
+ });
+
+ it('calculates correctly', function() {
+ var resolution4326 = 5;
+
+ var resolution3857 = ol.reproj.calculateSourceResolution(
+ proj3857, proj4326, point4326, resolution4326);
+ expect(resolution3857).not.to.be(resolution4326);
+ expect(resolution3857).to.roughlyEqual(555974.3714343394, 1e-6);
+
+ var result = ol.reproj.calculateSourceResolution(
+ proj4326, proj3857, point3857, resolution3857);
+ expect(result).to.be(resolution4326);
+ });
+ });
+});
+
+
+goog.require('ol.reproj');
+goog.require('ol.proj');
diff --git a/test/spec/ol/reproj/tile.test.js b/test/spec/ol/reproj/tile.test.js
new file mode 100644
index 0000000000..1354b804fb
--- /dev/null
+++ b/test/spec/ol/reproj/tile.test.js
@@ -0,0 +1,100 @@
+goog.provide('ol.test.reproj.Tile');
+
+describe('ol.reproj.Tile', function() {
+ beforeEach(function() {
+ proj4.defs('EPSG:27700', '+proj=tmerc +lat_0=49 +lon_0=-2 ' +
+ '+k=0.9996012717 +x_0=400000 +y_0=-100000 +ellps=airy ' +
+ '+towgs84=446.448,-125.157,542.06,0.15,0.247,0.842,-20.489 ' +
+ '+units=m +no_defs');
+ var proj27700 = ol.proj.get('EPSG:27700');
+ proj27700.setExtent([0, 0, 700000, 1300000]);
+ });
+
+ afterEach(function() {
+ delete proj4.defs['EPSG:27700'];
+ });
+
+
+ function createTile(pixelRatio, opt_tileSize) {
+ var proj4326 = ol.proj.get('EPSG:4326');
+ var proj3857 = ol.proj.get('EPSG:3857');
+ return new ol.reproj.Tile(
+ proj3857, ol.tilegrid.createForProjection(proj3857), proj4326,
+ ol.tilegrid.createForProjection(proj4326, 3, opt_tileSize),
+ 3, 2, -2, pixelRatio, function(z, x, y, pixelRatio) {
+ return new ol.ImageTile([z, x, y], ol.TileState.IDLE,
+ 'data:image/gif;base64,' +
+ 'R0lGODlhAQABAIAAAP///wAAACwAAAAAAQABAAACAkQBADs=', null,
+ function(tile, src) {
+ tile.getImage().src = src;
+ });
+ });
+ }
+
+ it('changes state as expected', function(done) {
+ var tile = createTile(1);
+ expect(tile.getState()).to.be(ol.TileState.IDLE);
+ tile.listen('change', function() {
+ if (tile.getState() == ol.TileState.LOADED) {
+ done();
+ }
+ });
+ tile.load();
+ });
+
+ it('is empty when outside target tile grid', function() {
+ var proj4326 = ol.proj.get('EPSG:4326');
+ var proj3857 = ol.proj.get('EPSG:3857');
+ var tile = new ol.reproj.Tile(
+ proj3857, ol.tilegrid.createForProjection(proj3857),
+ proj4326, ol.tilegrid.createForProjection(proj4326),
+ 0, -1, 0, 1, function() {
+ expect().fail('No tiles should be required');
+ });
+ expect(tile.getState()).to.be(ol.TileState.EMPTY);
+ });
+
+ it('is empty when outside source tile grid', function() {
+ var proj4326 = ol.proj.get('EPSG:4326');
+ var proj27700 = ol.proj.get('EPSG:27700');
+ var tile = new ol.reproj.Tile(
+ proj27700, ol.tilegrid.createForProjection(proj27700),
+ proj4326, ol.tilegrid.createForProjection(proj4326),
+ 3, 2, -2, 1, function() {
+ expect().fail('No tiles should be required');
+ });
+ expect(tile.getState()).to.be(ol.TileState.EMPTY);
+ });
+
+ it('respects tile size of target tile grid', function(done) {
+ var tile = createTile(1, [100, 40]);
+ tile.listen('change', function() {
+ if (tile.getState() == ol.TileState.LOADED) {
+ var canvas = tile.getImage();
+ expect(canvas.width).to.be(100);
+ expect(canvas.height).to.be(40);
+ done();
+ }
+ });
+ tile.load();
+ });
+
+ it('respects pixelRatio', function(done) {
+ var tile = createTile(3, [60, 20]);
+ tile.listen('change', function() {
+ if (tile.getState() == ol.TileState.LOADED) {
+ var canvas = tile.getImage();
+ expect(canvas.width).to.be(180);
+ expect(canvas.height).to.be(60);
+ done();
+ }
+ });
+ tile.load();
+ });
+});
+
+
+goog.require('ol.ImageTile');
+goog.require('ol.TileState');
+goog.require('ol.proj');
+goog.require('ol.reproj.Tile');
diff --git a/test/spec/ol/reproj/triangulation.test.js b/test/spec/ol/reproj/triangulation.test.js
new file mode 100644
index 0000000000..3142498ff9
--- /dev/null
+++ b/test/spec/ol/reproj/triangulation.test.js
@@ -0,0 +1,44 @@
+goog.provide('ol.test.reproj.Triangulation');
+
+describe('ol.reproj.Triangulation', function() {
+ beforeEach(function() {
+ proj4.defs('EPSG:27700', '+proj=tmerc +lat_0=49 +lon_0=-2 ' +
+ '+k=0.9996012717 +x_0=400000 +y_0=-100000 +ellps=airy ' +
+ '+towgs84=446.448,-125.157,542.06,0.15,0.247,0.842,-20.489 ' +
+ '+units=m +no_defs');
+ var proj27700 = ol.proj.get('EPSG:27700');
+ proj27700.setExtent([0, 0, 700000, 1300000]);
+ });
+
+ afterEach(function() {
+ delete proj4.defs['EPSG:27700'];
+ });
+
+ describe('constructor', function() {
+ it('is trivial for identity', function() {
+ var proj4326 = ol.proj.get('EPSG:4326');
+ var triangulation = new ol.reproj.Triangulation(proj4326, proj4326,
+ [20, 20, 30, 30], [-180, -90, 180, 90], 0);
+ expect(triangulation.getTriangles().length).to.be(2);
+ });
+
+ it('is empty when outside source extent', function() {
+ var proj4326 = ol.proj.get('EPSG:4326');
+ var proj27700 = ol.proj.get('EPSG:27700');
+ var triangulation = new ol.reproj.Triangulation(proj27700, proj4326,
+ [0, 0, 10, 10], proj27700.getExtent(), 0);
+ expect(triangulation.getTriangles().length).to.be(0);
+ });
+
+ it('can handle null source extent', function() {
+ var proj4326 = ol.proj.get('EPSG:4326');
+ var triangulation = new ol.reproj.Triangulation(proj4326, proj4326,
+ [20, 20, 30, 30], null, 0);
+ expect(triangulation.getTriangles().length).to.be(2);
+ });
+ });
+});
+
+
+goog.require('ol.proj');
+goog.require('ol.reproj.Triangulation');
diff --git a/test/spec/ol/source/tileimagesource.test.js b/test/spec/ol/source/tileimagesource.test.js
new file mode 100644
index 0000000000..2b90a23717
--- /dev/null
+++ b/test/spec/ol/source/tileimagesource.test.js
@@ -0,0 +1,94 @@
+goog.provide('ol.test.source.TileImageSource');
+
+describe('ol.source.TileImage', function() {
+ function createSource(opt_proj, opt_tileGrid) {
+ var proj = opt_proj || 'EPSG:3857';
+ return new ol.source.TileImage({
+ projection: proj,
+ tileGrid: opt_tileGrid ||
+ ol.tilegrid.createForProjection(proj, undefined, [2, 2]),
+ tileUrlFunction: ol.TileUrlFunction.createFromTemplate(
+ '')
+ });
+ }
+
+ describe('#setTileGridForProjection', function() {
+ it('uses the tilegrid for given projection', function() {
+ var source = createSource();
+ var tileGrid = ol.tilegrid.createForProjection('EPSG:4326', 3, [10, 20]);
+ source.setTileGridForProjection('EPSG:4326', tileGrid);
+ var retrieved = source.getTileGridForProjection(ol.proj.get('EPSG:4326'));
+ expect(retrieved).to.be(tileGrid);
+ });
+ });
+
+ describe('#getTile', function() {
+ it('does not do reprojection for identity', function() {
+ var source3857 = createSource('EPSG:3857');
+ var tile3857 = source3857.getTile(0, 0, -1, 1, ol.proj.get('EPSG:3857'));
+ expect(tile3857).to.be.a(ol.ImageTile);
+ expect(tile3857).not.to.be.a(ol.reproj.Tile);
+
+ var projXXX = new ol.proj.Projection({
+ code: 'XXX',
+ units: 'degrees'
+ });
+ var sourceXXX = createSource(projXXX);
+ var tileXXX = sourceXXX.getTile(0, 0, -1, 1, projXXX);
+ expect(tileXXX).to.be.a(ol.ImageTile);
+ expect(tileXXX).not.to.be.a(ol.reproj.Tile);
+ });
+
+ beforeEach(function() {
+ proj4.defs('4326_noextentnounits', '+proj=longlat +datum=WGS84 +no_defs');
+ });
+
+ afterEach(function() {
+ delete proj4.defs['4326_noextentnounits'];
+ });
+
+ it('can handle source projection without extent and units', function(done) {
+ var source = createSource('4326_noextentnounits', ol.tilegrid.createXYZ({
+ extent: [-180, -90, 180, 90],
+ tileSize: [2, 2]
+ }));
+ var tile = source.getTile(0, 0, -1, 1, ol.proj.get('EPSG:3857'));
+ expect(tile).to.be.a(ol.reproj.Tile);
+
+ tile.listen('change', function() {
+ if (tile.getState() == ol.TileState.LOADED) {
+ done();
+ }
+ });
+ tile.load();
+ });
+
+ it('can handle target projection without extent and units', function(done) {
+ var proj = ol.proj.get('4326_noextentnounits');
+ var source = createSource();
+ source.setTileGridForProjection(proj,
+ ol.tilegrid.createXYZ({
+ extent: [-180, -90, 180, 90],
+ tileSize: [2, 2]
+ }));
+ var tile = source.getTile(0, 0, -1, 1, proj);
+ expect(tile).to.be.a(ol.reproj.Tile);
+
+ tile.listen('change', function() {
+ if (tile.getState() == ol.TileState.LOADED) {
+ done();
+ }
+ });
+ tile.load();
+ });
+ });
+});
+
+goog.require('ol.ImageTile');
+goog.require('ol.Tile');
+goog.require('ol.TileState');
+goog.require('ol.TileUrlFunction');
+goog.require('ol.proj');
+goog.require('ol.proj.Projection');
+goog.require('ol.reproj.Tile');
+goog.require('ol.source.TileImage');
diff --git a/test/spec/ol/source/tilesource.test.js b/test/spec/ol/source/tilesource.test.js
index 59d8fbbc4d..e660f2f520 100644
--- a/test/spec/ol/source/tilesource.test.js
+++ b/test/spec/ol/source/tilesource.test.js
@@ -26,7 +26,7 @@ describe('ol.source.Tile', function() {
var zoom = 3;
var range = grid.getTileRangeForExtentAndZ(extent, zoom);
- source.forEachLoadedTile(zoom, range, callback);
+ source.forEachLoadedTile(source.getProjection(), zoom, range, callback);
expect(callback.callCount).to.be(0);
});
@@ -38,7 +38,7 @@ describe('ol.source.Tile', function() {
var zoom = 3;
var range = grid.getTileRangeForExtentAndZ(extent, zoom);
- source.forEachLoadedTile(zoom, range, callback);
+ source.forEachLoadedTile(source.getProjection(), zoom, range, callback);
expect(source.getTile.callCount).to.be(0);
source.getTile.restore();
});
@@ -55,7 +55,7 @@ describe('ol.source.Tile', function() {
var zoom = 1;
var range = new ol.TileRange(0, 1, 0, 1);
- source.forEachLoadedTile(zoom, range, callback);
+ source.forEachLoadedTile(source.getProjection(), zoom, range, callback);
expect(callback.callCount).to.be(3);
});
@@ -71,9 +71,10 @@ describe('ol.source.Tile', function() {
var zoom = 1;
var range = new ol.TileRange(0, 1, 0, 1);
- var covered = source.forEachLoadedTile(zoom, range, function() {
- return true;
- });
+ var covered = source.forEachLoadedTile(source.getProjection(), zoom,
+ range, function() {
+ return true;
+ });
expect(covered).to.be(true);
});
@@ -89,9 +90,10 @@ describe('ol.source.Tile', function() {
var zoom = 1;
var range = new ol.TileRange(0, 1, 0, 1);
- var covered = source.forEachLoadedTile(zoom, range, function() {
- return true;
- });
+ var covered = source.forEachLoadedTile(source.getProjection(), zoom,
+ range, function() {
+ return true;
+ });
expect(covered).to.be(false);
});
@@ -107,9 +109,10 @@ describe('ol.source.Tile', function() {
var zoom = 1;
var range = new ol.TileRange(0, 1, 0, 1);
- var covered = source.forEachLoadedTile(zoom, range, function() {
- return false;
- });
+ var covered = source.forEachLoadedTile(source.getProjection(), zoom,
+ range, function() {
+ return false;
+ });
expect(covered).to.be(false);
});
diff --git a/test/test-extensions.js b/test/test-extensions.js
index 70b7c2dbdc..7d7551264e 100644
--- a/test/test-extensions.js
+++ b/test/test-extensions.js
@@ -390,6 +390,7 @@
done();
});
};
+ global.resembleCanvas = resembleCanvas;
function expectResembleCanvas(map, referenceImage, tolerance, done) {
map.render();
diff --git a/test_rendering/spec/ol/data/tiles/4326/0/0/0.png b/test_rendering/spec/ol/data/tiles/4326/0/0/0.png
new file mode 100644
index 0000000000..0b99038d5a
Binary files /dev/null and b/test_rendering/spec/ol/data/tiles/4326/0/0/0.png differ
diff --git a/test_rendering/spec/ol/data/tiles/osm-512x256/5/3/12.png b/test_rendering/spec/ol/data/tiles/osm-512x256/5/3/12.png
new file mode 100644
index 0000000000..b25a05ace1
Binary files /dev/null and b/test_rendering/spec/ol/data/tiles/osm-512x256/5/3/12.png differ
diff --git a/test_rendering/spec/ol/data/tiles/osm/5/5/13.png b/test_rendering/spec/ol/data/tiles/osm/5/5/13.png
new file mode 100644
index 0000000000..84a6879b3e
Binary files /dev/null and b/test_rendering/spec/ol/data/tiles/osm/5/5/13.png differ
diff --git a/test_rendering/spec/ol/data/tiles/osm/5/6/12.png b/test_rendering/spec/ol/data/tiles/osm/5/6/12.png
new file mode 100644
index 0000000000..54b877726e
Binary files /dev/null and b/test_rendering/spec/ol/data/tiles/osm/5/6/12.png differ
diff --git a/test_rendering/spec/ol/data/tiles/osm/5/6/13.png b/test_rendering/spec/ol/data/tiles/osm/5/6/13.png
new file mode 100644
index 0000000000..9fe6d7bc55
Binary files /dev/null and b/test_rendering/spec/ol/data/tiles/osm/5/6/13.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/4326-to-3857.png b/test_rendering/spec/ol/reproj/expected/4326-to-3857.png
new file mode 100644
index 0000000000..8374eb0028
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/4326-to-3857.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/512x256-to-64x128.png b/test_rendering/spec/ol/reproj/expected/512x256-to-64x128.png
new file mode 100644
index 0000000000..7823071688
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/512x256-to-64x128.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/dateline-merc-180.png b/test_rendering/spec/ol/reproj/expected/dateline-merc-180.png
new file mode 100644
index 0000000000..eaec26b428
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/dateline-merc-180.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/dateline-pole.png b/test_rendering/spec/ol/reproj/expected/dateline-pole.png
new file mode 100644
index 0000000000..30ff7dcf12
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/dateline-pole.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/image-3857-to-4326.png b/test_rendering/spec/ol/reproj/expected/image-3857-to-4326.png
new file mode 100644
index 0000000000..a32f5c0264
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/image-3857-to-4326.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/image-dateline-merc-180.png b/test_rendering/spec/ol/reproj/expected/image-dateline-merc-180.png
new file mode 100644
index 0000000000..b99cee8343
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/image-dateline-merc-180.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/image-dateline-pole.png b/test_rendering/spec/ol/reproj/expected/image-dateline-pole.png
new file mode 100644
index 0000000000..ee93205125
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/image-dateline-pole.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/osm4326.png b/test_rendering/spec/ol/reproj/expected/osm4326.png
new file mode 100644
index 0000000000..af1a03b9be
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/osm4326.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/osm5070.png b/test_rendering/spec/ol/reproj/expected/osm5070.png
new file mode 100644
index 0000000000..2de3fa8a3e
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/osm5070.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/osm54009.png b/test_rendering/spec/ol/reproj/expected/osm54009.png
new file mode 100644
index 0000000000..655e02eb20
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/osm54009.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/stitch-osm3740.png b/test_rendering/spec/ol/reproj/expected/stitch-osm3740.png
new file mode 100644
index 0000000000..9f67ee4acf
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/stitch-osm3740.png differ
diff --git a/test_rendering/spec/ol/reproj/expected/stitch-osm4326.png b/test_rendering/spec/ol/reproj/expected/stitch-osm4326.png
new file mode 100644
index 0000000000..d0b67e4e0f
Binary files /dev/null and b/test_rendering/spec/ol/reproj/expected/stitch-osm4326.png differ
diff --git a/test_rendering/spec/ol/reproj/image.test.js b/test_rendering/spec/ol/reproj/image.test.js
new file mode 100644
index 0000000000..2413e945dc
--- /dev/null
+++ b/test_rendering/spec/ol/reproj/image.test.js
@@ -0,0 +1,61 @@
+goog.provide('ol.test.rendering.reproj.Image');
+
+describe('ol.rendering.reproj.Image', function() {
+
+ function testSingleImage(source, targetProj,
+ targetExtent, targetResolution, pixelRatio, expectedUrl, done) {
+ var sourceProj = source.getProjection();
+
+ var imagesRequested = 0;
+
+ var image = new ol.reproj.Image(sourceProj, ol.proj.get(targetProj),
+ targetExtent, targetResolution, pixelRatio,
+ function(extent, resolution, pixelRatio) {
+ imagesRequested++;
+ return source.getImage(extent, resolution, pixelRatio, sourceProj);
+ });
+ if (image.getState() == ol.ImageState.IDLE) {
+ image.listen('change', function(e) {
+ if (image.getState() == ol.ImageState.LOADED) {
+ expect(imagesRequested).to.be(1);
+ resembleCanvas(image.getImage(), expectedUrl, IMAGE_TOLERANCE, done);
+ }
+ });
+ image.load();
+ }
+ }
+
+ var source;
+
+ describe('image reprojections from EPSG:3857', function() {
+ beforeEach(function() {
+ source = new ol.source.ImageStatic({
+ url: 'spec/ol/data/tiles/osm/5/5/12.png',
+ imageExtent: ol.tilegrid.createXYZ().getTileCoordExtent([5, 5, -13]),
+ projection: ol.proj.get('EPSG:3857')
+ });
+ });
+
+ it('works for identity reprojection', function(done) {
+ testSingleImage(source, 'EPSG:3857',
+ ol.tilegrid.createXYZ().getTileCoordExtent([5, 5, -13]),
+ 2 * ol.proj.EPSG3857.HALF_SIZE / (256 * (1 << 5)), 1,
+ 'spec/ol/data/tiles/osm/5/5/12.png', done);
+ });
+
+ it('to EPSG:4326', function(done) {
+ testSingleImage(source, 'EPSG:4326',
+ ol.tilegrid.createForProjection('EPSG:4326').
+ getTileCoordExtent([6, 10, -10]),
+ 360 / (256 * (1 << 4)), 1,
+ 'spec/ol/reproj/expected/image-3857-to-4326.png', done);
+ });
+ });
+});
+
+goog.require('ol.extent');
+goog.require('ol.proj');
+goog.require('ol.proj.EPSG3857');
+goog.require('ol.reproj.Image');
+goog.require('ol.source.ImageStatic');
+goog.require('ol.ImageState');
diff --git a/test_rendering/spec/ol/reproj/tile.test.js b/test_rendering/spec/ol/reproj/tile.test.js
new file mode 100644
index 0000000000..2102b2f7f4
--- /dev/null
+++ b/test_rendering/spec/ol/reproj/tile.test.js
@@ -0,0 +1,178 @@
+goog.provide('ol.test.rendering.reproj.Tile');
+
+describe('ol.rendering.reproj.Tile', function() {
+
+ function testSingleTile(source, targetProjection, targetTileGrid, z, x, y,
+ pixelRatio, expectedUrl, expectedRequests, done) {
+ var sourceProjection = source.getProjection();
+
+ var tilesRequested = 0;
+
+ var tile = new ol.reproj.Tile(sourceProjection, source.getTileGrid(),
+ ol.proj.get(targetProjection), targetTileGrid, z, x, y, pixelRatio,
+ function(z, x, y, pixelRatio) {
+ tilesRequested++;
+ return source.getTile(z, x, y, pixelRatio, sourceProjection);
+ });
+ if (tile.getState() == ol.TileState.IDLE) {
+ tile.listen('change', function(e) {
+ if (tile.getState() == ol.TileState.LOADED) {
+ expect(tilesRequested).to.be(expectedRequests);
+ resembleCanvas(tile.getImage(), expectedUrl,
+ IMAGE_TOLERANCE, done);
+ }
+ });
+ tile.load();
+ }
+ }
+
+ var source;
+
+ describe('single tile reprojections from EPSG:3857', function() {
+ beforeEach(function() {
+ source = new ol.source.XYZ({
+ projection: 'EPSG:3857',
+ url: 'spec/ol/data/tiles/osm/{z}/{x}/{y}.png'
+ });
+ });
+
+ it('works for identity reprojection', function(done) {
+ testSingleTile(source, 'EPSG:3857', source.getTileGrid(), 5, 5, -13, 1,
+ 'spec/ol/data/tiles/osm/5/5/12.png', 1, done);
+ });
+
+ it('to EPSG:4326', function(done) {
+ var tileGrid = ol.tilegrid.createForProjection('EPSG:4326', 7, [64, 64]);
+ testSingleTile(source, 'EPSG:4326', tileGrid, 7, 21, -20, 1,
+ 'spec/ol/reproj/expected/osm4326.png', 1, done);
+ });
+
+ it('to EPSG:5070', function(done) {
+ proj4.defs('EPSG:5070',
+ '+proj=aea +lat_1=29.5 +lat_2=45.5 +lat_0=23 +lon_0=-96 +x_0=0 ' +
+ '+y_0=0 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs');
+ var proj5070 = ol.proj.get('EPSG:5070');
+ proj5070.setExtent([-6e6, 0, 4e6, 6e6]);
+
+ var tileGrid = ol.tilegrid.createForProjection('EPSG:5070', 5, [64, 64]);
+ testSingleTile(source, 'EPSG:5070', tileGrid, 5, 13, -15, 1,
+ 'spec/ol/reproj/expected/osm5070.png', 1, done);
+ });
+
+ it('to ESRI:54009', function(done) {
+ proj4.defs('ESRI:54009',
+ '+proj=moll +lon_0=0 +x_0=0 +y_0=0 +datum=WGS84 +units=m +no_defs');
+ var proj54009 = ol.proj.get('ESRI:54009');
+ proj54009.setExtent([-18e6, -9e6, 18e6, 9e6]);
+
+ var tileGrid = ol.tilegrid.createForProjection('ESRI:54009', 7, [64, 64]);
+ testSingleTile(source, 'ESRI:54009', tileGrid, 7, 27, -16, 1,
+ 'spec/ol/reproj/expected/osm54009.png', 1, done);
+ });
+ });
+
+ describe('stitching several tiles from EPSG:3857', function() {
+ beforeEach(function() {
+ source = new ol.source.XYZ({
+ projection: 'EPSG:3857',
+ url: 'spec/ol/data/tiles/osm/{z}/{x}/{y}.png'
+ });
+ });
+
+ it('to EPSG:4326', function(done) {
+ var tileGrid = ol.tilegrid.createForProjection('EPSG:4326', 7, [64, 64]);
+ testSingleTile(source, 'EPSG:4326', tileGrid, 7, 23, -21, 1,
+ 'spec/ol/reproj/expected/stitch-osm4326.png', 2, done);
+ });
+
+ it('to EPSG:3740', function(done) {
+ proj4.defs('EPSG:3740',
+ '+proj=utm +zone=10 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 ' +
+ '+units=m +no_defs');
+ var proj3740 = ol.proj.get('EPSG:3740');
+ proj3740.setExtent([318499.05, 2700792.39, 4359164.89, 7149336.98]);
+
+ var tileGrid = ol.tilegrid.createForProjection('EPSG:3740', 4, [64, 64]);
+ testSingleTile(source, 'EPSG:3740', tileGrid, 4, 4, -13, 1,
+ 'spec/ol/reproj/expected/stitch-osm3740.png', 4, done);
+ });
+ });
+
+ describe('tile projection from EPSG:4326', function() {
+ beforeEach(function() {
+ source = new ol.source.XYZ({
+ projection: 'EPSG:4326',
+ maxZoom: 0,
+ url: 'spec/ol/data/tiles/4326/{z}/{x}/{y}.png'
+ });
+ });
+
+ it('works for identity reprojection', function(done) {
+ testSingleTile(source, 'EPSG:4326', source.getTileGrid(), 0, 0, -1, 1,
+ 'spec/ol/data/tiles/4326/0/0/0.png', 1, done);
+ });
+
+ it('to EPSG:3857', function(done) {
+ var tileGrid = ol.tilegrid.createForProjection('EPSG:3857', 0, [64, 64]);
+ testSingleTile(source, 'EPSG:3857', tileGrid, 0, 0, -1, 1,
+ 'spec/ol/reproj/expected/4326-to-3857.png', 1, done);
+ });
+ });
+
+ describe('non-square source tiles', function() {
+ beforeEach(function() {
+ source = new ol.source.XYZ({
+ projection: 'EPSG:3857',
+ url: 'spec/ol/data/tiles/osm-512x256/{z}/{x}/{y}.png',
+ tileSize: [512, 256]
+ });
+ });
+
+ it('works for identity reprojection', function(done) {
+ testSingleTile(source, 'EPSG:3857', source.getTileGrid(), 5, 3, -13, 1,
+ 'spec/ol/data/tiles/osm-512x256/5/3/12.png', 1, done);
+ });
+
+ it('to 64x128 EPSG:4326', function(done) {
+ var tileGrid = ol.tilegrid.createForProjection('EPSG:4326', 7, [64, 128]);
+ testSingleTile(source, 'EPSG:4326', tileGrid, 7, 27, -10, 1,
+ 'spec/ol/reproj/expected/512x256-to-64x128.png', 1, done);
+ });
+ });
+
+ describe('dateline wrapping', function() {
+ beforeEach(function() {
+ source = new ol.source.XYZ({
+ projection: 'EPSG:4326',
+ maxZoom: 0,
+ url: 'spec/ol/data/tiles/4326/{z}/{x}/{y}.png'
+ });
+ });
+
+ it('wraps X when prime meridian is shifted', function(done) {
+ proj4.defs('merc_180', '+proj=merc +lon_0=180 +units=m +no_defs');
+ var proj_ = ol.proj.get('merc_180');
+ proj_.setExtent([-20026376.39, -20048966.10, 20026376.39, 20048966.10]);
+
+ var tileGrid = ol.tilegrid.createForProjection('merc_180', 0, [64, 64]);
+ testSingleTile(source, 'merc_180', tileGrid, 0, 0, -1, 1,
+ 'spec/ol/reproj/expected/dateline-merc-180.png', 2, done);
+ });
+
+ it('displays north pole correctly (EPSG:3413)', function(done) {
+ proj4.defs('EPSG:3413', '+proj=stere +lat_0=90 +lat_ts=70 +lon_0=-45 ' +
+ '+k=1 +x_0=0 +y_0=0 +datum=WGS84 +units=m +no_defs');
+ var proj3413 = ol.proj.get('EPSG:3413');
+ proj3413.setExtent([-4194304, -4194304, 4194304, 4194304]);
+
+ var tileGrid = ol.tilegrid.createForProjection('EPSG:3413', 0, [64, 64]);
+ testSingleTile(source, 'EPSG:3413', tileGrid, 0, 0, -1, 1,
+ 'spec/ol/reproj/expected/dateline-pole.png', 2, done);
+ });
+ });
+});
+
+goog.require('ol.proj');
+goog.require('ol.reproj.Tile');
+goog.require('ol.source.XYZ');
+goog.require('ol.TileState');