webgis端通用的几何要素LineString分割Polygon

完整代码类如下:

import * as turf from '@turf/turf'

/**
 * geoJson数据处理模块
 * (需要引入turf.js)
 * 输入输出数据均为标准geoJson格式
 */
const geoClipUtil = {
    /**
     * 合并多边形
     */
    unionPolygon: function (polygons) {
        var polygon = polygons[0];
        for (var i = 0; i < polygons.length; i++) {
            polygon = turf.union(polygon, polygons[i]);
        };
        return polygon;
    },

    /**
     * 线分割面
     * 面类型只能是polygon或是环
     * 线与多边形必须有两个交点
     */
    polygonClipByLine: function (polygon, clipLine) {
        if (polygon.geometry.type === 'Polygon') {
            var polyLine = turf.polygonToLine(polygon)
            if (polyLine.geometry.type === 'LineString') { // 切割普通多边形
                return this.singlePolygonClip(polyLine, clipLine);
            } else if (polyLine.geometry.type === 'MultiLineString') { //切割环
                return this.multiPolygonClip(polyLine, clipLine);
            }
        } else if (polygon.geometry.type === 'MultiPolygon') {
            // 若输入的多边形类型为Multipolygon则拆分成多个Polygon
            var polygons = this.multiPolygon2polygons(polygon)
            var clipPolygon = null;
            var clipPolygonIndex = -1
            // 获取MultiPolygon中与切割线相交的多边形(有且只能有一个多边形相交2个交点)
            polygons.forEach(function (polygon, index) {
                var polyLine = turf.polygonToLine(polygon)
                if (turf.lineIntersect(polyLine, clipLine).features.length === 2) {
                    if (!clipPolygon) {
                        clipPolygon = polygon
                        clipPolygonIndex = index
                    } else {
                        return { state: 'error', message: 'MultiPolygon只能有一个多边形与切割线存在交点' };
                    }
                }
            })
            if (clipPolygonIndex !== -1) {
                polygons.splice(clipPolygonIndex, 1)
                return turf.featureCollection(polygons.concat(this.polygonClipByLine(clipPolygon, clipLine).features));
            } else {
                return { state: 'error', message: 'MultiPolygon与切割线无交点' };
            }

        } else {
            return { state: 'error', message: '输入的多边形类型为错误' };
        }
    },


    /**
     * 线分割面
     * 切割普通多边形
    */
    singlePolygonClip: function (polyLine, clipLine) {
        // 获得裁切点
        var intersects = turf.lineIntersect(polyLine, clipLine);
        if (intersects.features.length !== 2) {
            return { state: 'error', message: '切割线与多边形交点应该为2个,当前交点个数为' + intersects.features.length };
        }
        // 检查切割线与多边形的位置关系 (切割线的起点和终点不能落在多边形内部)
        var clipLineLength = clipLine.geometry.coordinates.length;
        var clipLineStartPoint = turf.point(clipLine.geometry.coordinates[0])
        var clipLineEndPoint = turf.point(clipLine.geometry.coordinates[clipLineLength - 1])
        var polygon = turf.polygon([polyLine.geometry.coordinates])
        if (turf.booleanPointInPolygon(clipLineStartPoint, polygon) || turf.booleanPointInPolygon(clipLineEndPoint, polygon)) {
            return { state: 'error', message: '切割线起点或终点不能在 裁剪多边形内部' };
        }
        // 通过裁切点 分割多边形(只能获得多边形的一部分)
        var slicedPolyLine = turf.lineSlice(intersects.features[0], intersects.features[1], polyLine);
        // 裁剪线分割 保留多边形内部部分
        var slicedClipLine = turf.lineSlice(intersects.features[0], intersects.features[1], clipLine);
        // 重新拼接多边形 存在 对接的问题 所以先进行判断 如何对接裁剪的多边形和裁剪线
        var resultPolyline1 = this.connectLine(slicedPolyLine, slicedClipLine)
        resultPolyline1.properties.clipName = 'clip_01'
        // 闭合线 来构造多边形
        resultPolyline1.geometry.coordinates.push(resultPolyline1.geometry.coordinates[0])
        var resultPolygon1 = turf.lineToPolygon(resultPolyline1);
        // 构造切割的另一面多边形
        var firstPointOnLine = this.isOnLine(turf.point(polyLine.geometry.coordinates[0]), slicedPolyLine);
        var pointList = [];
        if (firstPointOnLine) {
            for (var i = 0; i < polyLine.geometry.coordinates.length; i++) {
                var coordinate = polyLine.geometry.coordinates[i];
                if (!this.isOnLine(turf.point(coordinate), slicedPolyLine)) {
                    pointList.push(coordinate)
                }
            };
        } else {
            var skipNum = 0; // 记录前面被跳过的点的个数
            var isStartPush = false;
            for (var i = 0; i < polyLine.geometry.coordinates.length; i++) {
                var coordinate = polyLine.geometry.coordinates[i];
                if (!this.isOnLine(turf.point(coordinate), slicedPolyLine)) {
                    if (isStartPush) {
                        pointList.push(coordinate)
                    } else {
                        skipNum++
                    }

                } else {
                    isStartPush = true;
                }
            };
            // 将前面跳过的点补充到 点数组中
            for (var i = 0; i < skipNum; i++) {
                pointList.push(polyLine.geometry.coordinates[i])
            }
        }
        var slicedPolyLine_2 = turf.lineString(pointList, slicedPolyLine.properties);
        var resultPolyline2 = this.connectLine(slicedPolyLine_2, slicedClipLine)
        resultPolyline2.properties.clipName = 'clip_02'
        // 闭合线 来构造多边形
        resultPolyline2.geometry.coordinates.push(resultPolyline2.geometry.coordinates[0])
        var resultPolygon2 = turf.lineToPolygon(resultPolyline2);
        // 返回面要素集
        return {
            features: turf.featureCollection([
                resultPolygon1,
                resultPolygon2
            ]), state: 'success'
        };

    },

    /**
     * 线分割面
     * 切割环
    */
    multiPolygonClip: function (polyLine, clipLine) {
        // 将环 多边形分割成 内部逆时针多边形+外部多边形
        var outPolyline, insidePolylineList = [];
        for (var i = 0; i < polyLine.geometry.coordinates.length; i++) {
            var splitPolyline = turf.lineString(polyLine.geometry.coordinates[i]);
            if (turf.booleanClockwise(splitPolyline)) {
                if (outPolyline) {
                    return { state: 'error', message: '出现了两个外部多边形无法处理' };
                } else {
                    outPolyline = splitPolyline
                }
            } else {
                var intersects = turf.lineIntersect(splitPolyline, clipLine);
                if (intersects.features.length > 0) {
                    return { state: 'error', message: '切割线不能与内环有交点' };
                }
                insidePolylineList.push(splitPolyline)
            }
        }
        var resultCollection = this.singlePolygonClip(outPolyline, clipLine)

        for (var i = 0; i < resultCollection.features.length; i++) {
            for (var j = 0; j < insidePolylineList.length; j++) {
                var startPoint = turf.point(insidePolylineList[j].geometry.coordinates[0]);
                if (turf.booleanPointInPolygon(startPoint, resultCollection.features[i])) {
                    resultCollection.features[i] = turf.mask(resultCollection.features[i], turf.lineToPolygon(insidePolylineList[j]));
                }
            }
        }
        return resultCollection
    },

    /**
     * 连接两条线
     * 方法会将两条线段最近的一段直接连接
     */
    connectLine: function (line1, line2) {
        var line2_length = line2.geometry.coordinates.length;
        var line1_startPoint = line1.geometry.coordinates[0]
        var line2_startPoint = line2.geometry.coordinates[0]
        var line2_endPoint = line2.geometry.coordinates[line2_length - 1]
        var pointList = [];
        // 获取line1 所有点坐标
        for (var i = 0; i < line1.geometry.coordinates.length; i++) {
            var coordinate = line1.geometry.coordinates[i];
            pointList.push(coordinate)
        };

        // 判断两条线的 起点是否接近,如果接近 逆转line2线 进行连接
        if (turf.distance(line1_startPoint, line2_startPoint) < turf.distance(line1_startPoint, line2_endPoint)) {
            line2.geometry.coordinates = line2.geometry.coordinates.reverse();
        }
        for (var i = 0; i < line2.geometry.coordinates.length; i++) {
            var coordinate = line2.geometry.coordinates[i];
            pointList.push(coordinate)
        };
        return turf.lineString(pointList, line1.properties);
    },

    /**
     * 判断点是否在线里面
     * 注:线组成的坐标对比
     */
    isOnLine: function (point, line) {
        for (var i = 0; i < line.geometry.coordinates.length; i++) {
            var coordinate = line.geometry.coordinates[i];
            if (point.geometry.coordinates[0] === coordinate[0] && point.geometry.coordinates[1] === coordinate[1]) {
                return true;
            }
        };
        return false;
    },

    /**
     * 获得两条线交点
     */
    getIntersectPoints: function (line1, line2) {
        return turf.lineIntersect(line1, line2);
    },

    /**
     * multiPolygon转polygons
     */
    multiPolygon2polygons: function (multiPolygon) {
        if (multiPolygon.geometry.type !== 'MultiPolygon') {
            return
        }
        var polygons = [];
        multiPolygon.geometry.coordinates.forEach((item) => {
            var polygon = {
                type: 'Feature',
                properties: {},
                geometry: {
                    type: 'Polygon',
                    coordinates: []
                }
            };
            polygon.geometry.coordinates = item;
            polygons.push(polygon)
        });
        return polygons;
    },

    /**
    * polygons转multiPolygon
    * 考虑polygons中就存在多面的情况
    */
    polygons2MultiPolygon: function (geoJson) {
        var newGeoJson = {
            type: "FeatureCollection",
            features: [{ geometry: { coordinates: [], type: "MultiPolygon" }, type: "Feature", properties: {} }]
        };
        geoJson.features.forEach((item) => {
            if (item.geometry.type === "Polygon") {
                newGeoJson.features[0].geometry.coordinates.push(item.geometry.coordinates);
            } else {
                item.geometry.coordinates.forEach((item) => {
                    newGeoJson.features[0].geometry.coordinates.push(item);
                })
            }
        })
        return newGeoJson;
    },
}

export default geoClipUtil

基于turf的相关api链接:GET START | Turf.js中文网

示例:多边形切割demo

上一篇:javascript内置对象:Date


下一篇:配置centos 8 静态IP址