// Copyright (C) Stichting Deltares 2023. All rights reserved.
//
// This file is part of the application DAM - UI.
//
// DAM - UI is free software: you can redistribute it and/or modify
// it under the terms of the GNU 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 General Public License for more details.
//
// You should have received a copy of the GNU 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.IO;
using System.Linq;
using System.Xml.Serialization;
using Deltares.Dam.Data.CsvImporters;
using Deltares.Geotechnics.Converter;
using Deltares.Geotechnics.Soils;
using Deltares.Geotechnics.SurfaceLines;
using Deltares.Geotechnics.Validation;
using Deltares.Standard;
using Deltares.Standard.Attributes;
using Deltares.Standard.Data;
using Deltares.Standard.Extensions;
using Deltares.Standard.Language;
using Deltares.Standard.Logging;
using Deltares.Standard.Validation;
namespace Deltares.Dam.Data;
public class DikeException : Exception
{
public DikeException(string message)
: base(message) {}
}
public class DikeParameterNames
{
public const string MapForSoilGeometries2D = "MapForSoilGeometries2D";
}
[TrackChanges]
public class Dike : IVisibleEnabled, IDisposable
{
private IList locations;
private IList pl1Lines;
private SoilList soilList;
private IList soilProfiles;
public Dike()
{
Name = "Dijkring";
MapForSoilGeometries2D = "";
locations = new List();
soilProfiles = new List();
surfaceLines = new DelegatedList
{
AddMethod = ConvertAddedOldSurfaceLineToNewFormat
};
SurfaceLines2 = new List();
pl1Lines = new List();
soilList = new SoilList();
IsRemoveStiFiles = true;
}
public virtual string MapForSoilGeometries2D { get; set; }
public bool IsRemoveStiFiles { get; set; }
public MStabShearStrength ShearStrengthModel { get; set; }
public virtual string Name { get; set; }
///
/// Gets the locations.
///
///
/// The locations.
///
public virtual IList Locations
{
get => locations;
private set => locations = value;
}
public IList SurfaceLines2 { get; set; }
public virtual IList PL1Lines
{
get => pl1Lines;
set => pl1Lines = value;
}
public virtual IList SoilProfiles
{
get => soilProfiles;
set => soilProfiles = value;
}
public virtual SoilList SoilList
{
get => soilList;
set => soilList = value;
}
[Browsable(false)] public virtual IList Gauges { get; set; } = new List();
[Browsable(false)] public virtual IList GaugePLLines { get; set; } = new List();
public bool UsesGauges => (GaugePLLines != null) && (GaugePLLines.Count > 0) && (Gauges != null);
public virtual List Scenarios
{
get
{
var scenarios = new List();
foreach (Location location in Locations)
{
scenarios.AddRange(location.Scenarios);
}
return scenarios;
}
}
public string Description { get; set; } = "";
///
/// Updates the locations for scenarios.
///
public void UpdateLocationsForScenarios()
{
foreach (Location location in Locations)
{
foreach (Scenario scenario in location.Scenarios)
{
scenario.Location = location;
}
}
}
///
/// Sorts the locations.
///
public void SortLocations()
{
locations = Locations.OrderBy(o => o.Name).ToList();
}
public void Validate()
{
if ((Locations == null) || (Locations.Count < 1))
{
throw new DikeException("The dike ring has no locations defined");
}
foreach (Location location in Locations)
{
if (location.LocalXZSurfaceLine2 != null)
{
var validator = new SurfaceLine2Validator();
ValidationResult[] validationResults = validator.ValidateCharacteristicPointsAreOrdered(location.LocalXZSurfaceLine2)
.Concat(validator.ValidateGeometryPointsAreOrdered(location.LocalXZSurfaceLine2)).ToArray();
if (validationResults.Length > 0)
{
throw new SurfaceLineException(validationResults[0].Text);
}
}
}
}
///
/// Add to soil data as read from the csv-file to the dike soils
///
public void FillImportedCsvSoilsFromCsvFile(IEnumerable soilRecords)
{
if (soilRecords == null)
{
return;
}
foreach (CsvImporterSoils.SoilRecord soilRecord in soilRecords)
{
var soil = new Soil();
soil.Name = soilRecord.SoilName;
soil.Color = soilRecord.SoilColor;
soil.SoilType = soilRecord.SoilType;
if (soilRecord.UnsaturatedUnitWeight.HasValue)
{
soil.AbovePhreaticLevel = soilRecord.UnsaturatedUnitWeight.Value;
}
if (soilRecord.SaturatedUnitWeight.HasValue)
{
soil.BelowPhreaticLevel = soilRecord.SaturatedUnitWeight.Value;
}
if (soilRecord.Cohesion.HasValue)
{
soil.Cohesion = soilRecord.Cohesion.Value;
}
if (soilRecord.FrictionAngle.HasValue)
{
soil.FrictionAngle = soilRecord.FrictionAngle.Value;
}
if (soilRecord.DiameterD70.HasValue)
{
soil.DiameterD70 = soilRecord.DiameterD70.Value;
}
if (soilRecord.PermeabilityX.HasValue)
{
soil.PermeabKx = soilRecord.PermeabilityX.Value;
}
soil.UseDefaultShearStrengthModel = false;
soil.ShearStrengthModel = soilRecord.ShearStrengthModel;
if (soilRecord.StrengthIncreaseExponent.HasValue)
{
soil.StrengthIncreaseExponent = soilRecord.StrengthIncreaseExponent.Value;
}
if (soilRecord.RatioSuPc.HasValue)
{
soil.RatioCuPc = soilRecord.RatioSuPc.Value;
}
soil.UsePop = soilRecord.UsePop;
if (soilRecord.Pop.HasValue && soilRecord.UsePop)
{
soil.POP = soilRecord.Pop.Value;
}
ImportedCsvSoils.Add(soil);
}
}
///
/// Adapt data so it is consistent
///
/// The soils as imported from the csv file
public IEnumerable MakeDataConsistent(List importedCsvSoils)
{
List errorSoils = TryToMakeSoilDataConsistent(importedCsvSoils);
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 csv-file and so have no parameters.
logMessages.AddRange(DeleteLocationsWithProfilesWithUnknownSoils(errorSoils));
return logMessages;
}
public Dictionary GetParametersAsNameValuePairs()
{
var nameValuePairs = new Dictionary();
nameValuePairs.Add(DikeParameterNames.MapForSoilGeometries2D, MapForSoilGeometries2D);
return nameValuePairs;
}
public void SetParameterFromNameValuePair(string parameterName, string parameterValue)
{
if (parameterName.Equals(DikeParameterNames.MapForSoilGeometries2D))
{
MapForSoilGeometries2D = parameterValue;
}
}
public void UpdateLocation(Location location)
{
location.SoilList = SoilList;
location.MapForSoilGeometries2D = MapForSoilGeometries2D;
location.Gauges.Clear();
location.Gauges.AddRange(Gauges);
location.GaugePLLines.Clear();
location.GaugePLLines.AddRange(GaugePLLines);
}
public override string ToString()
{
return Name;
}
public void Dispose()
{
foreach (Location location in Locations)
{
location.Dispose();
}
foreach (SurfaceLine2 surfaceLine2 in SurfaceLines2)
{
surfaceLine2.Dispose();
}
}
public bool IsVisible(string property)
{
return true;
}
public bool IsEnabled(string property)
{
switch (property)
{
case "Name": return false;
case "SoilList": return false;
default: return true;
}
}
internal List ImportedCsvSoils { get; } = new();
///
/// 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).
///
/// The soils as imported from the csv file
///
private List TryToMakeSoilDataConsistent(List importedCsvSoils)
{
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 = importedCsvSoils.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 importedCsvSoils)
{
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.SoilProfile.Layers.Find(l => string.Equals(l.Soil.Name, invalidSoil.Name, StringComparison.CurrentCultureIgnoreCase));
if (fl != null)
{
soilProf = spp.SoilProfile.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.SurfaceLine2 != null));
// Report which locations are not added because no valid surfaceline is found
var deletedLocations = new List();
deletedLocations.AddRange(Locations.Where(loc => loc.SurfaceLine2 == 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;
}
#region Backwards compatibility
private IList surfaceLines;
///
/// Gets or sets all surface lines instances associated with this dike.
///
///
/// Composite owner of all that are part of the
/// application that are associated with this dike.
///
[Obsolete("Do not use this member; Only exists for backwards compatibility.", true)]
public virtual IList SurfaceLines
{
get => surfaceLines;
set => surfaceLines = value;
}
///
/// Performs backwards compatibility conversion from old-style surfaceline to new-style
/// surfaceline, and ensures that shared references are set correctly.
///
/// Persisted version of the old-style surfaceline.
private void ConvertAddedOldSurfaceLineToNewFormat(SurfaceLine argument)
{
SurfaceLine2 surfaceLine2 = new OldSurfaceLineToNewConverter().Convert(argument);
SurfaceLines2.Add(surfaceLine2);
foreach (Location location in Locations)
{
location.SetNewSurfaceLineIfMatchesWithOldPersistedSurfaceLine(surfaceLine2);
}
// Remove old entry; it is no longer required.
surfaceLines.Remove(argument);
argument.Dispose();
}
#endregion
}