Как буферизовать точку (географию) на 1 метр в Entity Framework Core с помощью Net Topology Suite

Я использую Entity Framework Core (3.1.0) с пакетами Net Topology Suite как рекомендуется здесь, и мне очень трудно буферизовать точку, чтобы создать круговой многоугольник вокруг нее. Если я передам значение 1 в Buffer(), где я полагаю, что это означает 1 метр, я получу круг с радиусом, близким к 120 милям.

Какое значение я должен использовать для представления 1 метра? Я использую GeometryFactory с SRID 4326 для создания точек и полигонов. Вот примерно код, который я использую:

GeometryFactory Geography = NtsGeometryServices.Instance.CreateGeometryFactory(4326);

var point = Geography.CreatePoint(coordinate.Latitude, coordinate.Longitude);
var polygonCoordinates = point.Buffer(1).Normalized().Reverse().Coordinates;
var polygon = Geography.CreatePolygon(polygonCoordinates);

person Gup3rSuR4c    schedule 02.01.2020    source источник
comment
В документации EF говорится, что SRID не имеет значения для клиентской стороны — NTS игнорирует значения SRID во время операций. Предполагается плоская система координат, т.е. 1 означает один градус. В нем есть несколько советов, что делать: docs.microsoft.com/en -us/ef/core/modeling/spatial   -  person Michael Entin    schedule 02.01.2020


Ответы (1)


После того, как я опубликовал свой вопрос, я решил снова прочитать Spatial docs для EF Core. Как отметил @Michael Entin в комментариях, в документах указано, что на стороне клиента NTS игнорирует SRID при выполнении вычислений. Ваш ввод должен быть спроецирован в другую систему координат, рассчитан и спроецирован обратно. В моем случае мне пришлось перейти с 4326 на 2855.

ПРИМЕЧАНИЕ: 2855 — это преобразование США в метры, для футов используйте 2926. Для пространственных привязок смотрите здесь, а их визуализации смотрите здесь .

Я просмотрел пример кода и решил скопировать его в LINQPad, чтобы изучить его более подробно. В настоящее время образец кода "работает" но только на геометрии с одной координатой. Если вы передаете геометрию многоугольника, которая имеет несколько координат (например, используемый мной буферный многоугольник), то он просто преобразует первый и игнорирует остальные. Вы должны перебрать оставшиеся координаты, сделать из них точки, преобразовать точки и получить из них координаты.

После еще нескольких проб и ошибок я обновил код примера, чтобы он мог обрабатывать одно- и многокоординатные геометрии. Проверка преобразованных координат на Google Maps, похоже, подтверждает их правильность.

Вот обновленный код проекции для тех, кто еще борется с этим:

public static class CoordinateExtensions {
    public static Coordinate ProjectTo(
        this Coordinate coordinate,
        int fromSrid,
        int toSrid) {
        var point = new Point(coordinate) {
            SRID = fromSrid
        };

        return point.ProjectTo(toSrid).Coordinate;
    }
}

public static class GeometryExtensions {
    private static readonly CoordinateSystemServices Services = new CoordinateSystemServices(new Dictionary<int, string> {
        { 4326, GeographicCoordinateSystem.WGS84.WKT },
        { 2855, @"
            PROJCS[""NAD83(HARN) / Washington North"",
                GEOGCS[""NAD83(HARN)"",
                    DATUM[""NAD83_High_Accuracy_Reference_Network"",
                        SPHEROID[""GRS 1980"",6378137,298.257222101,
                            AUTHORITY[""EPSG"",""7019""]],
                        TOWGS84[0, 0, 0, 0, 0, 0, 0],
                        AUTHORITY[""EPSG"", ""6152""]],
                    PRIMEM[""Greenwich"", 0,
                        AUTHORITY[""EPSG"", ""8901""]],
                    UNIT[""degree"", 0.0174532925199433,
                        AUTHORITY[""EPSG"", ""9122""]],
                    AUTHORITY[""EPSG"", ""4152""]],
                PROJECTION[""Lambert_Conformal_Conic_2SP""],
                PARAMETER[""standard_parallel_1"", 48.73333333333333],
                PARAMETER[""standard_parallel_2"", 47.5],
                PARAMETER[""latitude_of_origin"", 47],
                PARAMETER[""central_meridian"", -120.8333333333333],
                PARAMETER[""false_easting"", 500000],
                PARAMETER[""false_northing"", 0],
                UNIT[""metre"", 1,
                    AUTHORITY[""EPSG"", ""9001""]],
                AXIS[""X"", EAST],
                AXIS[""Y"", NORTH],
                AUTHORITY[""EPSG"", ""2855""]]
        " }
    });

    public static Geometry ProjectTo(
        this Geometry geometry,
        int toSrid) {
        if (geometry.Coordinates.Length == 1) {
            return geometry.ProjectToSingle(toSrid);
        }

        return geometry.ProjectToMany(toSrid);
    }

    private static Geometry ProjectToSingle(
        this Geometry geometry,
        int toSrid) {
        var transformation = Services.CreateTransformation(geometry.SRID, toSrid);
        var transformer = new MathTransformFilter(transformation.MathTransform);
        var transformed = geometry.Copy();

        transformed.Apply(transformer);
        transformed.SRID = toSrid;

        return transformed;
    }

    private static Geometry ProjectToMany(
        this Geometry geometry,
        int toSrid) {
        var fromSrid = geometry.SRID;
        var transformed = geometry.Copy();

        for (var i = 0; i < transformed.Coordinates.Length; i++) {
            var coordinate = transformed.Coordinates[i].ProjectTo(fromSrid, toSrid);

            transformed.Coordinates[i].CoordinateValue = coordinate.CoordinateValue;
        }

        transformed.SRID = toSrid;

        return transformed;
    }
}

public sealed class MathTransformFilter :
    ICoordinateSequenceFilter {
    private readonly MathTransform Transform;

    public MathTransformFilter(
        MathTransform transform) => Transform = transform;

    public bool Done => false;
    public bool GeometryChanged => true;

    public void Filter(
        CoordinateSequence seq,
        int i) {
        var result = Transform.Transform(new[] {
            seq.GetOrdinate(i, Ordinate.X),
            seq.GetOrdinate(i, Ordinate.Y)
        });

        seq.SetOrdinate(i, Ordinate.X, result[0]);
        seq.SetOrdinate(i, Ordinate.Y, result[1]);
    }
}
person Gup3rSuR4c    schedule 02.01.2020