目标:使用ArcGIS的行驶时间计算数据,并使用Folium显示结果,而不是使用ArcGIS/ESRI的映射。
说明:使用arcgis软件包,我能够获得给定点周围的驾驶时间数据,为我提供要在地图上显示的边界区域。使用ArcGIS可以很好地显示来自ArcGIS地图的数据,但是,在ArcGIS地图上显示其他元素通常会非常慢,而且要耗费更多的时间,例如半径、单个点和任何带有自定义图标的元素。然而,Folium在显示所有这些额外数据方面工作得非常好,并且加载速度比ArcGIS快得多。
障碍:我正在寻找一种方法来使用从ArcGIS返回的数据,然后使用Folium显示它。我遇到的问题是,来自ArcGIS的数据,虽然它说它包括坐标数据,但它的值太大了,不适合作为纬度/经度。似乎有一些方法可以将这些值“解码”用于Folium,或者当我尝试在Folium中使用ArcGIS数据时,我可能没有使用它作为正确的数据类型。
import pandas
import arcgis
import webview
data = {'address': {0:'2015 Terminal Way'}, 'city': {0:'Reno'}, 'state': {0: 'NV'}, 'zip': {0:'89502'}}
df = pandas.DataFrame(data)
# Obviously I'm unable to include my username and password - which I understand probably limits who can help with this question since without logging in, you wouldn't be able to test my code, but there's nothing I can do about it
my_gis = arcgis.gis.GIS("https://www.arcgis.com", username, password)
address_layer = my_gis.content.import_data(df, address_fields={"Address":"address","City":"city","State":"state","Zip":"zip"})
target = arcgis.features.use_proximity.create_drive_time_areas(input_layer=address_layer, break_values=[5], break_units="Minutes", overlap_policy="Overlap")
# NOTE - ArcGIS documentation states that if an output name is given, the return object is a FeatureSet, but since I leave it blank, it is a FeatureCollection - not sure if that matters either
drivetime_data_geojson = target.query().to_geojson
# The above line returns as below, though trimmed down a bit to save space - I'm so sorry it's so long but pasting it in was the only way I could think of to allow others to test it in Folium
drivetime_data_geojson = '{"type": "FeatureCollection", "features": [{"type": "Feature", "geometry":
{"type": "Polygon", "coordinates": [[[-13334722.942400001, 4801659.346199997], [-13334622.9429,
4801594.495999999], [-13334572.9431, 4801335.0988000035], [-13334572.9431, 4800232.736199997],
[-13334197.9448, 4800232.736199997], [-13334122.9452, 4800297.577799998], [-13334022.945700001,
4800167.895199999], [-13334047.945500001, 4800005.794399999], [-13334122.9452, 4799940.954700001],
[-13334572.9431, 4799940.954700001], [-13334622.9429, 4799227.746699996], [-13334497.943500001,
4799195.329400003], [-13334447.9437, 4799065.6611], [-13334222.944699999, 4799065.6611],
[-13333947.9459, 4798968.4108000025], [-13333522.947900001, 4798676.666100003], [-13332722.9515,
4798579.419799998], [-13332622.952, 4798449.759499997], [-13332722.9515, 4798287.686399996],
[-13333247.9492, 4798320.100699998]]]}}]}'
# This is how it would be displayed using ArcGIS
# Creating the basemap image
map = my_gis.map(location='2015 Terminal Way Reno, NV 89502')
# Adding the layer to the map
map.add_layer(drivetime_data)
folder_path = os.getcwd()
path_to_map = folder_path + '\\arcgis_to_folium.html'
# webview is just a lightweight browser package that allows for easy viewing of these maps
webview.create_window('MAP', path_to_map)
webview.start() Folium看起来和这个差不多--尽管正如我所说的,GeoJson数据现在不会被拉进来:
import folium
folium_map = folium.Map(location=[39.505745, -119.77869])
drivetime_layer = folium.FeatureGroup(name='Drive Time')
folium.GeoJson(drivetime_data_geojson).add_to(drivetime_layer)
drivetime_layer.add_to(folium_map)
folium.LayerControl().add_to(folium_map)
folium_map.save('folium_drivetime_attempt.html')
path_to_map = folder_path + '\\folium_drivetime_attempt.html'
webview.create_window('MAP', path_to_map)
webview.start()此地图将加载,但不会有任何图层。我确实知道这个geojson会起作用:
drivetime_data_geojson = '{"type": "FeatureCollection", "features": [{
"type": "Feature", "geometry": {"type": "Polygon", "coordinates": [
[[-119.766227, 39.856011], [-120.260612, 39.251417], [-119.067222, 39.246099]]]}}]}'因为它使用的是GPS坐标,而不是来自ArcGIS的这些奇怪的值。因此,实际上,问题是找出如何从ArcGIS中“解码”这些值或将其转换为正确的数据类型。任何帮助,建议或想法都是非常感谢的。
发布于 2020-03-28 16:03:54
有两个问题你必须要处理。第一个是将ArcGIS x,y转换为lat,这可以通过如下函数来完成:
import math
def meters_to_coords(y, x):
if y > 0:
z = -20037508.3427892
else:
z = 20037508.3427892
lon = (y / z) * 180
lat = (x / z) * 180
lat = 180 / math.pi * (2 * math.atan(math.exp(lat * math.pi / 180)) - math.pi / 2)
return [lon, lat]您将注意到,当您将其转换为字典时,关键字“drivetime_data_geojson = target.query().to_geojson”将更改为“drivetime_data_dict = target.query().to_dict()”,这对下一步很重要。接下来,迭代字典并对坐标值运行上面的函数,以将它们更新为GPS坐标。然后,你需要调整字典的格式,使之成为folium可以识别的格式,本质上是拉入“环”数据,这些数据现在包含了你的坐标值,如下所示:
# Grab the same list of drive time values you sent into the ArcGIS request and use it here,
# since you'll get a separate set of x,y data for each driving time you request -
# per above, break_values was just [5]
drivetime_ranges = break_values
first_pass = False
for n in range(len(drivetime_ranges)):
for i, each in enumerate(drivetime_data_dict['features'][n]['geometry']['rings'][0]):
drivetime_data_dict['features'][n]['geometry']['rings'][0][i] = meters_to_coords(each[0], each[1])
if first_pass is not True:
default_geojson_structure = {
"type": "FeatureCollection",
"features": [{"type": "Feature",
"geometry": {"type": "Polygon",
"coordinates":
[drivetime_data_dict['features'][n]
['geometry']['rings'][0]]}}]
}
first_pass = True
else:
default_geojson_structure['features'].append({"type": "Feature",
"geometry": {"type": "Polygon", "coordinates":
[dt_map_overlay['features'][n]['geometry']['rings'][0]]}})
print(default_geojson_structure)现在只需创建一个folium FeatureGroup:
dt_layer = folium.FeatureGroup(name='Drive Time')并将default_geojson_structure加载到以下函数中,并将其添加到您的层中。
folium.GeoJson(default_geojson_structure, style_function=lambda x: {'fillColor':
'#0000ff'}).add_to(dt_layer)剩下的就像你在那里做的那样,保存地图和运行webview,你就会看到带有你的驾驶时间边界的地图。

https://stackoverflow.com/questions/60895953
复制相似问题