// 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.Collections.Specialized; using System.ComponentModel; using Deltares.DamEngine.Data.General; using Deltares.DamEngine.Data.Geotechnics; using Deltares.DamEngine.Data.Standard.Calculation; using Deltares.DamEngine.Data.Standard.Language; using Deltares.DamEngine.Data.Standard.Validation; namespace Deltares.DamEngine.Data.Design; /// /// Calculation scenario class /// public class DesignScenario { private Dictionary redesignedSurfaceLines; private Dictionary resultMessages; /// /// Constructor /// public DesignScenario() { ClearResults(); ClearErrors(); } public CalculationResult CalculationResult { get; set; } = CalculationResult.NoRun; /// /// Gets or sets the location scenario identifier. /// /// /// The location scenario identifier. /// public string LocationScenarioID { get; set; } /// /// The name of the location to which the scenario belongs /// public string LocationName { get; set; } /// /// Gets or sets the polder level. /// /// /// The polder level. /// public double PolderLevel { get; set; } [Browsable(false)] public StringCollection Errors { get; private set; } /// /// Gets or sets the river level. /// /// /// The river level. /// [Description("River level")] public double RiverLevel { get; set; } /// /// Gets or sets the river level low. /// /// /// The river level low. /// [Description("River level low")] public double? RiverLevelLow { get; set; } /// /// Gets or sets the height of the dike table. /// /// /// The height of the dike table. /// [Description("Dike table height")] public double? DikeTableHeight { get; set; } /// /// Gets or sets the required safety factor stability inner slope. /// /// /// The required safety factor stability inner slope. /// [Description("Required safety factor stability inner slope")] public double RequiredSafetyFactorStabilityInnerSlope { get; set; } /// /// Gets or sets the required safety factor stability outer slope. /// /// /// The required safety factor stability outer slope. /// [Description("Required safety factor stability outer slope")] public double RequiredSafetyFactorStabilityOuterSlope { get; set; } /// /// Gets or sets the required safety factor piping. /// /// /// The required safety factor piping. /// [Description("Required safety factor piping")] public double RequiredSafetyFactorPiping {get; set; } /// /// Gets or sets the uplift criterion piping. /// /// /// The uplift criterion piping. /// [Description("Uplift criterion piping")] public double UpliftCriterionPiping {get; set; } /// /// Gets or sets the uplift criterion stability. /// /// /// The uplift criterion stability. /// [Description("Uplift criterion stability")] public double UpliftCriterionStability { get; set; } public double PlLineOffsetBelowDikeTopAtRiver { get; set; } = 0.5; /// /// Gets or sets the pl line offset below dike top at polder. /// /// /// The pl line offset below dike top at polder. /// public double PlLineOffsetBelowDikeTopAtPolder { get; set; } = 1.5; /// /// Gets or sets the pl line offset below shoulder base inside. /// /// /// The pl line offset below shoulder base inside. /// public double PlLineOffsetBelowShoulderBaseInside { get; set; } = 0.1; /// /// Gets or sets the pl line offset below dike toe at polder. /// /// /// The pl line offset below dike toe at polder. /// public double PlLineOffsetBelowDikeToeAtPolder { get; set; } = 0.1; /// /// Gets or sets the use pl line offset below dike crest middle. /// /// /// The use pl line offset below dike crest middle. /// public bool? UsePlLineOffsetBelowDikeCrestMiddle { get; set; } /// /// Gets or sets the pl line offset below dike crest middle. /// /// /// The pl line offset below dike crest middle. /// public double? PlLineOffsetBelowDikeCrestMiddle { get; set; } /// /// Gets or sets the use pl line offset factor below shoulder crest. /// /// /// The use pl line offset factor below shoulder crest. /// public bool? UsePlLineOffsetFactorBelowShoulderCrest { get; set; } /// /// Gets or sets the pl line offset factor below shoulder crest. /// /// /// The pl line offset factor below shoulder crest. /// public double? PlLineOffsetFactorBelowShoulderCrest { get; set; } /// /// Gets or sets the head PL2. /// /// /// The head PL2. /// public virtual double? HeadPl2 { get; set; } /// /// Gets or sets the head PL3. /// /// /// The head PL3. /// public double? HeadPl3 { get; set; } /// /// Gets or sets the head PL4. /// /// /// The head PL4. /// public double? HeadPl4 { get; set; } /// /// Clone the input part of the current scenario to a new instance /// /// /// public DesignScenario CloneInput() { // Synchronize scenario data (Note: do not attempt to synchronize the ModelFactors as these are set by the direct // properties in the scenario such as the Required Safeties. ModelFactors is just a place holder. var newCurrentScenario = new DesignScenario(); newCurrentScenario.LocationName = LocationName; newCurrentScenario.LocationScenarioID = LocationScenarioID; newCurrentScenario.PolderLevel = PolderLevel; newCurrentScenario.RiverLevel = RiverLevel; newCurrentScenario.RiverLevelLow = RiverLevelLow; newCurrentScenario.DikeTableHeight = DikeTableHeight; newCurrentScenario.RequiredSafetyFactorStabilityInnerSlope = RequiredSafetyFactorStabilityInnerSlope; newCurrentScenario.RequiredSafetyFactorStabilityOuterSlope = RequiredSafetyFactorStabilityOuterSlope; newCurrentScenario.RequiredSafetyFactorPiping = RequiredSafetyFactorPiping; newCurrentScenario.UpliftCriterionPiping = UpliftCriterionPiping; newCurrentScenario.UpliftCriterionStability = UpliftCriterionStability; newCurrentScenario.PlLineOffsetBelowDikeTopAtPolder = PlLineOffsetBelowDikeTopAtPolder; newCurrentScenario.PlLineOffsetBelowDikeTopAtRiver = PlLineOffsetBelowDikeTopAtRiver; newCurrentScenario.PlLineOffsetBelowShoulderBaseInside = PlLineOffsetBelowShoulderBaseInside; newCurrentScenario.PlLineOffsetBelowDikeToeAtPolder = PlLineOffsetBelowDikeToeAtPolder; newCurrentScenario.UsePlLineOffsetBelowDikeCrestMiddle = UsePlLineOffsetBelowDikeCrestMiddle; newCurrentScenario.PlLineOffsetBelowDikeCrestMiddle = PlLineOffsetBelowDikeCrestMiddle; newCurrentScenario.UsePlLineOffsetFactorBelowShoulderCrest = UsePlLineOffsetFactorBelowShoulderCrest; newCurrentScenario.PlLineOffsetFactorBelowShoulderCrest = PlLineOffsetFactorBelowShoulderCrest; newCurrentScenario.HeadPl2 = HeadPl2; newCurrentScenario.HeadPl3 = HeadPl3; newCurrentScenario.HeadPl4 = HeadPl4; return newCurrentScenario; } /// /// Clears the errors. /// public void ClearErrors() { Errors = new StringCollection(); } /// /// Clears the results /// public void ClearResults() { CalculationResult = CalculationResult.NoRun; resultMessages = new Dictionary(); redesignedSurfaceLines = new Dictionary(); } /// /// Determines whether [is pl line offset factor below shoulder crest valid]. /// /// [Validate] public ValidationResult[] IsPlLineOffsetFactorBelowShoulderCrestValid() { if (UsePlLineOffsetFactorBelowShoulderCrest.HasValue && UsePlLineOffsetFactorBelowShoulderCrest.Value && PlLineOffsetFactorBelowShoulderCrest.HasValue) { if (PlLineOffsetFactorBelowShoulderCrest.Value < DamGlobalConstants.PlLineOffsetFactorBelowShoulderCrestMinValue) { return new[] { new ValidationResult(ValidationResultType.Error, LocalizationManager.GetTranslatedText(this, "PlLineOffsetFactorBelowShoulderCrestTooSmall"), this) }; } if (PlLineOffsetFactorBelowShoulderCrest.Value > DamGlobalConstants.PlLineOffsetFactorBelowShoulderCrestMaxValue) { return new[] { new ValidationResult(ValidationResultType.Error, LocalizationManager.GetTranslatedText(this, "PlLineOffsetFactorBelowShoulderCrestTooLarge"), this) }; } } return new ValidationResult[0]; } /// /// Get Result Message /// /// /// /// public string GetResultMessage(SoilProfile1D soilProfile, SoilProfile2D soilProfile2D) { Object objectAsKey = GetObjectAsKey(soilProfile, soilProfile2D); return GetResultMessage(objectAsKey); } /// /// Set Result Message /// /// /// /// public void SetResultMessage(SoilProfile1D soilProfile, SoilProfile2D soilProfile2D, string resultMessage) { Object objectAsKey = GetObjectAsKey(soilProfile, soilProfile2D); SetResultMessage(objectAsKey, resultMessage); } /// /// Set Redesigned surfaceline /// /// /// /// /// soilProfile1D and soilProfile2D can not both be null public void SetRedesignedSurfaceLine(SoilProfile1D soilProfile1D, SoilProfile2D soilProfile2D, SurfaceLine2 surfaceLine) { if ((soilProfile1D == null) && (soilProfile2D == null)) { throw new ArgumentException("Both soilProfile1D and soilProfile2D are null."); } object objectAsKey = GetObjectAsKey(soilProfile1D, soilProfile2D); SetRedesignedSurfaceLine(objectAsKey, surfaceLine); } /// /// Get Most recent surfaceline /// /// /// /// public SurfaceLine2 GetMostRecentSurfaceLine(SoilProfile soilProfile, SoilProfile2D soilProfile2D) { var profile1D = soilProfile as SoilProfile1D; Object objectAsKey = GetObjectAsKey(profile1D, soilProfile2D); return GetMostRecentSurfaceLine(objectAsKey); } /// /// Implementation of ToString /// /// public override string ToString() { return $"Location={LocationName}, ID={LocationScenarioID} RiverLevel={RiverLevel} " + $"RiverLevelLow={(RiverLevelLow.HasValue ? RiverLevelLow.ToString() : "?")} " + $"DikeTableHeight={(DikeTableHeight.HasValue ? DikeTableHeight.ToString() : "?")} " + $"SafetyFactorStabilityInnerSlope={RequiredSafetyFactorStabilityInnerSlope} " + $"RequiredSafetyFactorStabilityOuterSlope={RequiredSafetyFactorStabilityOuterSlope} " + $"PlLineOffsetBelowDikeTopAtRiver={PlLineOffsetBelowDikeTopAtRiver} " + $"PlLineOffsetBelowDikeToeAtPolder={PlLineOffsetBelowDikeToeAtPolder} " + $"PlLineOffsetBelowDikeTopAtPolder={PlLineOffsetBelowDikeTopAtPolder} " + $"PlLineOffsetBelowShoulderBaseInside={PlLineOffsetBelowShoulderBaseInside} " + $"UsePlLineOffsetBelowDikeCrestMiddle {(UsePlLineOffsetBelowDikeCrestMiddle.HasValue ? UsePlLineOffsetBelowDikeCrestMiddle.ToString() : "?")} " + $"PlLineOffsetBelowDikeCrestMiddle {(PlLineOffsetBelowDikeCrestMiddle.HasValue ? PlLineOffsetBelowDikeCrestMiddle.ToString() : "?")} " + $"UsePlLineOffsetFactorBelowShoulderCrest {(UsePlLineOffsetFactorBelowShoulderCrest.HasValue ? UsePlLineOffsetFactorBelowShoulderCrest.ToString() : "?")} " + $"PlLineOffsetFactorBelowShoulderCrest {(PlLineOffsetFactorBelowShoulderCrest.HasValue ? PlLineOffsetFactorBelowShoulderCrest.ToString() : "?")} " + $"HeadPl2={(HeadPl2.HasValue ? HeadPl2.ToString() : "?")} " + $"HeadPl3={(HeadPl3.HasValue ? HeadPl3.ToString() : "?")} " + $"HeadPl4={(HeadPl4.HasValue ? HeadPl4.ToString() : "?")} " + $"PolderLevel={PolderLevel} "; } /// /// Get Result Message /// /// /// private string GetResultMessage(Object objectAsKey) { if (objectAsKey != null) { return resultMessages.ContainsKey(objectAsKey) ? resultMessages[objectAsKey] : null; } return null; } /// /// Set Result Message /// /// /// private void SetResultMessage(Object objectAsKey, string resultMessage) { resultMessages[objectAsKey] = resultMessage; } /// /// Set Redesigned surfaceline /// /// /// private void SetRedesignedSurfaceLine(Object objectAsKey, SurfaceLine2 surfaceLine) { redesignedSurfaceLines[objectAsKey] = surfaceLine; } /// /// Get Most recent surfaceline /// /// /// private SurfaceLine2 GetMostRecentSurfaceLine(Object objectAsKey) { if (objectAsKey != null) { //return redesignedSurfaceLines.ContainsKey(objectAsKey) ? redesignedSurfaceLines[objectAsKey] ?? Location.SurfaceLine : Location.SurfaceLine; // For debugging: following code is the same as above, but better debuggable SurfaceLine2 surfaceLine = null; if (redesignedSurfaceLines.ContainsKey(objectAsKey)) { SurfaceLine2 s = redesignedSurfaceLines[objectAsKey]; if (s != null) { surfaceLine = s; } } return surfaceLine; } return null; } /// /// Determine object key based on either 1d-soilprofile or 2d-geometry /// /// /// /// /// private Object GetObjectAsKey(SoilProfile1D soilProfile, SoilProfile2D soilProfile2D) { if (soilProfile != null) { return soilProfile; } return soilProfile2D; } }