Sunday, 29 March 2015

extents - How to arrange own tiles without projection in OpenLayers 3?


I have own tiles (.png) of 512x512 pixels and I would like to display them in OpenLayers 3 without projection (for now). Therefore I found a way to use the ZOOMIFY projection but with a "ol.layer.Tile"-Layer. (original source: https://groups.google.com/forum/#!topic/ol3-dev/VVdNXHwiZEk)


So far it works more or less, but I am encountering problems providing the right extent of the layer. I have a map DIV of 1024x512 pixels and strangely for zoom level 0: only the left tile is displayed (the right one is missing).


Zoom level 0 has a resolution of 2 (check the variable resolutions) and therefore I thought that I need an extent of 2048(x) and 1024(y) to display 2 tiles of zoom level 0. [1][2]


My guess is that the provided extent causes the problems.


Any ideas how to improve on this?



Here's a jsfiddle: http://jsfiddle.net/vb53xz20/


[1]tile1: services.arcgisonline.com/arcgis/rest/services/ESRI_Imagery_World_2D/MapServer/tile/0/0/0.png


[2]tile2: services.arcgisonline.com/arcgis/rest/services/ESRI_Imagery_World_2D/MapServer/tile/0/0/1.png


And the code:


tileSize = 512;

projection = new ol.proj.Projection({
code: 'ZOOMIFY',
units: 'pixels',
extent: [0, 0, 1024, 1024]

});

projectionExtent = projection.getExtent();

var maxResolution = ol.extent.getWidth(projectionExtent) / tileSize;
window.resolutions = [];
for (var z = 0; z <= 10; z++) {
resolutions[z] = maxResolution / Math.pow(2, z);

}


urlTemplate = 'http://services.arcgisonline.com/arcgis/rest/services/ESRI_Imagery_World_2D/MapServer/tile/{z}/{y}/{x}';

tileLayer = new ol.layer.Tile({
source: new ol.source.TileImage({
tileUrlFunction: function(tileCoord, pixelRatio, projection) {
tileCoordGlobal = tileCoord;

return urlTemplate
.replace('{z}', (tileCoord[0]).toString())

.replace('{x}', (tileCoord[1]).toString())
.replace('{y}', (((-tileCoord[2])-1)).toString())
;
},
wrapX: true,
projection: projection,
tileGrid: new ol.tilegrid.TileGrid({
origin: ol.extent.getTopLeft(projectionExtent),
resolutions: resolutions,
tileSize: tileSize

}),
}),
extent: projectionExtent
});

view = new ol.View ({
projection: projection,
center: [1024, 512],
extent: projectionExtent,
zoom: 0,

resolutions: resolutions
});

window.map = new ol.Map({
layers: [tileLayer],
target: 'map',
view: view
});

Answer



Alrighty, I finally found out how to set the parameters. The problems were due to the resolution and the extent arrays.



I) The resolution array needed a "z+1" instead of "z":


for (var z = 0; z <= 16; z++) {
resolutions[z] = maxResolution / Math.pow(2, z+1);
}

(btw: the number of z in the for-loop defines the number of available zoom levels)



II) And the extent array some adaptions.


The 4 values of the extent array are: [MinX, MinY, MaxX, MaxY]. We can start with 0s for both min-values. The max values depend on:


a) the number of tiles

b) the resolutions
c) the tilesize


I found that this formula worked well for my purposes:
MaxX(MaxY) = No of tiles x resolution[0] x tilesize


For the example above this would mean:


MaxX = 2 x 2 x 512 = 2048
MaxY = 1 x 2 x 512 = 1024


var minX = 0;
var minY = 0;
var maxX = 2048;

var maxY = 1024;

extent: [minX, minY, maxX, maxY]


III) The center can be set to [MaxX/2, MaxY/2]


center: [maxX/2, maxY/2]


Here is the updated jsfiddle: http://jsfiddle.net/vb53xz20/2/



No comments:

Post a Comment

arcpy - Changing output name when exporting data driven pages to JPG?

Is there a way to save the output JPG, changing the output file name to the page name, instead of page number? I mean changing the script fo...