-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtile_writer.py
213 lines (170 loc) · 7.5 KB
/
tile_writer.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
# tile_writer.py
# Version 0.3 Oct 2018
# Copyright 2015 Alexander Hajnal
# Generates slippy map tiles from within QGIS
# http://alephnull.net/software/gis/tile_writer.shtml
# See the readme.txt file for usage instructions
# ==============================================================================
# Start user-editable settings
# Minimum zoom level
start_z = 12
# Maximum zoom level
end_z = 17
# How many map tiles to place in each regional tile
# (the actual number of map tiles per regional tile is step * step)
# Higher number speed up rendering (assuming enough RAM is available).
# Lower numbers decrease memory usage but increase rendering time.
step = 16
# Number of extra map tiles to render along each edge of a regional tile
# By rendering extra, unused border tiles we can avoid shifting labels,
# truncated images, and line-drawing inconsistancies at tile boundaries.
# If this value is set to 0 you will encounter rendering issues at tile
# boundaries.
border = 2
# Directory to write the regional images and level subdirectories to
output_path = ''
# Path of shapefile defining the area of interest
# The extent of the shapefile is used to limit rendering to a particular area.
# Note that no clipping is done so at lower zoom levels tiles from outside the
# area of interest will be generated.
area_of_interest = 'border.shp'
# Filename convention to follow
#tile_format = 'google'
tile_format = 'tms'
# End user-editable settings
# ==============================================================================
from qgis.core import *
import qgis.utils
from qgis.utils import iface
from qgis.gui import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
import sys
import os
import os.path
import shutil
import math
from globalmercator import GlobalMercator
def delay( millisecondsToWait ):
dieTime = QTime().currentTime().addMSecs( millisecondsToWait )
while ( QTime.currentTime() < dieTime ):
QCoreApplication.processEvents( QEventLoop().AllEvents, 100 )
borderLayer = QgsVectorLayer(area_of_interest, 'border', 'ogr')
borderRect = borderLayer.extent()
borderCRS = borderLayer.crs()
WGS84 = QgsCoordinateReferenceSystem('EPSG:4326')
mapRenderer = qgis.utils.iface.mapCanvas()
mapRect = QgsCoordinateTransform(borderCRS, WGS84, QgsProject.instance()).transform(borderRect) # WGS84
xStart = mapRect.xMinimum()
xEnd = mapRect.xMaximum()
yStart = mapRect.yMinimum()
yEnd = mapRect.yMaximum()
width = mapRect.width()
height = mapRect.height()
print("xStart:", xStart)
print("xEnd: ", xEnd)
print
print("yStart:", yStart)
print("yEnd: ", yEnd)
print
print("width: ", width)
print("height:", height)
print
for z in range(start_z, end_z+1, 1):
print
print("zoom:%i step:%i border_size:%i" % (z, step, border))
gm = GlobalMercator()
lon = xStart
lat = yStart
mx_min, my_min = gm.LatLonToMeters(lat, lon)
tx_min, ty_min = gm.MetersToTile(mx_min, my_min, z)
print("Min: %i, %i @ %i" % ( tx_min, ty_min, z ))
lon = xEnd
lat = yEnd
mx_max, my_max = gm.LatLonToMeters(lat, lon)
tx_max, ty_max = gm.MetersToTile(mx_max, my_max, z)
print("Max: %i, %i @ %i" % ( tx_max, ty_max, z ))
dirPath = "%s/%i" % (output_path,z)
QDir().mkpath(dirPath)
width = 256 * ( step + border + border )
height = 256 * ( step + border + border )
print
print("While generating the regional tiles, QGIS may appear to have locked up.")
print("This is not the case. Please be patient.")
print
print("Generating regional tiles... (%i x %i tiles -> %i x %i regional tiles)" % (tx_max-tx_min+1, ty_max-ty_min+1, math.ceil((tx_max-tx_min+1.0)/step), math.ceil((ty_max-ty_min+1.0)/step)))
total_regional_tiles = 0
for x in range(tx_min, tx_max+1, step):
for y in range(ty_min, ty_max+1, step):
total_regional_tiles += 1
imagePath = "%s/%i_%i_%i_s%i_b%i.png" % (output_path,z,x,y,step,border)
lat_min, lon_min, ignore, ignore = gm.TileLatLonBounds(x-border,y-border,z)
ignore, ignore, lat_max, lon_max = gm.TileLatLonBounds(x+step+border-1,y+step+border-1,z)
lat_min, lon_min = gm.LatLonToMeters(lat_min, lon_min)
lat_max, lon_max = gm.LatLonToMeters(lat_max, lon_max)
if os.path.isfile(imagePath):
# Regional tile exists, so skip it
sys.stdout.write("o")
else:
image = QImage(width, height, QImage.Format_ARGB32_Premultiplied)
settings = QgsMapSettings()
settings.setOutputDpi(95.0)
settings.setOutputImageFormat(QImage.Format_ARGB32_Premultiplied)
settings.setDestinationCrs(QgsCoordinateReferenceSystem('EPSG:3857'))
settings.setOutputSize(QSize(width, height))
settings.setLayers(mapRenderer.layers())
settings.setFlag(QgsMapSettings.DrawLabeling, True)
settings.setBackgroundColor(QColor(255, 255, 255, 1))
tileRect = QgsRectangle(lat_min, lon_min, lat_max, lon_max)
settings.setExtent(tileRect)
job = QgsMapRendererSequentialJob(settings)
job.start()
job.waitForFinished()
delay(10)
image = job.renderedImage()
image.save(imagePath, "PNG")
sys.stdout.write("*")
sys.stdout.write("\n")
print
print("Splitting regional tiles into TMS tiles...")
h = 256 * (border + border + step)
current_regional_tile = 1
for x in range(tx_min, tx_max+1, step):
for y in range(ty_min, ty_max+1, step):
srcPath = "%s/%i_%i_%i_s%i_b%i.png" % (output_path,z,x,y,step,border)
if os.path.isfile(srcPath):
print
print("Processing regional tile %i of %i: %s" % (current_regional_tile, total_regional_tiles, srcPath ))
current_regional_tile += 1
srcImage = QImage()
srcImage.load(srcPath)
px = 256 * border
for tile_x in range(x, x+step, 1):
iDirPath = "%s/%i/%i" % (output_path,z,tile_x)
QDir().mkpath(iDirPath)
py = 256 * ( border + 1 )
for tile_y in range(y, y+step, 1):
if tile_format == 'tms':
tile_y_tms = (1<<z) - tile_y - 1
dstPath = "%s/%i/%i/%i.png" % (output_path,z,tile_x,tile_y_tms)
else:
dstPath = "%s/%i/%i/%i.png" % (output_path,z,tile_x,tile_y)
if os.path.isfile(dstPath):
# Tile exists, so skip it
sys.stdout.write("o")
else:
dstImage = srcImage.copy(px,h-py,256,256)
dstImage.save(dstPath, "PNG")
sys.stdout.write("*")
py += 256
sys.stdout.write("\n")
px += 256
print
print("All done!")
# Warranty
# ------------------------------------------------------------------------------
# I make no warranty or representation, either express or implied, with respect
# the behavior of this script, its quality, performance, accuracy, merchantability,
# or fitness for a particular purpose. This script is provided 'as is', and
# you, by making use thereof, are assuming the entire risk. That said,
# I hope you this script useful. Have fun!