// 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.ComponentModel;
using System.Linq;
using Deltares.DamEngine.Data.Design;
using Deltares.DamEngine.Data.General.Gauges;
using Deltares.DamEngine.Data.General.PlLines;
using Deltares.DamEngine.Data.General.TimeSeries;
using Deltares.DamEngine.Data.Geotechnics;
using Deltares.DamEngine.Data.Properties;
using Deltares.DamEngine.Data.Standard;
using Deltares.DamEngine.Data.Standard.Language;
using Deltares.DamEngine.Data.Standard.Logging;
using Deltares.DamEngine.Data.Standard.Validation;
namespace Deltares.DamEngine.Data.General;
///
/// Class holding all info for a Dike.
///
public class Dike
{
private readonly List databaseSoils = new List();
private IList locations;
private IList pl1Lines;
private SoilList soilList;
private IList soilProfiles;
private IList soilProfiles2D;
///
/// Initializes a new instance of the class.
///
public Dike()
{
Name = "Dijkring";
locations = new List();
soilProfiles = new List();
soilProfiles2D = new List();
SurfaceLines2 = new List();
pl1Lines = new List();
soilList = new SoilList();
}
public virtual string Name { get; set; }
///
/// Gets the locations.
///
///
/// The locations.
///
public virtual IList Locations
{
get
{
return locations;
}
private set
{
locations = value;
}
}
public IList SurfaceLines2 { get; set; }
public virtual IList PL1Lines
{
get
{
return pl1Lines;
}
set
{
pl1Lines = value;
}
}
public virtual IList SoilProfiles
{
get
{
return soilProfiles;
}
set
{
soilProfiles = value;
}
}
///
/// Gets or sets the soilprofiles2d.
///
///
/// The soilprofiles2d.
///
public virtual IList SoilProfiles2D
{
get
{
return soilProfiles2D;
}
set
{
soilProfiles2D = value;
}
}
public virtual SoilList SoilList
{
get
{
return soilList;
}
set
{
soilList = value;
}
}
///
/// Gets or sets the degree of consolidation per material due to the traffic load.
///
public virtual IList TrafficLoadDegreeOfConsolidations { get; set; }
[Browsable(false)] public virtual IList Gauges { get; set; } = new List();
[Browsable(false)] public virtual IList GaugePlLines { get; set; } = new List();
public bool UsesGauges
{
get
{
return GaugePlLines != null && GaugePlLines.Count > 0 && Gauges != null;
}
}
public string Description { get; set; } = "";
///
/// Gets or sets the input time serie collection.
///
///
/// Input time series for operational use
///
public TimeSerieCollection InputTimeSerieCollection { get; set; } = null;
///
/// For all locations, clears all the scenarios except for the first one.
///
public void ClearLocationScenariosExceptFirst()
{
foreach (Location location in Locations)
{
DesignScenario scenario = location.Scenarios[0].CloneInput();
location.Scenarios.Clear();
location.Scenarios.Add(scenario);
}
}
///
/// Sorts the locations.
///
public void SortLocations()
{
locations = Locations.OrderBy(o => o.Name).ToList();
}
public void Validate(bool isOperational, bool isPiping)
{
if (Locations == null || Locations.Count < 1)
{
throw new DikeException("The dike ring has no locations defined");
}
foreach (Location location in Locations)
{
ValidateLocationForPiping(isPiping, location);
if (location.SurfaceLine != null)
{
var validator = new SurfaceLine2Validator();
ValidationResult[] validationResults = validator.ValidateCharacteristicPointsAreOrdered(location.SurfaceLine)
.Concat(validator.ValidateGeometryPointsAreOrdered(location.SurfaceLine)).ToArray();
if (validationResults.Length > 0)
{
throw new SurfaceLineException(validationResults[0].Text);
}
}
if (location.Scenarios.Count < 1)
{
throw new DikeException("Location " + location.Name + " has no scenarios, at least one scenario is required.");
}
ValidateLocationForOperational(isOperational, location);
}
}
private static void ValidateLocationForOperational(bool isOperational, Location location)
{
if (!isOperational)
{
return;
}
if (location.Scenarios.Count > 1)
{
throw new DikeException($"For Operational (DamLive), location {location.Name} has {location.Scenarios.Count} scenarios but only one is allowed!");
}
if ((location.SensorLocation != null) && location.SensorLocation.SensorGroup.SensorArray.Any(sensorArray => sensorArray.RelativeLocation < 0))
{
throw new DikeException($"For Operational (DamLive), location {location.Name} has sensor which are not relative to the starting point.");
}
}
private static void ValidateLocationForPiping(bool isPiping, Location location)
{
if (!isPiping)
{
return;
}
if (location.Segment == null)
{
throw new DikeException(string.Format(Resources.LocationHasNoSegmentDefined, location.Name));
}
if (location.Segment.SoilProfileProbabilities.Count < 1)
{
throw new DikeException(string.Format(Resources.LocationHasNoSoilProfilesDefined, location.Name));
}
if (location.Segment.SoilProfileProbabilities[0].SoilProfile2D != null)
{
throw new DikeException(Resources.PipingWith2DGeometryIsNotSupported);
}
}
///
/// Adapt data so it is consistent
///
public List MakeDataConsistent()
{
List errorSoils = TryToMakeSoilDataConsistent();
var logMessages = new List();
// Delete all locations without surfaceline
logMessages.AddRange(DeleteLocationsWithoutSurfaceLines());
// Delete all locations that have profiles (in their segment) which hold soils
// that are not in the soil database and so have no parameters.
logMessages.AddRange(DeleteLocationsWithProfilesWithUnknownSoils(errorSoils));
return logMessages;
}
public void UpdateLocation(Location location)
{
location.SoilList = SoilList;
location.Gauges.Clear();
location.Gauges.AddRange(Gauges);
location.GaugePlLines.Clear();
location.GaugePlLines.AddRange(GaugePlLines);
}
public override string ToString()
{
return Name;
}
///
/// Tries to make the soil data as read for 1D profiles consistent with the data in the soil database.
/// In the end we have a neat soil list with parameters for every soil as read from the database.
/// We might have a list with errors (soils that were not to be found in the database so soils for which
/// no parameters could be found).
///
///
private List TryToMakeSoilDataConsistent()
{
var errorSoils = new List();
// Fill the list of errorSoils with soils that are in the current soillist (as result of importing
// 1D profiles) but that are not found in the soil database because that are errors
foreach (Soil soil in soilList.Soils)
{
Soil fs = databaseSoils.Find(t => String.Equals(t.Name, soil.Name, StringComparison.CurrentCultureIgnoreCase));
if (fs == null)
{
errorSoils.Add(soil);
}
}
// Remove the error soils form the list
foreach (Soil errorSoil in errorSoils)
{
soilList.Soils.Remove(errorSoil);
}
// Get the parameters for every soil in the now proper soil list from the database. Add soils
// that are in the database but not yet in the soil list.
foreach (Soil soil in databaseSoils)
{
Soil existingSoil = soilList.GetSoilByName(soil.Name);
if (existingSoil == null)
{
soilList.Soils.Add(soil);
}
else
{
existingSoil.Assign(soil);
}
}
return errorSoils;
}
///
/// Removes all locations which have profiles that have invalid soils
///
private List DeleteLocationsWithProfilesWithUnknownSoils(List invalidSoils)
{
var logMessages = new List();
var invalidLocations = new List();
string soilProf;
string invSoil;
foreach (Location location in locations)
{
bool isInValid;
var message = "";
if (location.Segment == null)
{
isInValid = true;
message = String.Format(LocalizationManager.GetTranslatedText(GetType(), "LocationWitNameHasNoSegment"), location.Name);
}
else
{
isInValid = DoesLocationHaveInvalidSoils(invalidSoils, location, out soilProf, out invSoil);
if (isInValid)
{
message = String.Format(LocalizationManager.GetTranslatedText(GetType(), "locationHasProfileWithInvalidSoils"), location.Name, soilProf, invSoil);
}
}
if (isInValid)
{
invalidLocations.Add(location);
logMessages.Add(new LogMessage(LogMessageType.Warning, this, message));
}
}
foreach (Location invalidLocation in invalidLocations)
{
locations.Remove(invalidLocation);
}
return logMessages;
}
///
/// Checks wether a location (or rather the soilprofiles in the segement of the location) contains invalid soils.
/// A soil is hereby considered invalid if it is not found in the database.
///
///
///
///
///
///
private bool DoesLocationHaveInvalidSoils(List invalidSoils, Location location, out string soilProf, out string invSoil)
{
soilProf = " ";
invSoil = " ";
foreach (SoilGeometryProbability spp in location.Segment.SoilProfileProbabilities)
{
foreach (Soil invalidSoil in invalidSoils)
{
SoilLayer1D fl = spp.SoilProfile1D.Layers.Find(l => String.Equals(l.Soil.Name, invalidSoil.Name, StringComparison.CurrentCultureIgnoreCase));
if (fl != null)
{
soilProf = spp.SoilProfile1D.Name;
invSoil = invalidSoil.Name;
return true;
}
}
}
return false;
}
///
/// Delete all locations without surfacelines
///
private List DeleteLocationsWithoutSurfaceLines()
{
var logMessages = new List();
//Add all locations with valid surfaceline
var newLocations = new List();
newLocations.AddRange(Locations.Where(loc => loc.SurfaceLine != null));
// Report which locations are not added because no valid surfaceline is found
var deletedLocations = new List();
deletedLocations.AddRange(Locations.Where(loc => loc.SurfaceLine == null));
foreach (Location deletedLocation in deletedLocations)
{
string locationHasNoSurfaceLine = LocalizationManager.GetTranslatedText(GetType(), "LocationHasNoSurfaceLine");
logMessages.Add(new LogMessage(LogMessageType.Warning, this,
String.Format(locationHasNoSurfaceLine, deletedLocation.Name)));
}
Locations = newLocations;
return logMessages;
}
}