Skip to content

Instantly share code, notes, and snippets.

@om-henners
Created September 6, 2020 02:11
Show Gist options
  • Select an option

  • Save om-henners/7d39e31a01b08a9ca30e1843d31ab28a to your computer and use it in GitHub Desktop.

Select an option

Save om-henners/7d39e31a01b08a9ca30e1843d31ab28a to your computer and use it in GitHub Desktop.
Quick bit of Python code to calculate a 5km travel buffer given as a talk at PyConlineAU 2020!
Display the source blob
Display the rendered blob
Raw
{
"cells": [
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"# Calculating 5km of travel distance in Python\n",
"\n",
"Henry Walshaw\n",
"\n",
"<img src=\"extras/Twitter_Social_Icon_Circle_Color.svg\" style=\"display:inline;height:48px\" /> @om_henners\n",
"\n",
"<img src=\"extras/gis_stackexchange.svg\" style=\"display:inline;height:48px\" />\n",
" @om_henners"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"A quick content warning - I'll be mentioning COVID-19 and some of the impacts at least here in Melbourne."
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"In Melbourne we have a bit of a COVID-19 problem<sup>1</sup>.\n",
"\n",
"<img src=\"extras/Saturday_morning_degraves_street.jpg\" style=\"height:600px; margin-left: auto;margin-right: auto;\"/>\n",
"\n",
"Image credit: https://www.reddit.com/user/damn-croissants/ July 11, 2020\n",
"\n",
"<sup>1</sup>Of course it's not so bad for everyone. Without the boats Salvatore the fur-seal is having a great time in the Yarra!"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"At least until the 13th in Melbourne we're locked down to being with 5km of our homes.\n",
"\n",
"<img src=\"extras/from_the_government_website.png\" style=\"height:600px;margin-left: auto;margin-right: auto;\"/>\n",
"\n",
"Calculated at https://www.vic.gov.au/coronavirus-5km-from-home-map"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"Because I love maps<sup>2</sup> on hearing the news about the lockdown and the 5km I wanted to see what it looked like for myself, and compare the \"as the crow flies\" distance to how far I could get by travelling 5km instead.\n",
"\n",
"<sup>2</sup>And let's be honest, I'm a pedant about spatial data"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"First, there's a few imports to get out of the way<sup>3</sup>"
]
},
{
"cell_type": "code",
"execution_count": 1,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [],
"source": [
"import alphashape\n",
"from descartes import PolygonPatch\n",
"import folium\n",
"import geopandas as gpd\n",
"from geopy.geocoders import Nominatim\n",
"import matplotlib.pyplot as plt\n",
"import networkx as nx\n",
"import numpy as np\n",
"import osmnx as ox\n",
"import pandas as pd\n",
"from shapely import geometry"
]
},
{
"cell_type": "code",
"execution_count": 2,
"metadata": {
"slideshow": {
"slide_type": "skip"
}
},
"outputs": [],
"source": [
"%config InlineBackend.figure_format = 'retina'\n",
"plt.rcParams['figure.figsize'] = (10, 10)"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"<sup>3</sup>Unfortunately spatial libraries in Python can be a bit of a pain to install - feel free to `@` me later if you'd like a hand."
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Step 1: geocoding an address\n",
"\n",
"Geocoding is the process of converting an address into a geographic coordinate (latitude and longitude). For that `geopy` is great and wraps up a bunch of different geocoding APIs consistently. I'll use the Nominatim API as the service, and choose my local train station as my starting point."
]
},
{
"cell_type": "code",
"execution_count": 3,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"data": {
"text/plain": [
"Location(Footscray, Hyde Street, Footscray, City of Maribyrnong, Victoria, 3011, Australia, (-37.8015202, 144.9025869, 0.0))"
]
},
"execution_count": 3,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"address = 'Footscray Railway Station Victoria, 3011, Australia'\n",
"geocoder = Nominatim(user_agent='5km buffer calculator')\n",
"location = geocoder.geocode(address)\n",
"location"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "subslide"
}
},
"source": [
"As a side note, I can use this with the `folium`<sup>4</sup> library to embed a quick interactive map in my notebook so I can see the address is in the place that I expect:"
]
},
{
"cell_type": "code",
"execution_count": 4,
"metadata": {
"scrolled": false,
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"data": {
"text/html": [
"<div style=\"width:100%;\"><div style=\"position:relative;width:100%;height:0;padding-bottom:60%;\"><iframe src=\"data:text/html;charset=utf-8;base64,PCFET0NUWVBFIGh0bWw+CjxoZWFkPiAgICAKICAgIDxtZXRhIGh0dHAtZXF1aXY9ImNvbnRlbnQtdHlwZSIgY29udGVudD0idGV4dC9odG1sOyBjaGFyc2V0PVVURi04IiAvPgogICAgCiAgICAgICAgPHNjcmlwdD4KICAgICAgICAgICAgTF9OT19UT1VDSCA9IGZhbHNlOwogICAgICAgICAgICBMX0RJU0FCTEVfM0QgPSBmYWxzZTsKICAgICAgICA8L3NjcmlwdD4KICAgIAogICAgPHNjcmlwdCBzcmM9Imh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vbGVhZmxldEAxLjYuMC9kaXN0L2xlYWZsZXQuanMiPjwvc2NyaXB0PgogICAgPHNjcmlwdCBzcmM9Imh0dHBzOi8vY29kZS5qcXVlcnkuY29tL2pxdWVyeS0xLjEyLjQubWluLmpzIj48L3NjcmlwdD4KICAgIDxzY3JpcHQgc3JjPSJodHRwczovL21heGNkbi5ib290c3RyYXBjZG4uY29tL2Jvb3RzdHJhcC8zLjIuMC9qcy9ib290c3RyYXAubWluLmpzIj48L3NjcmlwdD4KICAgIDxzY3JpcHQgc3JjPSJodHRwczovL2NkbmpzLmNsb3VkZmxhcmUuY29tL2FqYXgvbGlicy9MZWFmbGV0LmF3ZXNvbWUtbWFya2Vycy8yLjAuMi9sZWFmbGV0LmF3ZXNvbWUtbWFya2Vycy5qcyI+PC9zY3JpcHQ+CiAgICA8bGluayByZWw9InN0eWxlc2hlZXQiIGhyZWY9Imh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vbGVhZmxldEAxLjYuMC9kaXN0L2xlYWZsZXQuY3NzIi8+CiAgICA8bGluayByZWw9InN0eWxlc2hlZXQiIGhyZWY9Imh0dHBzOi8vbWF4Y2RuLmJvb3RzdHJhcGNkbi5jb20vYm9vdHN0cmFwLzMuMi4wL2Nzcy9ib290c3RyYXAubWluLmNzcyIvPgogICAgPGxpbmsgcmVsPSJzdHlsZXNoZWV0IiBocmVmPSJodHRwczovL21heGNkbi5ib290c3RyYXBjZG4uY29tL2Jvb3RzdHJhcC8zLjIuMC9jc3MvYm9vdHN0cmFwLXRoZW1lLm1pbi5jc3MiLz4KICAgIDxsaW5rIHJlbD0ic3R5bGVzaGVldCIgaHJlZj0iaHR0cHM6Ly9tYXhjZG4uYm9vdHN0cmFwY2RuLmNvbS9mb250LWF3ZXNvbWUvNC42LjMvY3NzL2ZvbnQtYXdlc29tZS5taW4uY3NzIi8+CiAgICA8bGluayByZWw9InN0eWxlc2hlZXQiIGhyZWY9Imh0dHBzOi8vY2RuanMuY2xvdWRmbGFyZS5jb20vYWpheC9saWJzL0xlYWZsZXQuYXdlc29tZS1tYXJrZXJzLzIuMC4yL2xlYWZsZXQuYXdlc29tZS1tYXJrZXJzLmNzcyIvPgogICAgPGxpbmsgcmVsPSJzdHlsZXNoZWV0IiBocmVmPSJodHRwczovL3Jhd2Nkbi5naXRoYWNrLmNvbS9weXRob24tdmlzdWFsaXphdGlvbi9mb2xpdW0vbWFzdGVyL2ZvbGl1bS90ZW1wbGF0ZXMvbGVhZmxldC5hd2Vzb21lLnJvdGF0ZS5jc3MiLz4KICAgIDxzdHlsZT5odG1sLCBib2R5IHt3aWR0aDogMTAwJTtoZWlnaHQ6IDEwMCU7bWFyZ2luOiAwO3BhZGRpbmc6IDA7fTwvc3R5bGU+CiAgICA8c3R5bGU+I21hcCB7cG9zaXRpb246YWJzb2x1dGU7dG9wOjA7Ym90dG9tOjA7cmlnaHQ6MDtsZWZ0OjA7fTwvc3R5bGU+CiAgICAKICAgICAgICAgICAgPG1ldGEgbmFtZT0idmlld3BvcnQiIGNvbnRlbnQ9IndpZHRoPWRldmljZS13aWR0aCwKICAgICAgICAgICAgICAgIGluaXRpYWwtc2NhbGU9MS4wLCBtYXhpbXVtLXNjYWxlPTEuMCwgdXNlci1zY2FsYWJsZT1ubyIgLz4KICAgICAgICAgICAgPHN0eWxlPgogICAgICAgICAgICAgICAgI21hcF8xMzczMzJhNjhmZmI0NGIzOTgwYTZlNTkyYjAzOWMzYSB7CiAgICAgICAgICAgICAgICAgICAgcG9zaXRpb246IHJlbGF0aXZlOwogICAgICAgICAgICAgICAgICAgIHdpZHRoOiAxMDAuMCU7CiAgICAgICAgICAgICAgICAgICAgaGVpZ2h0OiA2MDAuMHB4OwogICAgICAgICAgICAgICAgICAgIGxlZnQ6IDAuMCU7CiAgICAgICAgICAgICAgICAgICAgdG9wOiAwLjAlOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICA8L3N0eWxlPgogICAgICAgIAo8L2hlYWQ+Cjxib2R5PiAgICAKICAgIAogICAgICAgICAgICA8ZGl2IGNsYXNzPSJmb2xpdW0tbWFwIiBpZD0ibWFwXzEzNzMzMmE2OGZmYjQ0YjM5ODBhNmU1OTJiMDM5YzNhIiA+PC9kaXY+CiAgICAgICAgCjwvYm9keT4KPHNjcmlwdD4gICAgCiAgICAKICAgICAgICAgICAgdmFyIG1hcF8xMzczMzJhNjhmZmI0NGIzOTgwYTZlNTkyYjAzOWMzYSA9IEwubWFwKAogICAgICAgICAgICAgICAgIm1hcF8xMzczMzJhNjhmZmI0NGIzOTgwYTZlNTkyYjAzOWMzYSIsCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgY2VudGVyOiBbLTM3LjgwMTUyMDIsIDE0NC45MDI1ODY5XSwKICAgICAgICAgICAgICAgICAgICBjcnM6IEwuQ1JTLkVQU0czODU3LAogICAgICAgICAgICAgICAgICAgIHpvb206IDE2LAogICAgICAgICAgICAgICAgICAgIHpvb21Db250cm9sOiB0cnVlLAogICAgICAgICAgICAgICAgICAgIHByZWZlckNhbnZhczogZmFsc2UsCiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICk7CgogICAgICAgICAgICAKCiAgICAgICAgCiAgICAKICAgICAgICAgICAgdmFyIHRpbGVfbGF5ZXJfNzZlYzhjMTQzODI1NGI4MmI1ZDY1NjQ3MjcxMGY2MGQgPSBMLnRpbGVMYXllcigKICAgICAgICAgICAgICAgICJodHRwczovL3tzfS50aWxlLm9wZW5zdHJlZXRtYXAub3JnL3t6fS97eH0ve3l9LnBuZyIsCiAgICAgICAgICAgICAgICB7ImF0dHJpYnV0aW9uIjogIkRhdGEgYnkgXHUwMDI2Y29weTsgXHUwMDNjYSBocmVmPVwiaHR0cDovL29wZW5zdHJlZXRtYXAub3JnXCJcdTAwM2VPcGVuU3RyZWV0TWFwXHUwMDNjL2FcdTAwM2UsIHVuZGVyIFx1MDAzY2EgaHJlZj1cImh0dHA6Ly93d3cub3BlbnN0cmVldG1hcC5vcmcvY29weXJpZ2h0XCJcdTAwM2VPRGJMXHUwMDNjL2FcdTAwM2UuIiwgImRldGVjdFJldGluYSI6IGZhbHNlLCAibWF4TmF0aXZlWm9vbSI6IDIwLCAibWF4Wm9vbSI6IDIwLCAibWluWm9vbSI6IDAsICJub1dyYXAiOiBmYWxzZSwgIm9wYWNpdHkiOiAxLCAic3ViZG9tYWlucyI6ICJhYmMiLCAidG1zIjogZmFsc2V9CiAgICAgICAgICAgICkuYWRkVG8obWFwXzEzNzMzMmE2OGZmYjQ0YjM5ODBhNmU1OTJiMDM5YzNhKTsKICAgICAgICAKICAgIAogICAgICAgICAgICB2YXIgbWFya2VyXzQ2MTJjMzEyZGNhOTQwMmE4ZWIzZTlmMTUyNDc4ZTgxID0gTC5tYXJrZXIoCiAgICAgICAgICAgICAgICBbLTM3LjgwMTUyMDIsIDE0NC45MDI1ODY5XSwKICAgICAgICAgICAgICAgIHt9CiAgICAgICAgICAgICkuYWRkVG8obWFwXzEzNzMzMmE2OGZmYjQ0YjM5ODBhNmU1OTJiMDM5YzNhKTsKICAgICAgICAKICAgIAogICAgICAgIHZhciBwb3B1cF9mNTlhZTFlM2Q0ODQ0ZTJjYjQ1OWQyY2IzYjk0ODYxNSA9IEwucG9wdXAoeyJtYXhXaWR0aCI6ICIxMDAlIn0pOwoKICAgICAgICAKICAgICAgICAgICAgdmFyIGh0bWxfZDhkMDQxM2UxZTJmNGVkNjgxNmViZGY2ZGY3YjJjZWEgPSAkKGA8ZGl2IGlkPSJodG1sX2Q4ZDA0MTNlMWUyZjRlZDY4MTZlYmRmNmRmN2IyY2VhIiBzdHlsZT0id2lkdGg6IDEwMC4wJTsgaGVpZ2h0OiAxMDAuMCU7Ij5Gb290c2NyYXkgUmFpbHdheSBTdGF0aW9uIFZpY3RvcmlhLCAzMDExLCBBdXN0cmFsaWE8L2Rpdj5gKVswXTsKICAgICAgICAgICAgcG9wdXBfZjU5YWUxZTNkNDg0NGUyY2I0NTlkMmNiM2I5NDg2MTUuc2V0Q29udGVudChodG1sX2Q4ZDA0MTNlMWUyZjRlZDY4MTZlYmRmNmRmN2IyY2VhKTsKICAgICAgICAKCiAgICAgICAgbWFya2VyXzQ2MTJjMzEyZGNhOTQwMmE4ZWIzZTlmMTUyNDc4ZTgxLmJpbmRQb3B1cChwb3B1cF9mNTlhZTFlM2Q0ODQ0ZTJjYjQ1OWQyY2IzYjk0ODYxNSkKICAgICAgICA7CgogICAgICAgIAogICAgCjwvc2NyaXB0Pg==\" style=\"position:absolute;width:100%;height:100%;left:0;top:0;border:none !important;\" allowfullscreen webkitallowfullscreen mozallowfullscreen></iframe></div></div>"
],
"text/plain": [
"<folium.folium.Map at 0x7fd851cce750>"
]
},
"execution_count": 4,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"m = folium.Map((location.latitude, location.longitude), max_zoom=20, zoom_start=16, height=600)\n",
"folium.Marker((location.latitude, location.longitude), popup=address).add_to(m)\n",
"m"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "subslide"
}
},
"source": [
"<sup>4</sup>Though after Friday I really want to try out Sangarshanan's `geopatra` library for this."
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "subslide"
}
},
"source": [
"### Added bonus: geocoding is built into GeoPandas\n",
"\n",
"If you haven't run into GeoPandas yet it's fantastics for working with spatial vector data. One of the tools it includes is a way to geocode a Pandas Series of addresses and get you back a GeoDataFrame of points:"
]
},
{
"cell_type": "code",
"execution_count": 5,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"data": {
"text/html": [
"<div>\n",
"<style scoped>\n",
" .dataframe tbody tr th:only-of-type {\n",
" vertical-align: middle;\n",
" }\n",
"\n",
" .dataframe tbody tr th {\n",
" vertical-align: top;\n",
" }\n",
"\n",
" .dataframe thead th {\n",
" text-align: right;\n",
" }\n",
"</style>\n",
"<table border=\"1\" class=\"dataframe\">\n",
" <thead>\n",
" <tr style=\"text-align: right;\">\n",
" <th></th>\n",
" <th>geometry</th>\n",
" <th>address</th>\n",
" </tr>\n",
" </thead>\n",
" <tbody>\n",
" <tr>\n",
" <th>0</th>\n",
" <td>POINT (144.90259 -37.80152)</td>\n",
" <td>Footscray, Hyde Street, Footscray, City of Mar...</td>\n",
" </tr>\n",
" </tbody>\n",
"</table>\n",
"</div>"
],
"text/plain": [
" geometry \\\n",
"0 POINT (144.90259 -37.80152) \n",
"\n",
" address \n",
"0 Footscray, Hyde Street, Footscray, City of Mar... "
]
},
"execution_count": 5,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"home = pd.Series([address], name='address')\n",
"home = gpd.tools.geocode(home, Nominatim, user_agent='Isochrone calculator')\n",
"home"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "subslide"
}
},
"source": [
"That's really handy, and we can easily put this on the same map as above:"
]
},
{
"cell_type": "code",
"execution_count": 6,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"data": {
"text/html": [
"<div style=\"width:100%;\"><div style=\"position:relative;width:100%;height:0;padding-bottom:60%;\"><iframe src=\"data:text/html;charset=utf-8;base64,PCFET0NUWVBFIGh0bWw+CjxoZWFkPiAgICAKICAgIDxtZXRhIGh0dHAtZXF1aXY9ImNvbnRlbnQtdHlwZSIgY29udGVudD0idGV4dC9odG1sOyBjaGFyc2V0PVVURi04IiAvPgogICAgCiAgICAgICAgPHNjcmlwdD4KICAgICAgICAgICAgTF9OT19UT1VDSCA9IGZhbHNlOwogICAgICAgICAgICBMX0RJU0FCTEVfM0QgPSBmYWxzZTsKICAgICAgICA8L3NjcmlwdD4KICAgIAogICAgPHNjcmlwdCBzcmM9Imh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vbGVhZmxldEAxLjYuMC9kaXN0L2xlYWZsZXQuanMiPjwvc2NyaXB0PgogICAgPHNjcmlwdCBzcmM9Imh0dHBzOi8vY29kZS5qcXVlcnkuY29tL2pxdWVyeS0xLjEyLjQubWluLmpzIj48L3NjcmlwdD4KICAgIDxzY3JpcHQgc3JjPSJodHRwczovL21heGNkbi5ib290c3RyYXBjZG4uY29tL2Jvb3RzdHJhcC8zLjIuMC9qcy9ib290c3RyYXAubWluLmpzIj48L3NjcmlwdD4KICAgIDxzY3JpcHQgc3JjPSJodHRwczovL2NkbmpzLmNsb3VkZmxhcmUuY29tL2FqYXgvbGlicy9MZWFmbGV0LmF3ZXNvbWUtbWFya2Vycy8yLjAuMi9sZWFmbGV0LmF3ZXNvbWUtbWFya2Vycy5qcyI+PC9zY3JpcHQ+CiAgICA8bGluayByZWw9InN0eWxlc2hlZXQiIGhyZWY9Imh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vbGVhZmxldEAxLjYuMC9kaXN0L2xlYWZsZXQuY3NzIi8+CiAgICA8bGluayByZWw9InN0eWxlc2hlZXQiIGhyZWY9Imh0dHBzOi8vbWF4Y2RuLmJvb3RzdHJhcGNkbi5jb20vYm9vdHN0cmFwLzMuMi4wL2Nzcy9ib290c3RyYXAubWluLmNzcyIvPgogICAgPGxpbmsgcmVsPSJzdHlsZXNoZWV0IiBocmVmPSJodHRwczovL21heGNkbi5ib290c3RyYXBjZG4uY29tL2Jvb3RzdHJhcC8zLjIuMC9jc3MvYm9vdHN0cmFwLXRoZW1lLm1pbi5jc3MiLz4KICAgIDxsaW5rIHJlbD0ic3R5bGVzaGVldCIgaHJlZj0iaHR0cHM6Ly9tYXhjZG4uYm9vdHN0cmFwY2RuLmNvbS9mb250LWF3ZXNvbWUvNC42LjMvY3NzL2ZvbnQtYXdlc29tZS5taW4uY3NzIi8+CiAgICA8bGluayByZWw9InN0eWxlc2hlZXQiIGhyZWY9Imh0dHBzOi8vY2RuanMuY2xvdWRmbGFyZS5jb20vYWpheC9saWJzL0xlYWZsZXQuYXdlc29tZS1tYXJrZXJzLzIuMC4yL2xlYWZsZXQuYXdlc29tZS1tYXJrZXJzLmNzcyIvPgogICAgPGxpbmsgcmVsPSJzdHlsZXNoZWV0IiBocmVmPSJodHRwczovL3Jhd2Nkbi5naXRoYWNrLmNvbS9weXRob24tdmlzdWFsaXphdGlvbi9mb2xpdW0vbWFzdGVyL2ZvbGl1bS90ZW1wbGF0ZXMvbGVhZmxldC5hd2Vzb21lLnJvdGF0ZS5jc3MiLz4KICAgIDxzdHlsZT5odG1sLCBib2R5IHt3aWR0aDogMTAwJTtoZWlnaHQ6IDEwMCU7bWFyZ2luOiAwO3BhZGRpbmc6IDA7fTwvc3R5bGU+CiAgICA8c3R5bGU+I21hcCB7cG9zaXRpb246YWJzb2x1dGU7dG9wOjA7Ym90dG9tOjA7cmlnaHQ6MDtsZWZ0OjA7fTwvc3R5bGU+CiAgICAKICAgICAgICAgICAgPG1ldGEgbmFtZT0idmlld3BvcnQiIGNvbnRlbnQ9IndpZHRoPWRldmljZS13aWR0aCwKICAgICAgICAgICAgICAgIGluaXRpYWwtc2NhbGU9MS4wLCBtYXhpbXVtLXNjYWxlPTEuMCwgdXNlci1zY2FsYWJsZT1ubyIgLz4KICAgICAgICAgICAgPHN0eWxlPgogICAgICAgICAgICAgICAgI21hcF9iYWQzZTU5N2Y0NWY0NDBkYjRkZThjYzZlZGQxM2VkMCB7CiAgICAgICAgICAgICAgICAgICAgcG9zaXRpb246IHJlbGF0aXZlOwogICAgICAgICAgICAgICAgICAgIHdpZHRoOiAxMDAuMCU7CiAgICAgICAgICAgICAgICAgICAgaGVpZ2h0OiA2MDAuMHB4OwogICAgICAgICAgICAgICAgICAgIGxlZnQ6IDAuMCU7CiAgICAgICAgICAgICAgICAgICAgdG9wOiAwLjAlOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICA8L3N0eWxlPgogICAgICAgIAogICAgCiAgICAgICAgICAgICAgICAgICAgPHN0eWxlPgogICAgICAgICAgICAgICAgICAgICAgICAuZm9saXVtdG9vbHRpcCB7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgICAgIC5mb2xpdW10b29sdGlwIHRhYmxlewogICAgICAgICAgICAgICAgICAgICAgICAgICAgbWFyZ2luOiBhdXRvOwogICAgICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgICAgIC5mb2xpdW10b29sdGlwIHRyewogICAgICAgICAgICAgICAgICAgICAgICAgICAgdGV4dC1hbGlnbjogbGVmdDsKICAgICAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgICAgICAuZm9saXVtdG9vbHRpcCB0aHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHBhZGRpbmc6IDJweDsgcGFkZGluZy1yaWdodDogOHB4OwogICAgICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgPC9zdHlsZT4KICAgICAgICAgICAgCjwvaGVhZD4KPGJvZHk+ICAgIAogICAgCiAgICAgICAgICAgIDxkaXYgY2xhc3M9ImZvbGl1bS1tYXAiIGlkPSJtYXBfYmFkM2U1OTdmNDVmNDQwZGI0ZGU4Y2M2ZWRkMTNlZDAiID48L2Rpdj4KICAgICAgICAKPC9ib2R5Pgo8c2NyaXB0PiAgICAKICAgIAogICAgICAgICAgICB2YXIgbWFwX2JhZDNlNTk3ZjQ1ZjQ0MGRiNGRlOGNjNmVkZDEzZWQwID0gTC5tYXAoCiAgICAgICAgICAgICAgICAibWFwX2JhZDNlNTk3ZjQ1ZjQ0MGRiNGRlOGNjNmVkZDEzZWQwIiwKICAgICAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgICAgICBjZW50ZXI6IFstMzcuODAxNTIwMiwgMTQ0LjkwMjU4NjldLAogICAgICAgICAgICAgICAgICAgIGNyczogTC5DUlMuRVBTRzM4NTcsCiAgICAgICAgICAgICAgICAgICAgem9vbTogMTYsCiAgICAgICAgICAgICAgICAgICAgem9vbUNvbnRyb2w6IHRydWUsCiAgICAgICAgICAgICAgICAgICAgcHJlZmVyQ2FudmFzOiBmYWxzZSwKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgKTsKCiAgICAgICAgICAgIAoKICAgICAgICAKICAgIAogICAgICAgICAgICB2YXIgdGlsZV9sYXllcl9iM2U1OTAwMGZmYjM0NjA4OTQ5NTk4YTc5NjQ1ZjUzOCA9IEwudGlsZUxheWVyKAogICAgICAgICAgICAgICAgImh0dHBzOi8ve3N9LnRpbGUub3BlbnN0cmVldG1hcC5vcmcve3p9L3t4fS97eX0ucG5nIiwKICAgICAgICAgICAgICAgIHsiYXR0cmlidXRpb24iOiAiRGF0YSBieSBcdTAwMjZjb3B5OyBcdTAwM2NhIGhyZWY9XCJodHRwOi8vb3BlbnN0cmVldG1hcC5vcmdcIlx1MDAzZU9wZW5TdHJlZXRNYXBcdTAwM2MvYVx1MDAzZSwgdW5kZXIgXHUwMDNjYSBocmVmPVwiaHR0cDovL3d3dy5vcGVuc3RyZWV0bWFwLm9yZy9jb3B5cmlnaHRcIlx1MDAzZU9EYkxcdTAwM2MvYVx1MDAzZS4iLCAiZGV0ZWN0UmV0aW5hIjogZmFsc2UsICJtYXhOYXRpdmVab29tIjogMjAsICJtYXhab29tIjogMjAsICJtaW5ab29tIjogMCwgIm5vV3JhcCI6IGZhbHNlLCAib3BhY2l0eSI6IDEsICJzdWJkb21haW5zIjogImFiYyIsICJ0bXMiOiBmYWxzZX0KICAgICAgICAgICAgKS5hZGRUbyhtYXBfYmFkM2U1OTdmNDVmNDQwZGI0ZGU4Y2M2ZWRkMTNlZDApOwogICAgICAgIAogICAgCiAgICAgICAgZnVuY3Rpb24gZ2VvX2pzb25fNmY3YzJiMTIwNmY5NDVkYjk2NDRjZWViZjc0MDg3Nzhfb25FYWNoRmVhdHVyZShmZWF0dXJlLCBsYXllcikgewogICAgICAgICAgICBsYXllci5vbih7CiAgICAgICAgICAgICAgICBjbGljazogZnVuY3Rpb24oZSkgewogICAgICAgICAgICAgICAgICAgIG1hcF9iYWQzZTU5N2Y0NWY0NDBkYjRkZThjYzZlZGQxM2VkMC5maXRCb3VuZHMoZS50YXJnZXQuZ2V0Qm91bmRzKCkpOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICB9KTsKICAgICAgICB9OwogICAgICAgIHZhciBnZW9fanNvbl82ZjdjMmIxMjA2Zjk0NWRiOTY0NGNlZWJmNzQwODc3OCA9IEwuZ2VvSnNvbihudWxsLCB7CiAgICAgICAgICAgICAgICBvbkVhY2hGZWF0dXJlOiBnZW9fanNvbl82ZjdjMmIxMjA2Zjk0NWRiOTY0NGNlZWJmNzQwODc3OF9vbkVhY2hGZWF0dXJlLAogICAgICAgICAgICAKICAgICAgICB9KS5hZGRUbyhtYXBfYmFkM2U1OTdmNDVmNDQwZGI0ZGU4Y2M2ZWRkMTNlZDApOwoKICAgICAgICBmdW5jdGlvbiBnZW9fanNvbl82ZjdjMmIxMjA2Zjk0NWRiOTY0NGNlZWJmNzQwODc3OF9hZGQgKGRhdGEpIHsKICAgICAgICAgICAgZ2VvX2pzb25fNmY3YzJiMTIwNmY5NDVkYjk2NDRjZWViZjc0MDg3NzguYWRkRGF0YShkYXRhKTsKICAgICAgICB9CiAgICAgICAgICAgIGdlb19qc29uXzZmN2MyYjEyMDZmOTQ1ZGI5NjQ0Y2VlYmY3NDA4Nzc4X2FkZCh7ImJib3giOiBbMTQ0LjkwMjU4NjksIC0zNy44MDE1MjAyLCAxNDQuOTAyNTg2OSwgLTM3LjgwMTUyMDJdLCAiZmVhdHVyZXMiOiBbeyJiYm94IjogWzE0NC45MDI1ODY5LCAtMzcuODAxNTIwMiwgMTQ0LjkwMjU4NjksIC0zNy44MDE1MjAyXSwgImdlb21ldHJ5IjogeyJjb29yZGluYXRlcyI6IFsxNDQuOTAyNTg2OSwgLTM3LjgwMTUyMDJdLCAidHlwZSI6ICJQb2ludCJ9LCAiaWQiOiAiMCIsICJwcm9wZXJ0aWVzIjogeyJhZGRyZXNzIjogIkZvb3RzY3JheSwgSHlkZSBTdHJlZXQsIEZvb3RzY3JheSwgQ2l0eSBvZiBNYXJpYnlybm9uZywgVmljdG9yaWEsIDMwMTEsIEF1c3RyYWxpYSJ9LCAidHlwZSI6ICJGZWF0dXJlIn1dLCAidHlwZSI6ICJGZWF0dXJlQ29sbGVjdGlvbiJ9KTsKICAgICAgICAKICAgIAogICAgZ2VvX2pzb25fNmY3YzJiMTIwNmY5NDVkYjk2NDRjZWViZjc0MDg3NzguYmluZFRvb2x0aXAoCiAgICBmdW5jdGlvbihsYXllcil7CiAgICBsZXQgZGl2ID0gTC5Eb21VdGlsLmNyZWF0ZSgnZGl2Jyk7CiAgICAKICAgIGxldCBoYW5kbGVPYmplY3QgPSBmZWF0dXJlPT50eXBlb2YoZmVhdHVyZSk9PSdvYmplY3QnID8gSlNPTi5zdHJpbmdpZnkoZmVhdHVyZSkgOiBmZWF0dXJlOwogICAgbGV0IGZpZWxkcyA9IFsiYWRkcmVzcyJdOwogICAgbGV0IGFsaWFzZXMgPSBbImFkZHJlc3MiXTsKICAgIGxldCB0YWJsZSA9ICc8dGFibGU+JyArCiAgICAgICAgU3RyaW5nKAogICAgICAgIGZpZWxkcy5tYXAoCiAgICAgICAgKHYsaSk9PgogICAgICAgIGA8dHI+CiAgICAgICAgICAgIDx0aD4ke2FsaWFzZXNbaV19PC90aD4KICAgICAgICAgICAgCiAgICAgICAgICAgIDx0ZD4ke2hhbmRsZU9iamVjdChsYXllci5mZWF0dXJlLnByb3BlcnRpZXNbdl0pfTwvdGQ+CiAgICAgICAgPC90cj5gKS5qb2luKCcnKSkKICAgICsnPC90YWJsZT4nOwogICAgZGl2LmlubmVySFRNTD10YWJsZTsKICAgIAogICAgcmV0dXJuIGRpdgogICAgfQogICAgLHsiY2xhc3NOYW1lIjogImZvbGl1bXRvb2x0aXAiLCAic3RpY2t5IjogdHJ1ZX0pOwogICAgICAgICAgICAgICAgICAgICAKPC9zY3JpcHQ+\" style=\"position:absolute;width:100%;height:100%;left:0;top:0;border:none !important;\" allowfullscreen webkitallowfullscreen mozallowfullscreen></iframe></div></div>"
],
"text/plain": [
"<folium.folium.Map at 0x7fd851e46c10>"
]
},
"execution_count": 6,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"m = folium.Map((location.latitude, location.longitude), max_zoom=20, zoom_start=16, height=600)\n",
"folium.GeoJson(home, tooltip=folium.GeoJsonTooltip(['address'])).add_to(m)\n",
"m"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "slide"
}
},
"source": [
"## Step 2: getting the street network within 5km of your home\n",
"\n",
"For now I'm going to gloss over projections, except to say that if you buffer a latitude and longitude point by 5000 you're buffering by a number of degrees - not so great - so you want to use a projection (GDA2020 MGA Zone 55 in this case, or EPSG:7855<sup>5</sup>) to get a projection measured in metres."
]
},
{
"cell_type": "code",
"execution_count": 7,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [],
"source": [
"buffer = home.to_crs(epsg=7855).buffer(5000).to_crs(epsg=4326) # this projects back to lat/lng after the buffer"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"source": [
"<sup>5</sup>If you need somewhere to look up projection EPSG codes try https://epsg.io"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "subslide"
}
},
"source": [
"It's definitely worth seeing this on a map - this will be a 5000 metre circle around your home. The same one we got before from the Victorian goverment website:"
]
},
{
"cell_type": "code",
"execution_count": 8,
"metadata": {
"scrolled": false,
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"data": {
"text/html": [
"<div style=\"width:100%;\"><div style=\"position:relative;width:100%;height:0;padding-bottom:60%;\"><iframe src=\"data:text/html;charset=utf-8;base64,<!DOCTYPE html>
<head>    
    <meta http-equiv="content-type" content="text/html; charset=UTF-8" />
    
        <script>
            L_NO_TOUCH = false;
            L_DISABLE_3D = false;
        </script>
    
    <script src="https://cdn.jsdelivr.net/npm/leaflet@1.6.0/dist/leaflet.js"></script>
    <script src="https://code.jquery.com/jquery-1.12.4.min.js"></script>
    <script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.2.0/js/bootstrap.min.js"></script>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.js"></script>
    <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/leaflet@1.6.0/dist/leaflet.css"/>
    <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.2.0/css/bootstrap.min.css"/>
    <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.2.0/css/bootstrap-theme.min.css"/>
    <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/font-awesome/4.6.3/css/font-awesome.min.css"/>
    <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.css"/>
    <link rel="stylesheet" href="https://rawcdn.githack.com/python-visualization/folium/master/folium/templates/leaflet.awesome.rotate.css"/>
    <style>html, body {width: 100%;height: 100%;margin: 0;padding: 0;}</style>
    <style>#map {position:absolute;top:0;bottom:0;right:0;left:0;}</style>
    
            <meta name="viewport" content="width=device-width,
                initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
            <style>
                #map_88180fb4e5e24ebc915d54466e2ecb0f {
                    position: relative;
                    width: 100.0%;
                    height: 600.0px;
                    left: 0.0%;
                    top: 0.0%;
                }
            </style>
        
    
                    <style>
                        .foliumtooltip {
                            
                        }
                       .foliumtooltip table{
                            margin: auto;
                        }
                        .foliumtooltip tr{
                            text-align: left;
                        }
                        .foliumtooltip th{
                            padding: 2px; padding-right: 8px;
                        }
                    </style>
            
</head>
<body>    
    
            <div class="folium-map" id="map_88180fb4e5e24ebc915d54466e2ecb0f" ></div>
        
</body>
<script>    
    
            var map_88180fb4e5e24ebc915d54466e2ecb0f = L.map(
                "map_88180fb4e5e24ebc915d54466e2ecb0f",
                {
                    center: [-37.8015202, 144.9025869],
                    crs: L.CRS.EPSG3857,
                    zoom: 12,
                    zoomControl: true,
                    preferCanvas: false,
                }
            );

            

        
    
            var tile_layer_e615ccbc02f341e7ae67eefb45fdcab4 = L.tileLayer(
                "https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png",
                {"attribution": "Data by \u0026copy; \u003ca href=\"http://openstreetmap.org\"\u003eOpenStreetMap\u003c/a\u003e, under \u003ca href=\"http://www.openstreetmap.org/copyright\"\u003eODbL\u003c/a\u003e.", "detectRetina": false, "maxNativeZoom": 20, "maxZoom": 20, "minZoom": 0, "noWrap": false, "opacity": 1, "subdomains": "abc", "tms": false}
            ).addTo(map_88180fb4e5e24ebc915d54466e2ecb0f);
        
    
        function geo_json_5c74aaa99d294602adfca6b407f9548d_onEachFeature(feature, layer) {
            layer.on({
                click: function(e) {
                    map_88180fb4e5e24ebc915d54466e2ecb0f.fitBounds(e.target.getBounds());
                }
            });
        };
        var geo_json_5c74aaa99d294602adfca6b407f9548d = L.geoJson(null, {
                onEachFeature: geo_json_5c74aaa99d294602adfca6b407f9548d_onEachFeature,
            
        }).addTo(map_88180fb4e5e24ebc915d54466e2ecb0f);

        function geo_json_5c74aaa99d294602adfca6b407f9548d_add (data) {
            geo_json_5c74aaa99d294602adfca6b407f9548d.addData(data);
        }
            geo_json_5c74aaa99d294602adfca6b407f9548d_add({"bbox": [144.8458298770797, -37.84655576195782, 144.95934676110906, -37.756484256295074], "features": [{"bbox": [144.8458298770797, -37.84655576195782, 144.95934676110906, -37.756484256295074], "geometry": {"coordinates": [[[144.95934676110906, -37.802517474611506], [144.95895190264648, -37.8069271687217], [144.95801408254005, -37.81128490987514], [144.9565422417238, -37.815548715308005], [144.95455047271625, -37.81967750034526], [144.95205788706744, -37.82363147490584], [144.94908843401726, -37.827372527761995], [144.94567067200592, -37.83086459481096], [144.94183749515372, -37.834074007765615], [144.9376258172844, -37.8369698198522], [144.93307621650274, -37.83952410532335], [144.9282325437461, -37.84171222984407], [144.9231414991038, -37.84351308909118], [144.9178521800373, -37.844909313215354], [144.91241560593042, -37.8458874351498], [144.90688422364872, -37.84643802110444], [144.90131139899006, -37.84655576195782], [144.89575089905762, -37.846239524645576], [144.89025637068266, -37.84549236304051], [144.88488082006705, -37.84432148822043], [144.8796760987996, -37.84273819842288], [144.87469240133268, -37.84075776938433], [144.8699777788809, -37.83839930615335], [144.865577674527, -37.83568555784615], [144.8615344840954, -37.832642697177604], [144.85788714707826, -37.829300066944334], [144.85467077158407, -37.82568989595791], [144.85191629692076, -37.82184698722124], [144.84965019703571, -37.81780838140702], [144.84789422761318, -37.81361299893061], [144.8466652191853, -37.8093012641113], [144.84597491814606, -37.80491471507993], [144.8458298770797, -37.80049560321953], [144.84623139532644, -37.79608648601646], [144.84717551021612, -37.79172981725198], [144.84865303891021, -37.78746753847873], [144.8506496703086, -37.783340675703265], [144.85314610600227, -37.77938894513535], [144.8561182487961, -37.775650371767625], [144.859537436884, -37.772160924417584], [144.86337072134182, -37.768954170698926], [144.86758118420997, -37.76606095519243], [144.87212829407474, -37.76350910386032], [144.87696829572286, -37.761323157494814], [144.88205463014333, -37.75952413671365], [144.8873383808858, -37.75812934071475], [144.89276874255432, -37.75715218168263], [144.8982935070243, -37.75660205640345], [144.9038595628172, -37.756484256295074], [144.90941340295424, -37.756799916698995], [144.91490163653702, -37.75754600591273], [144.92027149926955, -37.75871535407001], [144.92547135814544, -37.760296721602295], [144.9304512055713, -37.76227490664483], [144.9351631382873, -37.7646308903848], [144.93956181657524, -37.767342018991556], [144.94360489941306, -37.77038222042393], [144.94725345144263, -37.7737222540783], [144.9504723178626, -37.777329990928074], [144.95323046363964, -37.78117072151268], [144.9555012737473, -37.78520748886447], [144.95726281149, -37.78940144321894], [144.95849803234842, -37.79371221513851], [144.95919495118878, -37.79809830349593], [144.95934676110906, -37.802517474611506]]], "type": "Polygon"}, "id": "0", "properties": {}, "type": "Feature"}], "type": "FeatureCollection"});
        
    
        function geo_json_41a1ff91226e427190031ece55fabf48_onEachFeature(feature, layer) {
            layer.on({
                click: function(e) {
                    map_88180fb4e5e24ebc915d54466e2ecb0f.fitBounds(e.target.getBounds());
                }
            });
        };
        var geo_json_41a1ff91226e427190031ece55fabf48 = L.geoJson(null, {
                onEachFeature: geo_json_41a1ff91226e427190031ece55fabf48_onEachFeature,
            
        }).addTo(map_88180fb4e5e24ebc915d54466e2ecb0f);

        function geo_json_41a1ff91226e427190031ece55fabf48_add (data) {
            geo_json_41a1ff91226e427190031ece55fabf48.addData(data);
        }
            geo_json_41a1ff91226e427190031ece55fabf48_add({"bbox": [144.9025869, -37.8015202, 144.9025869, -37.8015202], "features": [{"bbox": [144.9025869, -37.8015202, 144.9025869, -37.8015202], "geometry": {"coordinates": [144.9025869, -37.8015202], "type": "Point"}, "id": "0", "properties": {"address": "Footscray, Hyde Street, Footscray, City of Maribyrnong, Victoria, 3011, Australia"}, "type": "Feature"}], "type": "FeatureCollection"});
        
    
    geo_json_41a1ff91226e427190031ece55fabf48.bindTooltip(
    function(layer){
    let div = L.DomUtil.create('div');
    
    let handleObject = feature=>typeof(feature)=='object' ? JSON.stringify(feature) : feature;
    let fields = ["address"];
    let aliases = ["address"];
    let table = '<table>' +
        String(
        fields.map(
        (v,i)=>
        `<tr>
            <th>${aliases[i]}</th>
            
            <td>${handleObject(layer.feature.properties[v])}</td>
        </tr>`).join(''))
    +'</table>';
    div.innerHTML=table;
    
    return div
    }
    ,{"className": "foliumtooltip", "sticky": true});
                     
</script>\" style=\"position:absolute;width:100%;height:100%;left:0;top:0;border:none !important;\" allowfullscreen webkitallowfullscreen mozallowfullscreen></iframe></div></div>"
],
"text/plain": [
"<folium.folium.Map at 0x7fd851e790d0>"
]
},
"execution_count": 8,
"metadata": {},
"output_type": "execute_result"
}
],
"source": [
"m = folium.Map((location.latitude, location.longitude), max_zoom=20, zoom_start=12, height=600)\n",
"folium.GeoJson(buffer).add_to(m)\n",
"folium.GeoJson(home, tooltip=folium.GeoJsonTooltip(['address'])).add_to(m)\n",
"m"
]
},
{
"cell_type": "markdown",
"metadata": {
"slideshow": {
"slide_type": "subslide"
}
},
"source": [
"To get the actual data underneath this to measure how far I can travel I'm going to use the `osmnx` library<sup>6</sup> - a tool that lets you download and extract Open Street Map data as a NetworkX graph along with the spatial data. (The data download can take a minute or two!)"
]
},
{
"cell_type": "code",
"execution_count": 9,
"metadata": {
"slideshow": {
"slide_type": "fragment"
}
},
"outputs": [
{
"data": {
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment