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