I am aware that the powerful package Basemap can be utilized to plot US map with state boundaries. I have adapted this example from Basemap GitHub repository to plot 48 states colored by their respective population density:
Now my question is: Is there a simple way to add Alaska and Hawaii to this map and place those at a custom location, e.g. bottom left corner? Something like this:
JavaScript
x
91
91
1
import numpy as np
2
import matplotlib.pyplot as plt
3
from mpl_toolkits.basemap import Basemap as Basemap
4
from matplotlib.colors import rgb2hex
5
from matplotlib.patches import Polygon
6
# Lambert Conformal map of lower 48 states.
7
m = Basemap(llcrnrlon=-119,llcrnrlat=22,urcrnrlon=-64,urcrnrlat=49,
8
projection='lcc',lat_1=33,lat_2=45,lon_0=-95)
9
# draw state boundaries.
10
# data from U.S Census Bureau
11
# http://www.census.gov/geo/www/cob/st2000.html
12
shp_info = m.readshapefile('st99_d00','states',drawbounds=True)
13
# population density by state from
14
# http://en.wikipedia.org/wiki/List_of_U.S._states_by_population_density
15
popdensity = {
16
'New Jersey': 438.00,
17
'Rhode Island': 387.35,
18
'Massachusetts': 312.68,
19
'Connecticut': 271.40,
20
'Maryland': 209.23,
21
'New York': 155.18,
22
'Delaware': 154.87,
23
'Florida': 114.43,
24
'Ohio': 107.05,
25
'Pennsylvania': 105.80,
26
'Illinois': 86.27,
27
'California': 83.85,
28
'Hawaii': 72.83,
29
'Virginia': 69.03,
30
'Michigan': 67.55,
31
'Indiana': 65.46,
32
'North Carolina': 63.80,
33
'Georgia': 54.59,
34
'Tennessee': 53.29,
35
'New Hampshire': 53.20,
36
'South Carolina': 51.45,
37
'Louisiana': 39.61,
38
'Kentucky': 39.28,
39
'Wisconsin': 38.13,
40
'Washington': 34.20,
41
'Alabama': 33.84,
42
'Missouri': 31.36,
43
'Texas': 30.75,
44
'West Virginia': 29.00,
45
'Vermont': 25.41,
46
'Minnesota': 23.86,
47
'Mississippi': 23.42,
48
'Iowa': 20.22,
49
'Arkansas': 19.82,
50
'Oklahoma': 19.40,
51
'Arizona': 17.43,
52
'Colorado': 16.01,
53
'Maine': 15.95,
54
'Oregon': 13.76,
55
'Kansas': 12.69,
56
'Utah': 10.50,
57
'Nebraska': 8.60,
58
'Nevada': 7.03,
59
'Idaho': 6.04,
60
'New Mexico': 5.79,
61
'South Dakota': 3.84,
62
'North Dakota': 3.59,
63
'Montana': 2.39,
64
'Wyoming': 1.96,
65
'Alaska': 0.42}
66
# choose a color for each state based on population density.
67
colors={}
68
statenames=[]
69
cmap = plt.cm.hot # use 'hot' colormap
70
vmin = 0; vmax = 450 # set range.
71
for shapedict in m.states_info:
72
statename = shapedict['NAME']
73
# skip DC and Puerto Rico.
74
if statename not in ['District of Columbia','Puerto Rico']:
75
pop = popdensity[statename]
76
# calling colormap with value between 0 and 1 returns
77
# rgba value. Invert color range (hot colors are high
78
# population), take sqrt root to spread out colors more.
79
colors[statename] = cmap(1.-np.sqrt((pop-vmin)/(vmax-vmin)))[:3]
80
statenames.append(statename)
81
# cycle through state names, color each one.
82
ax = plt.gca() # get current axes instance
83
for nshape,seg in enumerate(m.states):
84
# skip DC and Puerto Rico.
85
if statenames[nshape] not in ['District of Columbia','Puerto Rico']:
86
color = rgb2hex(colors[statenames[nshape]])
87
poly = Polygon(seg,facecolor=color,edgecolor=color)
88
ax.add_patch(poly)
89
plt.title('Filling State Polygons by Population Density')
90
plt.show()
91
Advertisement
Answer
For anyone interested, I was able to fix it myself. The (x,y) coordinates of each segment (for Alaska and Hawaii) should be translated. I also scale down Alaska to 35% before translating it.
The second for-loop should be modified as following:
JavaScript
1
14
14
1
for nshape,seg in enumerate(m.states):
2
# skip DC and Puerto Rico.
3
if statenames[nshape] not in ['Puerto Rico', 'District of Columbia']:
4
# Offset Alaska and Hawaii to the lower-left corner.
5
if statenames[nshape] == 'Alaska':
6
# Alaska is too big. Scale it down to 35% first, then transate it.
7
seg = list(map(lambda (x,y): (0.35*x + 1100000, 0.35*y-1300000), seg))
8
if statenames[nshape] == 'Hawaii':
9
seg = list(map(lambda (x,y): (x + 5100000, y-900000), seg))
10
11
color = rgb2hex(colors[statenames[nshape]])
12
poly = Polygon(seg,facecolor=color,edgecolor=color)
13
ax.add_patch(poly)
14
Here is the new US map (using the ‘Greens’ colormap).