// 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;
}
}