首页 > 其他分享 >几何计算-基于Turf.js实现多边形的拆分及合并

几何计算-基于Turf.js实现多边形的拆分及合并

时间:2023-10-13 10:57:59浏览次数:52  
标签:多边形 切割 合并 js let 拆分 Turf

几何计算-基于Turf.js实现多边形的拆分及合并

几何计算-基于Turf.js实现多边形的拆分及合并

阿飞 阿飞 红星美凯龙 3D前端开发工程师  
❝ JSAPI GL近期为支持物流行业实现了几何图形编辑器,用户可通过编辑器接口进行点、线、面、圆的绘制和编辑。在物流行业中常见的使用场景是配送区域及地理围栏的绘制,常会有对已有区域进行拆分或者合并的需要,所以编辑器也提供了相应的功能。本文介绍了如何基于Turf实现多边形的拆分及合并。

背景介绍

多边形的拆分合并

多边形的拆分是指将多边形沿着线切分为几个多边形。如下图所示,不仅可以沿线一分为二,当线与多边形有多段相交时也可以分为多份,另外当多边形带洞(环多边形)时也可以在拆分后保持洞的形状。

动图封面  

多边形的合并是指将多个多边形合并为一个多边形,其前提条件是多边形之间有交叉区域或者共边。如下图所示,完全共边或者部分共边都可以合并,当有交叉时会贯通交叉部分。

动图封面  

Turf.js

不难发现,多边形的拆分合并中会有大量且复杂的几何计算,包括点、线、面相互之间的相交、包含等计算。不过我们并不需要造轮子,可以使用Turf.js完成大部分的基础计算。Turf是由mapbox推出的空间几何计算库,常用于地理空间内的几何关系分析,功能非常强大,具体功能可见Turf.js | Advanced geospatial analysis

可是Turf.js目前还没有提供多边形的拆分方法,另外多边形的合并虽然已有union方法,但在实际应用中也无法很好解决部分共边的多边形的合并问题,所以只能在Turf的基础上自行实现符合业务需求的拆分合并功能。

多边形的拆分

基础方案

多边形拆分的核心思想是找到切割点,所以线对面的切割可以简化为线对线的切割。两条线互相切割得到子线段,将子线段互相组合形成多边形。

 

如上图所示,待拆分的多边形记为polygon,切割折线记为splitter。拆分步骤如下:

  • 面化为线:polygon从起点解开可以形成路径为[p0, p1, p2, p3, p0]的折线pline
  • 线互相切割:Turf提供了lineSplit方法,可以使用点或者线将一条折线切分为几部分。利用该方法可以将pline与splitter互相切割,得到子线段集合pieceCollection
  • 线组合为多边形:Turf提供了polygonize方法,将一组折线互相拼接组合成多边形。利用该方法可以将pieceCollection组合成多个多边形splitedCollection

这方案看似可行,实则有以下问题:

  • pline与splitter互相切割后得到的切割点不一致,导致polygonize无法将其拼接在一起
  • 切割线在多边形外的部分会形成外部多边形,如下图所示

解决切割点不一致问题

上文所述第一个切割点不一致的问题是指,使用线A切线B得到的切割点与使用线B切线A得到的切割点不同。

可以看看Turf的源码是如何实现lineSplit的:

function lineSplit(line, splitter) {
    ...

    var lineType = getType(line);
    var splitterType = getType(splitter);

    ...

    // remove excessive decimals from splitter
    // to avoid possible approximation issues in rbush
    var truncatedSplitter = truncate(splitter, {precision: 7});

    switch (splitterType) {
    case 'Point':
        return splitLineWithPoint(line, truncatedSplitter);
    case 'MultiPoint':
        return splitLineWithPoints(line, truncatedSplitter);
    case 'LineString':
    case 'MultiLineString':
    case 'Polygon':
    case 'MultiPolygon':
        return splitLineWithPoints(line, lineIntersect(line, truncatedSplitter));
    }
}

代码中truncate方法是用于保留指定位数的小数,即splitter被限制了精度,所以pline和splitter交换位置后实际计算中的坐标点就发生了变化,导致了不一致的问题。

如何保证两者一致?可以发现用线B切线A时,实际上是先计算线B与线A的交点,再使用splitLineWithPoints方法用这些交点对线A进行切割。那么先计算好两条线的交点,再用交点分别对两条线进行切割,就可以保证切割点的一致了。实现方法如下:

// truncate
let truncatedSplitter = truncate(splitter, {precision: 7});

// compute intersects of two lines
let intersectCollection = lineIntersect(outerLine, truncatedSplitter);
if (intersectCollection.features.length < 2) {
    return null;
}

// transform FeatureCollection[Point] to Feature[MultiPoint]
let intersectCombined = combine(intersectCollection).features[0];

// split lines with points
let outerPieceCollection = lineSplit(outerLine, intersectCombined);
let splitterPieceCollection = lineSplit(truncatedSplitter, intersectCombined);

// polygonize pieces
let pieceCollection = featureCollection(outerPieceCollection.features.concat(splitterPieceCollection.features));
let polygonCollection = polygonize(pieceCollection);

解决外部多边形问题

简单来说只要能筛选出在原大多边形内部的小多边形就好了,Turf提供了booleanContains、booleanDisjoint、booleanWithin等方法用于判断点、线、面的位置关系。但是由于小多边形的部分顶点是在原多边形的边线上计算出来的,且精度有限,位置关系非常微妙,计算时其落在多边形内外都有可能,所以误判率极高。

但是多边形的形心就没有这个问题了,在当前的场景下,我们无需判断小多边形的每个顶点是否都落在原多边形内,只要其形心落在原多边形内即可。

 

实现如下:

// filter polygons in outer poly
let innerPolygons = polygonCollection.features.filter(polygon => {
    let center = centroid(polygon);
    return booleanWithin(center, outerPolygon);
});

环多边形的拆分

环多边形是指内部带“洞”的多边形,其拆分时有两种情况,一是拆分线穿过了洞,那么洞就变成了外轮廓,二是拆分线没有穿过洞,那么洞还整个保留。但是这样的思考方式容易引导我们去将洞也进行拆分,然后再与外环拆分后的片段进行拼接。

还能有更简单的做法,将洞作为遮罩。即在拆分时只对外环多边形进行拆分,在拆分完成之后对小多边形进行遮罩剔除。如下图所示。

遮罩的剔除可以使用Turf的difference方法,具体实现如下:

let diffedPolygons = innerPolygons.map(polygon => {
    let diff = polygon;
    featureEach(holeCollection, (hole) => {
        diff = difference(diff, hole);
    });

    return diff;
});

至此即可完成多边形的拆分功能啦。

多边形的合并

turf.union

Turf提供union方法可以对有交集的多边形进行合并,可以处理完全共边、部分压盖、包含的情况,环多边形也是可以处理的。但是在处理部分共边的多边形时,仍然存在点、线关系判定没有容限的问题,导致点被判定在线外而无法完全合并。

这里先简单介绍一下判断点、线段关系的计算方法,用P表示点,S0和S1两点构成线段,那么首先判断向量P-S0和S1-S0的叉积(叉积表示其构成平行四边形的面积)是否为0,然后判断P是否在S0、S1两点之间。问题就出在叉积是否为0这一步,由于点坐标都是高精度浮点数,叉积很难严格等于0,一般会设定一个较小容限值,只要叉积绝对值小于容限值即可判定为点在线上。

部分共边多边形的合并

已定位合并失败的原因,但是没办法直接修改union的源码,因为Turf在union的实现上其实也使用了外部库martinez-polygon-clipping。不过可以转换思维方式,将部分共边的情况转换为完全共边,再交给union进行合并。这个转换过程我将其称为点注入,将多边形B的顶点注入到多边形A中,即遍历B的顶点进行判断,若其在A的某个线段上且不是线段端头,就将其插入到A的路径中。A与B互相注入顶点之后,所有部分共边的边线都变成完全共边了。

实现过程如下,其中没有使用booleanPointOnLine,而是基于其实现了isPointOnLine,一方面在点线关系判断时加入了容限值,同时排除了所有的端点,另一方面返回值里不仅包含了bool说明点是否在线上,同时还有index属性说明点在线的哪个线段上,以方便将其插入路径中:

/**
 * 将点注入到线上
 * @param {Feature[LineString]} line 
 * @param {FeatureCollection} pointCollection 
 */
function injectPointsOnSide(line, pointCollection) {
    let coords = getCoords(line);
    featureEach(pointCollection, (point, index) => {
        let isOnLine = isPointOnLine(point, line, {
            ignoreVertices: true
        });
        if (isOnLine.bool) {
            coords.splice(isOnLine.index + 1, 0, getCoord(point));
        }
    });
}

至此即可完成多边形的合并功能啦。

产品相关

在JSAPI GL上实现的图形编辑器集成了几何图形的绘制、编辑、删除功能,相较于JSAPI v2功能更加完善且便于使用。该功能已上线官网,欢迎使用~ JavaScript API GL | 腾讯位置服务

注:GIF图片较模糊且闪烁,不代表真实效果。

动图封面  

 

❝ 以下内容转载自totoro的文章《几何计算-基于Turf.js实现多边形的拆分及合并》
作者:totoro
链接:https://blog.totoroxiao.com/g...
来源:https://blog.totoroxiao.com/

总结

各位读者,如果觉得看完对你有帮助的话, 希望你不要吝啬你手中的 ,点个 和关注是对我最大的支持,知识输出不容易,而我勿忘初心,持续分享可视化的好文章,如果你对可视化感兴趣,你可以关注下面我的可视化专栏,或者可以关注我的公众号: 「前端图形」,持续分享计算机图形学知识。

发布于 2021-07-13 20:38

标签:多边形,切割,合并,js,let,拆分,Turf
From: https://www.cnblogs.com/sexintercourse/p/17761561.html

相关文章

  • js_下来菜单自动补全
    <scripttype="text/javascript">   //需要添加的数据内容,可以通过ajax请求获取   //定义加载部品番号的函数   functionloadcities(){       varurl="rawedge.aspx?action=get_matnrlist";       $.get(url,function(data){    ......
  • leaflet使用heatmap.js出现heatmap.js:527 Uncaught TypeError: Cannot assign to rea
    一、问题背景问题是这样发生的,因为项目中需要实现热力图的功能,所以使用了第三方的库heatmap.js。但是在一些浏览器中使用它时,会出现这个错误:>UncaughtTypeError:Cannotassigntoreadonlyproperty'data'ofobject'#<ImageData>'出现问题的原因是因为img.data=im......
  • Python中生成GeoJSON数据
    要在Python中生成GeoJSON数据,可以使用GeoJSON库,例如geojson库或geopandas库。以下是使用这些库生成GeoJSON数据的简单示例:使用geojson库生成GeoJSON数据首先,确保已安装了geojson库,可以使用pip进行安装:pipinstallgeojson然后,可以按照以下方式生成GeoJSON数据:importgeojso......
  • Skywalking APM监控系列(二、Mysql、Linux服务器与前端JS接入Skywalking监听)
    前言上篇我们介绍了Skywalking的基本概念与如何接入.NetCore项目,感兴趣可以去看看:SkywalkingAPM监控系列(一丶.NET5.0+接入Skywalking监听)本篇我们主要讲解一下Skywalking如何接入mysql数据库监听与Linux服务器的监听其实从Skywalking设计之初应该只是单独的链路跟踪,发......
  • 动态的中国地图代码--json+echarts实现
    动态获取后台数据(list类型)的那种--终于会啦~~~~~~~<!DOCTYPEhtml><htmllang="en"><head><metacharset="utf-8"><title>ECharts</title><!--引入echarts.js--><scriptsrc="/js/echarts.min.js&quo......
  • JS堆、栈以及事件循环的概念
    前言其实一开始对栈、堆的概念特别模糊,只知道好像跟内存有关,又好像事件循环也沾一点边。面试薄荷的时候,面试官正好也问到了这个问题,当时只能大方的承认不会。痛定思痛,回去好好的研究一番。我们将从JS的内存机制以及事件机制和大量的(例子)来了解栈、堆究竟是个什么玩意。概念比较多......
  • 4 PyExecJS模块
    PyExecJS模块pyexecjs是一个可以帮助我们运行js代码的一个第三方模块.其使用是非常容易上手的.但是它的运行是要依赖能运行js的第三方环境的.这里我们选择用node作为我们运行js的位置.1.1安装Nodejs切记.重启pycharm或者重启电脑.1.2安装pyexecjspipinstall......
  • 21 JSONP
    JSONP为了解决浏览器跨域问题.jquery提供了jsonp请求.在网页端如果见到了服务器返回的数据是:​ xxxxxxxxxxdjsfkldasjfkldasjklfjadsklfjasdlkj({json数据})​ 在Preview里面可以像看到json一样去调试這就是jsonp。这东西依然是ajax.jsonp的逻辑是.在发送请求的时候.带......
  • js封装获取当前周数据
     /**@Author:张大碗zhangjunhui@mangocosmos.com*@Date:2023-09-2017:36:15*@LastEditors:张大碗zhangjunhui@mangocosmos.com*@LastEditTime:2023-10-0811:04:08*@FilePath:\vue-vant2-template-master\vue-vant2-template-master\src\utils\week.j......
  • golang map json 结构体
    要将JSON转换为Go结构体,您可以使用json.Unmarshal()函数。首先,您需要定义一个与JSON数据结构匹配的Go结构体,然后使用json.Unmarshal()将JSON数据解码为该结构体。以下是一个示例:假设有如下JSON数据:{"name":"JohnDoe","age":30,"email":"johndoe@example.com"}......