%matplotlib inline import jinja2 import json import requests import StringIO import pandas as pd import matplotlib.pyplot as plt import numpy as np from collections import OrderedDict from datetime import datetime from dateutil import parser from io import BytesIO from IPython.html.widgets import interact from IPython.html import widgets from IPython.display import display, display_pretty, Javascript, HTML from IPython.utils.traitlets import Unicode from mpl_toolkits.basemap import Basemap from xml.etree import ElementTree %%javascript require.config({ paths: { async: '//cdnjs.cloudflare.com/ajax/libs/requirejs-async/0.1.1/async', open_layers: '//cdnjs.cloudflare.com/ajax/libs/openlayers/2.13.1/OpenLayers' } }); # Taken from http://nbviewer.ipython.org/github/ipython/ipython/blob/master/examples/Interactive%20Widgets/Custom%20Widgets.ipynb class DateWidget(widgets.DOMWidget): _view_name = Unicode('DatePickerView', sync=True) value = Unicode(sync=True) description = Unicode(sync=True) # This function automatically gets called by the traitlet machinery when # value is modified because of this function's name. def _value_changed(self, name, old_value, new_value): # Parse the date time value. try: parsed_date = parser.parse(new_value) parsed_date_string = parsed_date.strftime("%Y-%m-%d") except: parsed_date_string = '' # Set the parsed date string if the current date string is different. if self.value != parsed_date_string: self.value = parsed_date_string %%javascript // Taken from http://nbviewer.ipython.org/github/ipython/ipython/blob/master/examples/Interactive%20Widgets/Custom%20Widgets.ipynb require(["widgets/js/widget"], function(WidgetManager){ // Define the DatePickerView var DatePickerView = IPython.DOMWidgetView.extend({ render: function(){ this.$el.addClass('widget-hbox-single'); /* Apply this class to the widget container to make it fit with the other built in widgets.*/ // Create a label. this.$label = $('
') .addClass('widget-hlabel') .appendTo(this.$el) .hide(); // Hide the label by default. // Create the date picker control. this.$date = $('') .attr('type', 'date') .appendTo(this.$el); // FIXED: This line wasn't on the example, but the widget is not properly initialized if we // don't call update() on render() this.update(); }, update: function() { console.log('value', this.model.get('value')) // Set the value of the date control and then call base. this.$date.val(this.model.get('value')); // ISO format "YYYY-MM-DDTHH:mm:ss.sssZ" is required // Hide or show the label depending on the existance of a description. var description = this.model.get('description'); if (description == undefined || description == '') { this.$label.hide(); } else { this.$label.show(); this.$label.text(description); } return DatePickerView.__super__.update.apply(this); }, // Tell Backbone to listen to the change event of input controls (which the HTML date picker is) events: {"change": "handle_date_change"}, // Callback for when the date is changed. handle_date_change: function(event) { this.model.set('value', this.$date.val()); this.touch(); }, }); // Register the DatePickerView with the widget manager. WidgetManager.register_widget_view('DatePickerView', DatePickerView); }); def display_map(latitude, longitude, starttime, endtime, maxradiuskm, minmagnitude): # Make the request to the Earthquake Search and Web Service API response = requests.get('http://comcat.cr.usgs.gov/fdsnws/event/1/query', params={ 'starttime': parser.parse(starttime).isoformat(), 'endtime': parser.parse(endtime).isoformat(), 'latitude': latitude, 'longitude': longitude, 'maxradiuskm': maxradiuskm, 'minmagnitude': minmagnitude, 'magtype': 'md', 'format': 'csv' # We ask for a CSV formatted response }) # Create a DataFrame from the CSV response. Notice that we force Pandas to # interpret the latitude and longitude columns as float and that we use the # time column as index. earthquakes_df = pd.read_csv( StringIO.StringIO(response.text), dtype={'latitude':'float', 'longitude':'float'}, index_col='time', parse_dates=True ) figure, axes = plt.subplots(nrows=1, ncols=1, figsize=(12, 12)) # Create a new map with cylindrical projection centered around the supplied # coordinates m = Basemap( ax=axes, projection='cyl', lon_0=longitude, lat_0=latitude, llcrnrlat=latitude - 5.0, urcrnrlat=latitude + 5.0, llcrnrlon=longitude - 5.0, urcrnrlon=longitude + 5.0, resolution='i' ) m.drawmapboundary(fill_color='aqua') m.fillcontinents(color='coral',lake_color='aqua') m.drawcoastlines() m.drawstates() m.drawcountries() m.drawstates() # Plot the geolocical events as blue dots x, y = m(earthquakes_df.longitude, earthquakes_df.latitude) m.plot(x, y, 'bo', alpha=0.5) x, y = m([longitude], [latitude]) m.plot(x, y, 'rx', markersize=15.0) plt.title('Earthquakes around ({longitude}, {latitude}) from {starttime} to {endtime}'.format(longitude=longitude, latitude=latitude, starttime=starttime, endtime=endtime)) plt.show() i = interact( display_map, latitude=widgets.FloatTextWidget(value=19.4284700, min=-90.0, max=90.0), longitude=widgets.FloatTextWidget(value=-99.1276600, min=-180.0, max=180.0), starttime=DateWidget(value='1984-01-01'), endtime=DateWidget(value='2014-01-01'), maxradiuskm=widgets.FloatTextWidget(value=800, min=0.0), minmagnitude=widgets.FloatTextWidget(value=4.0, min=0.0) ) def display_earthquake_markers_map(latitude, longitude, feed, map_type, div): # Make a request to the supplied feed and parse the response as JSON response = requests.get(feed) response_json = response.json() # Create a DataFrame from a list comprehension built from the "features" # field of the JSON response. latest_earthquakes = [( datetime.fromtimestamp(float(feature['properties']['time']) / 1000.0), feature['geometry']['coordinates'][1], feature['geometry']['coordinates'][0], feature['properties']['mag'], feature['properties']['title'], ) for feature in response_json['features']] latest_earthquakes_df = pd.DataFrame(latest_earthquakes, columns=['time', 'latitude', 'longitude', 'mag', 'title']) latest_earthquakes_df.set_index('time', inplace=True) # Define a Jinja2 template that creates the map, and iterates of the # features to create a marker for each of them. latest_earthquakes_javascript = jinja2.Template( """ require(['async!http://maps.googleapis.com/maps/api/js?sensor=false&libraries=visualization&callback=console.log'], function() { var mapOptions = { zoom: 2.0, center: new google.maps.LatLng({{ latitude }}, {{ longitude }}), mapTypeId: {{ map_type }} }; var map = new google.maps.Map(document.getElementById('latest_earthquakes_map_markers'), mapOptions); var markers = Array(); var machinalis_marker = new google.maps.Marker({ position: new google.maps.LatLng(-31.4122599,-64.2010031), icon: { size: new google.maps.Size(30, 30), url: '' }, title: 'http://www.machinalis.com', map: map }); {% for earthquake in latest_earthquakes %} markers.push(new google.maps.Marker({ position: new google.maps.LatLng({{ earthquake[1] }}, {{ earthquake[2] }}), title: "{{ earthquake[4] }}", map: map })); {% endfor %} }); """ ) # Render the template and display it as Javascript display(Javascript(latest_earthquakes_javascript.render( latest_earthquakes=latest_earthquakes_df.itertuples(), map_type=map_type, latitude=latitude, longitude=longitude ))) i = interact( display_earthquake_markers_map, latitude=widgets.FloatTextWidget(value=-31.4122599, min=-90.0, max=90.0), longitude=widgets.FloatTextWidget(value=-64.2010031, min=-180.0, max=180.0), map_type=widgets.DropdownWidget(values={ 'Road Map': 'google.maps.MapTypeId.ROADMAP', 'Satellite': 'google.maps.MapTypeId.SATELLITE', 'Hybrid': 'google.maps.MapTypeId.HYBRID', 'Terrain': 'google.maps.MapTypeId.TERRAIN', }, value='google.maps.MapTypeId.SATELLITE'), feed=widgets.DropdownWidget(values={ 'Significant Earthquakes': 'http://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/significant_week.geojson', 'M4.5+ Earthquakes': 'http://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/4.5_week.geojson', 'M2.5+ Earthquakes': 'hytp://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/2.5_week.geojson', 'M1.0+ Earthquakes': 'http://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/1.0_week.geojson', 'All Earthquakes': 'http://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/all_week.geojson' }, value='http://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/4.5_week.geojson'), div=widgets.HTMLWidget(value='
') ) def display_earthquake_circles_map(latitude, longitude, feed, map_type, div): response = requests.get(feed) response_json = response.json() latest_earthquakes = [( datetime.fromtimestamp(float(feature['properties']['time']) / 1000.0), feature['geometry']['coordinates'][1], feature['geometry']['coordinates'][0], feature['properties']['mag'], feature['properties']['title'], ) for feature in response_json['features']] latest_earthquakes_df = pd.DataFrame(latest_earthquakes, columns=['time', 'latitude', 'longitude', 'mag', 'title']) latest_earthquakes_df.set_index('time', inplace=True) latest_earthquakes_javascript = jinja2.Template( """ require(['async!http://maps.googleapis.com/maps/api/js?sensor=false&libraries=visualization&callback=console.log'], function() { function getCircle(magnitude) { return { path: google.maps.SymbolPath.CIRCLE, fillColor: 'red', fillOpacity: .2, scale: Math.pow(2, magnitude) / Math.PI, strokeColor: 'white', strokeWeight: .5 }; } var mapOptions = { zoom: 2.0, center: new google.maps.LatLng({{ latitude }}, {{ longitude }}), mapTypeId: {{ map_type }} }; var map = new google.maps.Map(document.getElementById('latest_earthquakes_map_circles'), mapOptions); var machinalis_marker = new google.maps.Marker({ position: new google.maps.LatLng(-31.4122599,-64.2010031), icon: { size: new google.maps.Size(30, 30), url: '' }, title: 'http://www.machinalis.com', map: map }); var markers = Array(); {% for earthquake in latest_earthquakes %} markers.push(new google.maps.Marker({ position: new google.maps.LatLng({{ earthquake[1] }}, {{ earthquake[2] }}), title: "{{ earthquake[4] }}", icon: getCircle({{ earthquake[3] }}), map: map })); {% endfor %} }); """ ) display(Javascript(latest_earthquakes_javascript.render( latest_earthquakes=latest_earthquakes_df.itertuples(), map_type=map_type, latitude=latitude, longitude=longitude ))) i = interact( display_earthquake_circles_map, latitude=widgets.FloatTextWidget(value=-31.4122599, min=-90.0, max=90.0), longitude=widgets.FloatTextWidget(value=-64.2010031, min=-180.0, max=180.0), map_type = widgets.DropdownWidget(values={ 'Road Map': 'google.maps.MapTypeId.ROADMAP', 'Satellite': 'google.maps.MapTypeId.SATELLITE', 'Hybrid': 'google.maps.MapTypeId.HYBRID', 'Terrain': 'google.maps.MapTypeId.TERRAIN', }, value='google.maps.MapTypeId.SATELLITE'), feed = widgets.DropdownWidget(values={ 'Significant Earthquakes': 'http://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/significant_week.geojson', 'M4.5+ Earthquakes': 'http://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/4.5_week.geojson', 'M2.5+ Earthquakes': 'hytp://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/2.5_week.geojson', 'M1.0+ Earthquakes': 'http://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/1.0_week.geojson', 'All Earthquakes': 'http://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/all_week.geojson' }, value='http://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/4.5_week.geojson'), div=widgets.HTMLWidget(value='
') ) def display_heat_points_map(latitude, longitude, map_type, div): # Make the request and parse the response as XML using ElementTree response = requests.get('http://catalogos.conae.gov.ar/focos/focosdecalor.xml') response_atom = ElementTree.parse(BytesIO(response.content)) # Build a DataFrame extrating the data for each entry using XPath entries_id = [e.text for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.w3.org/2005/Atom}id')] entries_title = [e.text for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.w3.org/2005/Atom}title')] entries_updated = [e.text for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.w3.org/2005/Atom}updated')] entries_summary = [e.text for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.w3.org/2005/Atom}summary')] entries_point_lat = [float(e.text.split(" ")[0]) for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.georss.org/georss}point')] entries_point_lon = [float(e.text.split(" ")[1]) for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.georss.org/georss}point')] entries_df = pd.DataFrame({ 'id': entries_id, 'title': entries_title, 'summary': entries_summary, 'lat': entries_point_lat, 'lon': entries_point_lon }, index=entries_updated) # Define a Jinja2 template that creates the map, and iterates of the # entries to create a marker with a custom icon for each of them. heat_points_javascript = jinja2.Template( """ require(['async!http://maps.googleapis.com/maps/api/js?sensor=false&libraries=visualization&callback=console.log'], function() { var mapOptions = { zoom: 4, center: new google.maps.LatLng({{ latitude }}, {{ longitude }}), mapTypeId: {{ map_type }} }; var map = new google.maps.Map(document.getElementById('heat_points'), mapOptions); var machinalis_marker = new google.maps.Marker({ position: new google.maps.LatLng(-31.4122599,-64.2010031), icon: { size: new google.maps.Size(30, 30), url: '' }, title: 'http://www.machinalis.com', map: map }); // Taken from http://hubpic.com/flame-icon-vector-material-04.html var marker_icon = { size: new google.maps.Size(21, 32), url: '' } var markers = Array(); var marker_info_windows = Array(); {% for heat_point in heat_points %} marker_info_windows[{{ loop.index0 }}] = new google.maps.InfoWindow({ content: '{{ heat_point[4] }}' }); markers[{{ loop.index0 }}] = new google.maps.Marker({ position: new google.maps.LatLng({{ heat_point[2] }}, {{ heat_point[3] }}), title: '{{ heat_point[5] }}', icon: marker_icon, map: map }); google.maps.event.addListener(markers[{{ loop.index0 }}], 'click', function() { var _map = marker_info_windows[{{ loop.index0 }}].getMap(); if (_map !== null && typeof _map !== "undefined") { marker_info_windows[{{ loop.index0 }}].close(); } else { marker_info_windows[{{ loop.index0 }}].open(map, markers[{{ loop.index0 }}]); } }); {% endfor %} }) """ ) # Render the template and display it as Javascript display(Javascript(heat_points_javascript.render( heat_points=entries_df.itertuples(), map_type=map_type, latitude=latitude, longitude=longitude ))) i = interact( display_heat_points_map, latitude=widgets.FloatTextWidget(value=-31.4122599, min=-90.0, max=90.0), longitude=widgets.FloatTextWidget(value=-64.2010031, min=-180.0, max=180.0), map_type = widgets.DropdownWidget(values={ 'Road Map': 'google.maps.MapTypeId.ROADMAP', 'Satellite': 'google.maps.MapTypeId.SATELLITE', 'Hybrid': 'google.maps.MapTypeId.HYBRID', 'Terrain': 'google.maps.MapTypeId.TERRAIN', }, value='google.maps.MapTypeId.SATELLITE'), div=widgets.HTMLWidget(value='
'), ) def display_heat_points_map_ol(latitude, longitude, zoom, div): # Make the request and parse the response as XML using ElementTree response = requests.get('http://catalogos.conae.gov.ar/focos/focosdecalor.xml') response_atom = ElementTree.parse(BytesIO(response.content)) # Build a DataFrame extrating the data for each entry using XPath entries_id = [e.text for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.w3.org/2005/Atom}id')] entries_title = [e.text for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.w3.org/2005/Atom}title')] entries_updated = [e.text for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.w3.org/2005/Atom}updated')] entries_summary = [e.text for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.w3.org/2005/Atom}summary')] entries_point_lat = [float(e.text.split(" ")[0]) for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.georss.org/georss}point')] entries_point_lon = [float(e.text.split(" ")[1]) for e in response_atom.findall('.//{http://www.w3.org/2005/Atom}entry/{http://www.georss.org/georss}point')] entries_df = pd.DataFrame({ 'id': entries_id, 'title': entries_title, 'summary': entries_summary, 'lat': entries_point_lat, 'lon': entries_point_lon }, index=entries_updated) # Define a Jinja2 template that creates the map, and iterates of the # entries to create a marker with a custom icon for each of them. heat_points_javascript = jinja2.Template( """ require(["open_layers"], function() { var map = new OpenLayers.Map('heat_points_ol', {}); // var layer = new OpenLayers.Layer.WMS('OpenLayers WMS', 'http://vmap0.tiles.osgeo.org/wms/vmap0', {layers: 'basic'}); var layer = new OpenLayers.Layer.WMS( 'Global Imagery', 'http://maps.opengeo.org/geowebcache/service/wms', {layers: 'bluemarble'}, {isBaseLayer: true} ); map.addLayer(layer); map.setCenter( new OpenLayers.LonLat({{ longitude }}, {{ latitude }}), {{ zoom }} ); var heat_points_layer = new OpenLayers.Layer.Markers("Heat Points Layer"); var machinalis_logo_icon_url = ''; var heat_point_icon_url = ''; var machinalis_logo_icon = new OpenLayers.Icon(machinalis_logo_icon_url, new OpenLayers.Size(30, 30)); heat_points_layer.addMarker(new OpenLayers.Marker(new OpenLayers.LonLat(-64.2010031, -31.4122599), machinalis_logo_icon)); var heat_point_icon = new OpenLayers.Icon(heat_point_icon_url, new OpenLayers.Size(21, 32)); {% for heat_point in heat_points %} heat_points_layer.addMarker(new OpenLayers.Marker(new OpenLayers.LonLat({{ heat_point[3] }}, {{ heat_point[2] }}), heat_point_icon.clone())); {% endfor %} map.addLayer(heat_points_layer); }) """ ) # Render the template and display it as Javascript display(Javascript(heat_points_javascript.render( heat_points=entries_df.itertuples(), zoom=zoom, latitude=latitude, longitude=longitude ))) i = interact( display_heat_points_map_ol, latitude=widgets.FloatTextWidget(value=-31.4122599, min=-90.0, max=90.0), longitude=widgets.FloatTextWidget(value=-64.2010031, min=-180.0, max=180.0), zoom=widgets.IntSliderWidget(value=4, min=0, max=100), div=widgets.HTMLWidget(value='
'), )