// Copyright (C) Stichting Deltares 2024. All rights reserved. // // This file is part of the Dam Engine. // // The Dam Engine is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . // // All names, logos, and references to "Deltares" are registered trademarks of // Stichting Deltares and remain full property of Stichting Deltares at all times. // All rights reserved. using System; using System.Collections.Generic; using System.Linq; using Deltares.DamEngine.Data.Geometry; using Deltares.DamEngine.Data.Standard; using Deltares.DamEngine.Data.Standard.Language; using Deltares.DamEngine.Data.Standard.Validation; namespace Deltares.DamEngine.Data.Geotechnics; /// /// Validator for . /// public class SurfaceLine2Validator { /// /// Performs all validation checks for a surfaceline. /// /// The surfaceline being evaluated. /// The collection of validation results. public IEnumerable Validate(SurfaceLine2 surfaceline) { return ValidateGeometryPointsAreOrdered(surfaceline) .Concat(ValidateCharacteristicPointsAreOrdered(surfaceline)) .Concat(ValidateDikeShape(surfaceline)); } /// /// Checks if all instances in /// are ordered on ascending X. /// /// The surfaceline being evaluated. /// The collection of validation results. public IEnumerable ValidateGeometryPointsAreOrdered(SurfaceLine2 surfaceline) { if (!ArePointsAscending(surfaceline)) { yield return new ValidationResult(ValidationResultType.Error, this.Translate("SurfacePointsNotAscending"), surfaceline, surfaceline.GetMemberName(sl => sl.Geometry), ""); } } /// /// Checks if all characteristic points that require proper ordering based on X coordinate /// are indeed properly ordered. /// /// The surfaceline being evaluated. /// The collection of validation results. public IEnumerable ValidateCharacteristicPointsAreOrdered(SurfaceLine2 surfaceline) { if (!AreAllCharacteristicPointsXCoordinatesAscending(surfaceline)) { yield return new ValidationResult(ValidationResultType.Error, this.Translate("ChartPointsNotAscending"), surfaceline, surfaceline.GetMemberName(sl => sl.CharacteristicPoints), ""); } } /// /// Are all characteristic points x coordinates ascending. /// /// The line. /// public static bool AreAllCharacteristicPointsXCoordinatesAscending(SurfaceLine2 line) { CharacteristicPoint[] points = line.GetCharacteristicPointsRequiringAscendingX().ToArray(); return AreGeometryPointsXCoordinatesAscending(points); } /// /// Are all points x coordinates ascending. /// /// /// public static bool AreAllPointsXCoordinatesAscending(SurfaceLine2 line) { return AreGeometryPointsXCoordinatesAscending(line.Geometry.Points); } /// /// Determines whether all characteristic points are in correct shape. /// /// true if all characteristic are in correct shape, otherwise false public static bool AreAllCharacteristicPointsInCorrectShape(SurfaceLine2 line) { // when there are no points to check, make sure true is returned. var result = true; if (line.IsDefined(CharacteristicPointType.DikeToeAtRiver) && line.IsDefined(CharacteristicPointType.DikeTopAtRiver)) { result = line.CharacteristicPoints.GetPoint2D(CharacteristicPointType.DikeToeAtRiver).Z < line.CharacteristicPoints.GetPoint2D(CharacteristicPointType.DikeTopAtRiver).Z; } if (line.IsDefined(CharacteristicPointType.DikeToeAtPolder) && line.IsDefined(CharacteristicPointType.DikeTopAtPolder)) { result = result && line.CharacteristicPoints.GetPoint2D(CharacteristicPointType.DikeToeAtPolder).Z < line.CharacteristicPoints.GetPoint2D(CharacteristicPointType.DikeTopAtPolder).Z; } return result; } /// /// Checks if the shape of the dike is proper. /// /// The surfaceline being evaluated. /// The collection of validation results. private IEnumerable ValidateDikeShape(SurfaceLine2 surfaceline) { if (!AreAllCharacteristicPointsInCorrectShape(surfaceline)) { yield return new ValidationResult(ValidationResultType.Error, this.Translate("ImproperDikeShape"), surfaceline); } } private bool ArePointsAscending(SurfaceLine2 line) { return AreGeometryPointsXCoordinatesAscending(line.Geometry.Points); } private static bool AreGeometryPointsXCoordinatesAscending(IList characteristicPoints) { List points = new List(); foreach (CharacteristicPoint characteristicPoint in characteristicPoints) { points.Add(characteristicPoint.Point); } return AreGeometryPointsXCoordinatesAscending(points); } private static bool AreGeometryPointsXCoordinatesAscending(IList points) { for (var i = 1; i < points.Count; i++) { if (points[i].X < points[i - 1].X + GeometryConstants.Accuracy) { // #Bka: in fact we now should return FALSE immediately but there is a snag. Due to faults with adding characteristic // points to a surface line, some surface lines can have identical points (in Location) but with different // characteristic (e.g. diketop and trafficload combined). These points should use the same reference to the point // but it occurs that TWO fysical (and identical) points are added. // So these points must now be ignored (so identical points must be seen as OK). This is in fact ok for the purpose // of the validation as the main problem is that points may never give a vertical part in the geometry. if ((Math.Abs(points[i].X - points[i - 1].X) < GeometryConstants.Accuracy) && (Math.Abs(points[i].Z - points[i - 1].Z) < GeometryConstants.Accuracy)) { // Make sure points are made identical points[i].X = points[i - 1].X; points[i].Z = points[i - 1].Z; } else { return false; } } } return true; } }