-
-
Save sebastianleonte/617628973f88792cd097941220110233 to your computer and use it in GitHub Desktop.
#!/usr/bin/python | |
# GoogleMapDownloader.py | |
# Created by Hayden Eskriett [http://eskriett.com] | |
# | |
# A script which when given a longitude, latitude and zoom level downloads a | |
# high resolution google map | |
# Find the associated blog post at: http://blog.eskriett.com/2013/07/19/downloading-google-maps/ | |
import urllib.request | |
from PIL import Image | |
import os | |
import math | |
class GoogleMapsLayers: | |
ROADMAP = "v" | |
TERRAIN = "p" | |
ALTERED_ROADMAP = "r" | |
SATELLITE = "s" | |
TERRAIN_ONLY = "t" | |
HYBRID = "y" | |
class GoogleMapDownloader: | |
""" | |
A class which generates high resolution google maps images given | |
a longitude, latitude and zoom level | |
""" | |
def __init__(self, lat, lng, zoom=12, layer=GoogleMapsLayers.ROADMAP): | |
""" | |
GoogleMapDownloader Constructor | |
Args: | |
lat: The latitude of the location required | |
lng: The longitude of the location required | |
zoom: The zoom level of the location required, ranges from 0 - 23 | |
defaults to 12 | |
""" | |
self._lat = lat | |
self._lng = lng | |
self._zoom = zoom | |
self._layer = layer | |
def getXY(self): | |
""" | |
Generates an X,Y tile coordinate based on the latitude, longitude | |
and zoom level | |
Returns: An X,Y tile coordinate | |
""" | |
tile_size = 256 | |
# Use a left shift to get the power of 2 | |
# i.e. a zoom level of 2 will have 2^2 = 4 tiles | |
numTiles = 1 << self._zoom | |
# Find the x_point given the longitude | |
point_x = (tile_size / 2 + self._lng * tile_size / 360.0) * numTiles // tile_size | |
# Convert the latitude to radians and take the sine | |
sin_y = math.sin(self._lat * (math.pi / 180.0)) | |
# Calulate the y coorindate | |
point_y = ((tile_size / 2) + 0.5 * math.log((1 + sin_y) / (1 - sin_y)) * -( | |
tile_size / (2 * math.pi))) * numTiles // tile_size | |
return int(point_x), int(point_y) | |
def generateImage(self, **kwargs): | |
""" | |
Generates an image by stitching a number of google map tiles together. | |
Args: | |
start_x: The top-left x-tile coordinate | |
start_y: The top-left y-tile coordinate | |
tile_width: The number of tiles wide the image should be - | |
defaults to 5 | |
tile_height: The number of tiles high the image should be - | |
defaults to 5 | |
Returns: | |
A high-resolution Goole Map image. | |
""" | |
start_x = kwargs.get('start_x', None) | |
start_y = kwargs.get('start_y', None) | |
tile_width = kwargs.get('tile_width', 5) | |
tile_height = kwargs.get('tile_height', 5) | |
# Check that we have x and y tile coordinates | |
if start_x == None or start_y == None: | |
start_x, start_y = self.getXY() | |
# Determine the size of the image | |
width, height = 256 * tile_width, 256 * tile_height | |
# Create a new image of the size require | |
map_img = Image.new('RGB', (width, height)) | |
for x in range(0, tile_width): | |
for y in range(0, tile_height): | |
url = f'https://mt0.google.com/vt?lyrs={self._layer}&x=' + str(start_x + x) + '&y=' + str(start_y + y) + '&z=' + str( | |
self._zoom) | |
current_tile = str(x) + '-' + str(y) | |
urllib.request.urlretrieve(url, current_tile) | |
im = Image.open(current_tile) | |
map_img.paste(im, (x * 256, y * 256)) | |
os.remove(current_tile) | |
return map_img | |
def main(): | |
# Create a new instance of GoogleMap Downloader | |
gmd = GoogleMapDownloader(51.5171, 0.1062, 13, GoogleMapsLayers.SATELLITE) | |
print("The tile coorindates are {}".format(gmd.getXY())) | |
try: | |
# Get the high resolution image | |
img = gmd.generateImage() | |
except IOError: | |
print("Could not generate the image - try adjusting the zoom level and checking your coordinates") | |
else: | |
# Save the image to disk | |
img.save("high_resolution_image.png") | |
print("The map has successfully been created") | |
if __name__ == '__main__': main() |
The script doesn't seem to center the image at the given lat long, any idea how to make it do this?
I think it can't always center the image at a given point.
Because these data are fixed pictures from google map.
We can only calculate which picture contains the point we give but can't change the data itself.
@sebastianleonte Hi there, I found your adaptation awesome ! it worked for me. I have a question: do you know any way you could turn off labels with this method of getting an image of google map from a url query? do you know any documentation about flags of url query of google maps?
Hi, I took a look at this and I found an answer in Stackoverflow you may like: https://stackoverflow.com/questions/29692737/customizing-google-map-tile-server-url/35266313
Basically there is a key "apistyle" you can use in this case to specify the visibility of the labels, as in this link will show no labels at all:
https://mt0.google.com/vt?x=4102&y=2726&z=13&apistyle=s.t%3A0|s.e%3Al|p.v%3Aoff
In the stackoverflow answer there is detailed what is what, you can hide all '0' in this case, roads is '3'... nothing really meaningful, but there you go.
Okay just to add to this, i found this link for a encoder / something more readable: https://github.com/julienben/gmaps-apistyle-encoder but it is not python.
okay, I went a converted the code into python, here you go:
https://gist.github.com/sebastianleonte/69a5f62220fbf25dca7de86c3b6d23ac
You can go for example to https://mapstyle.withgoogle.com/ or https://snazzymaps.com to create a style and the apply it, like so:
OH MY GOD! you are awesome! that absolutely made the trick! Solve! Thank you very much!!! @sebastian
okay, I went a converted the code into python, here you go:
https://gist.github.com/sebastianleonte/69a5f62220fbf25dca7de86c3b6d23acYou can go for example to https://mapstyle.withgoogle.com/ or https://snazzymaps.com to create a style and the apply it, like so:
This works great but is there an easy way to change it so the screenshot is on the 'satellite' view instead of the map?
Hi @acrete, I have added a new parameter for the class.
So you can simply do:
gmd = GoogleMapDownloader(51.5171, 0.1062, 13, GoogleMapsLayers.SATELLITE)
Or the layer you want!
As a way of thank you for your help guys I want to share this version that works perfectly fine. Enjoy!
https://gist.github.com/Militoarg/e33a843ed4caed3a60561edb69d75f3d
Is there a way to get this to collect all the images at a given zoom level for an entire area?
Why does it always show for me - "Could not generate the image - try adjusting the zoom level and checking your coordinates"?
Hi there. @jackhirsh @Anjali1808
I've made an edited gist to download the image with the center of the given lat and long.
And also, the way of sending download request has changed. So images can be downloaded at any zoom and any size without a single problem in 2022.
https://gist.github.com/farhadinima75/c3e73f8d7d9f8d2e452063e611fb9a8c
Have fun :)
+1 to that.
It would mean changing this part:
Find the x_point given the longitude