引言
在地图中精确的绘制一系列的图形一直是一个让人头疼的问题,一个个的输入坐标烦不胜烦,而且绘制完毕之后,如何保存分享也是问题。
这里我们将介绍一种简单的方法来解决这个问题:使用ArcGIS Earth 1.16版本AutomationAPI的drawings接口,可以在地图上精确的绘制一系列的图形。
效果如下:(原图审核不过,我就用这幅图片代替)
ArcGIS Earth
ArcGIS Earth作为一款免费的、轻量级的、易于使用的交互式三维地球平台,给非GIS专业的用户带来友好又便捷的使用体验。使用ArcGIS Earth不仅可以免费使用Esri提供的全球地形、影像服务,足不出户浏览探索地球;还可以添加自己数据,实现在三维空间内的数据展示,并满足用户绘制要素、量测、分享等需求。
今天ArcGIS Earth Desktop 1.16 版本正好发布啦,从这个页面就能看到它的新特性:
- New templates for drawing lines and polygons have been added to support creating predefined shapes.
- Filter is supported to define expressions for determining which features are shown on the map.
- Remember certificate for PKI portal
- No Basemap option
- Summary tab for layer properties
- Automation API:support drawings
- Save/Save as in combination
这里就先暂时不翻译了,后面应该会有官方的介绍的。就先列一下相关的链接吧:
Drawings API
ArcGIS Earth Automation API就是在Earth运行的时候可以启动一个后台服务,用户可以以REST的方式和ArcGIS Earth Desktop通讯,在1.16版本里面新增了一个drawings功能,支持向ArcGIS Earth 发送POST DELETE等请求来绘制和删除图形。
然后通过软件原本的分享和保存功能,可以将图形以KML的形式保存到本地,并且可以将KML文件通过Arcgis online或其他形式分享给其他用户。
接口的详细说明在这个页面。添加图形的过程大致就是把一个json格式的点线面传给ArcGIS Earth。点图形的json格式示例如下:
{
"id": "4b75ea4a-e10f-d676-307d-aa945e2a0712",
"visible": true,
"title": "addPoint",
"geometry": {
"x": -100,
"y": 40,
"spatialReference": {"wkid": 4326}
},
"symbol": {
"type": "picture-marker",
"url":"https://static.arcgis.com/images/Symbols/Shapes/BlackStarLargeB.png",
"size": "64px"
},
"labelSymbol":{
"type":"text",
"color": [76,115,0,255],
"size":12
}
}
也是由geometry
,symbol
等字段组成的json。
开启autoamtion api
那么如何开启并使用drawings功能呢?这里简单介绍一下吧,需要打开ArcGIS Earth进行一下设置
- 从工具栏点击设置
- 从设置面板选择高级
- 滑动滚动条到Automation API区域,并勾选Enable Automation API
- 从Sample页面和Help页面获取更多信息
打开sample页面之后,可以用下面的方法进行绘制:
同样,打开helpe页面,可以使用swagger提供的功能了解API的详情或者Try it out。
官方还有一个repo来提供了.NET的客户端,可以使用这个客户端来调用API。
earth-api for python
通过上面的介绍,我们已经简单的了解了ArcGISEarth Desktop的相关功能, ArcGIS Earth Automation API的基本原理和使用方法。
但如果要高效的进行绘制,还是需要通过一些封装好的接口来实现的。这里介绍一下python的接口。
在这个页面可以看到arcgis-earth-automation-api for python的基本的介绍。
在我们完成了上面的步骤开启了automation api之后,我们可以使用earth-api for python来进行绘制。
首先进行安装
pip install earth-api
然后已获取相机位置为例:
from earth_api import EarthAPI
earth = EarthAPI()
status_code, camera = earth.get_camera()
通过上述三行代码就能获取相机的基本信息了,是不是非常简单呢?下面对这三行代码做一个简单地说明:
from earth_api import EarthAPI
这行代码是从earth_api这个库中引入了EarthAPI类,EarthAPI类中封装了所有的接口。
earth = EarthAPI()
这行代码创建了earth api的对象,传入参数还支持api的版本号等。
status_code, camera = earth.get_camera()
上面一行代码是通过api来获取相机位置,实际上是通过该PythonAPI向ArcGIS Earth发送了一个GET Camera的请求。然后返回了状态码和camera的对象,在此camera对象以dict的形式表现。
稍复杂的绘制
通过上面的介绍,想必大家已经对earth-api的使用有了基本的了解了。下面我们就以开头的图形为例来说明一下如何自动化的绘制图形。
首先,我们需要从文章中复制出区域的基本信息。
a1 = "北纬25°15′26″、东经120°29′20″,北纬24°50′30″、东经120°05′45″,北纬25°04′32″、东经119°51′22″,北纬25°28′12″、东经120°14′30″四点连线。"
a2 = "北纬26°07′00″、东经121°57′00″,北纬25°30′00″、东经121°57′00″,北纬25°30′00″、东经121°28′00″,北纬26°07′00″、东经121°28′00″四点连线。"
a3 = "北纬25°34′00″、东经122°50′00″,北纬25°03′00″、东经122°50′00″,北纬25°03′00″、东经122°11′00″,北纬25°34′00″、东经122°11′00″四点连线。"
a4 = "北纬22°56′00″、东经122°40′00″,北纬23°38′00″、东经122°51′00″,北纬23°38′00″、东经123°23′00″,北纬22°56′00″、东经123°09′00″四点连线。"
a5 = "北纬21°14′00″、东经121°33′00″,北纬21°33′00″、东经121°18′00″,北纬21°07′00″、东经120°43′00″,北纬20°48′00″、东经120°59′00″四点连线。"
a6 = "北纬22°43′00″、东经119°14′00″,北纬22°10′00″、东经119°06′00″,北纬21°33′00″、东经120°29′00″,北纬22°09′00″、东经120°32′00″四点连线。"
然后简单的提取出经纬度信息,由于上面的描述格式比较统一,我就用简单的方式来提取出经纬度信息吧。
# get degress, minutes, seconds from string
def get_degress_minutes_seconds(desc_string):
# get degress
degress = desc_string.split('°')[0][2:]
# get minutes
minutes = desc_string.split('°')[1].split('′')[0]
# get seconds
seconds = desc_string.split('°')[1].split('′')[1].split('″')[0]
return degress, minutes, seconds
接着把经纬度信息构建为多边形的几何体,这里面我就以rings作为基本的几何单元吧。
首先将度分秒转换为度。
# convert degress, minutes, seconds to decimal degree
def convert_degress_minutes_seconds_to_decimal_degree(degress, minutes, seconds):
decimal_degree = float(degress) + float(minutes) / 60 + float(seconds) / 3600
return decimal_degree
然后构建ring,ring是首尾相连的线段。
def consturct_ring(lat_array, lon_array):
ring = []
for i in range(len(lat_array)):
ring.append([lon_array[i], lat_array[i]])
ring.append([lon_array[0], lat_array[0]])
return ring
最后我们把这些代码组合起来,实现从句子中获取了ring的信息。
# get location from string
def get_location_from_string(desc_string):
# split string by '、' and ','
sections = re.split(',|、', desc_string)
lat_array = []
lon_array = []
for s in sections:
if '北纬' in s:
degress, minutes, seconds = get_degress_minutes_seconds(s)
lat = convert_degress_minutes_seconds_to_decimal_degree(degress, minutes, seconds)
lat_array.append(lat)
if '东经' in s:
degress, minutes, seconds = get_degress_minutes_seconds(s)
lon = convert_degress_minutes_seconds_to_decimal_degree(degress, minutes, seconds)
lon_array.append(lon)
return consturct_ring(lat_array, lon_array)
接着我们要创建一个arcgis earth automation api能识别的drawing对象,一会儿就把这个对象给传递给ArcGIS Earth。
def generate_drawings_from_string(name, desc_string):
ring = get_location_from_string(desc_string)
drawing = {
"visible": True,
"title":name,
"geometry":{
"rings":[
ring
],
"spatialReference":{
"wkid":4326
}
}
}
return drawing
最后,我们需要调用earth-api将该对象添加到地图上。
earth_api = EarthAPI()
earth_api.add_drawing(drawing)
然后就实现了文章开头的效果。完整代码在这个链接,大概81行代码吧(含空行)
最后,使用earth-api给大家比个❤送给大家吧。
使用earth-api绘制爱心
今天就先到这里吧,更详细的介绍以后再慢慢展开。谢谢观看。