// Copyright (C) Stichting Deltares 2016. All rights reserved. // // This file is part of Ringtoets. // // Ringtoets 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.Collections.ObjectModel; using System.Linq; using Core.Common.Base; using Ringtoets.Common.Data.Calculation; using Ringtoets.Common.Data.Hydraulics; using Ringtoets.Common.Service; using Ringtoets.Piping.Data; using Ringtoets.Piping.Primitives; namespace Ringtoets.Piping.Service { /// /// Service for synchronizing piping data. /// public static class PipingDataSynchronizationService { /// /// Clears the output for all calculations in the . /// /// The which contains the calculations. /// An of calculations which are affected by clearing the output. /// Thrown when is null. public static IEnumerable ClearAllCalculationOutput(PipingFailureMechanism failureMechanism) { if (failureMechanism == null) { throw new ArgumentNullException(nameof(failureMechanism)); } return failureMechanism.Calculations .Cast() .SelectMany(ClearCalculationOutput) .ToArray(); } /// /// Clears the output of the given . /// /// The to clear the output for. /// Thrown when is null. /// All objects that have been changed. public static IEnumerable ClearCalculationOutput(PipingCalculation calculation) { if (calculation == null) { throw new ArgumentNullException(nameof(calculation)); } if (calculation.HasOutput) { calculation.ClearOutput(); return new[] { calculation }; } return Enumerable.Empty(); } /// /// Clears the and output for all the calculations in the . /// /// The which contains the calculations. /// An of objects which are affected by removing data. /// Thrown when is null. public static IEnumerable ClearAllCalculationOutputAndHydraulicBoundaryLocations(PipingFailureMechanism failureMechanism) { if (failureMechanism == null) { throw new ArgumentNullException(nameof(failureMechanism)); } var affectedItems = new List(); foreach (var calculation in failureMechanism.Calculations.Cast()) { affectedItems.AddRange(ClearCalculationOutput(calculation) .Concat(ClearHydraulicBoundaryLocation(calculation.InputParameters))); } return affectedItems; } /// /// Clears all data dependent, either directly or indirectly, on the parent reference line. /// /// The failure mechanism to be cleared. /// The results of the clear action. /// Thrown when /// is null. public static ClearResults ClearReferenceLineDependentData(PipingFailureMechanism failureMechanism) { if (failureMechanism == null) { throw new ArgumentNullException(nameof(failureMechanism)); } var changedObjects = new Collection(); var removedObjects = failureMechanism.Sections.OfType() .Concat(failureMechanism.SectionResults) .Concat(failureMechanism.CalculationsGroup.GetAllChildrenRecursive()) .Concat(failureMechanism.StochasticSoilModels) .Concat(failureMechanism.SurfaceLines) .ToArray(); failureMechanism.ClearAllSections(); changedObjects.Add(failureMechanism); failureMechanism.CalculationsGroup.Children.Clear(); changedObjects.Add(failureMechanism.CalculationsGroup); failureMechanism.StochasticSoilModels.Clear(); changedObjects.Add(failureMechanism.StochasticSoilModels); failureMechanism.SurfaceLines.Clear(); changedObjects.Add(failureMechanism.SurfaceLines); return new ClearResults(changedObjects, removedObjects); } /// /// Removes a given from the /// and clears all data that depends on it, either directly or indirectly. /// /// The failure mechanism containing . /// The surfaceline residing in /// that should be removed. /// All observable objects affected by this method. /// Thrown when /// or is null. public static IEnumerable RemoveSurfaceLine(PipingFailureMechanism failureMechanism, RingtoetsPipingSurfaceLine surfaceLine) { if (failureMechanism == null) { throw new ArgumentNullException(nameof(failureMechanism)); } if (surfaceLine == null) { throw new ArgumentNullException(nameof(surfaceLine)); } var changedObservables = new List(); IEnumerable pipingCalculationScenarios = failureMechanism.Calculations .Cast() .Where(pcs => ReferenceEquals(pcs.InputParameters.SurfaceLine, surfaceLine)); foreach (PipingCalculation pipingCalculationScenario in pipingCalculationScenarios) { changedObservables.AddRange(RingtoetsCommonDataSynchronizationService.ClearCalculationOutput(pipingCalculationScenario)); changedObservables.AddRange(ClearSurfaceLine(pipingCalculationScenario.InputParameters)); } failureMechanism.SurfaceLines.Remove(surfaceLine); changedObservables.Add(failureMechanism.SurfaceLines); return changedObservables; } /// /// Removes a given from the /// and clears all data that depends on it, either directly or indirectly. /// /// The failure mechanism containing . /// The soil model residing in /// that should be removed. /// All observable objects affected by this method. /// Thrown when /// or is null. public static IEnumerable RemoveStochasticSoilModel(PipingFailureMechanism failureMechanism, StochasticSoilModel soilModel) { if (failureMechanism == null) { throw new ArgumentNullException(nameof(failureMechanism)); } if (soilModel == null) { throw new ArgumentNullException(nameof(soilModel)); } var changedObservables = new List(); IEnumerable pipingCalculationScenarios = failureMechanism.Calculations .Cast() .Where(pcs => ReferenceEquals(pcs.InputParameters.StochasticSoilModel, soilModel)); foreach (PipingCalculation pipingCalculationScenario in pipingCalculationScenarios) { changedObservables.AddRange(RingtoetsCommonDataSynchronizationService.ClearCalculationOutput(pipingCalculationScenario)); changedObservables.AddRange(ClearStochasticSoilModel(pipingCalculationScenario.InputParameters)); } failureMechanism.StochasticSoilModels.Remove(soilModel); changedObservables.Add(failureMechanism.StochasticSoilModels); return changedObservables; } /// /// Removes a given from calculations in the /// and clears all data that depends on it, either directly or indirectly. /// /// The failure mechanism containing . /// The soil profile residing in /// that has beeen removed. /// All observable objects affected by this method. /// Thrown when /// or is null. public static IEnumerable RemoveStochasticSoilProfileFromInput(PipingFailureMechanism failureMechanism, StochasticSoilProfile soilProfile) { if (failureMechanism == null) { throw new ArgumentNullException(nameof(failureMechanism)); } if (soilProfile == null) { throw new ArgumentNullException(nameof(soilProfile)); } var changedObservables = new List(); foreach (PipingCalculation pipingCalculationScenario in GetCalculationsWithSoilProfileAssigned(failureMechanism, soilProfile)) { changedObservables.AddRange(RingtoetsCommonDataSynchronizationService.ClearCalculationOutput(pipingCalculationScenario)); changedObservables.AddRange(ClearStochasticSoilProfile(pipingCalculationScenario.InputParameters)); } return changedObservables; } /// /// Clears data dependent on a given , either directly or indirectly, /// from calculations in the . /// /// The failure mechanism containing . /// The soil profile residing in /// that has been updated. /// All observable objects affected by this method. /// Thrown when /// or is null. public static IEnumerable ClearStochasticSoilProfileDependentData(PipingFailureMechanism failureMechanism, StochasticSoilProfile soilProfile) { if (failureMechanism == null) { throw new ArgumentNullException(nameof(failureMechanism)); } if (soilProfile == null) { throw new ArgumentNullException(nameof(soilProfile)); } var changedObservables = new List(); foreach (PipingCalculation calculation in GetCalculationsWithSoilProfileAssigned(failureMechanism, soilProfile)) { changedObservables.AddRange(RingtoetsCommonDataSynchronizationService.ClearCalculationOutput(calculation)); changedObservables.Add(calculation.InputParameters); } return changedObservables; } private static IEnumerable GetCalculationsWithSoilProfileAssigned(PipingFailureMechanism failureMechanism, StochasticSoilProfile soilProfile) { IEnumerable pipingCalculationScenarios = failureMechanism.Calculations .Cast() .Where(pcs => ReferenceEquals(pcs.InputParameters.StochasticSoilProfile, soilProfile)); return pipingCalculationScenarios; } private static IEnumerable ClearSurfaceLine(PipingInput input) { input.SurfaceLine = null; return new[] { input }; } private static IEnumerable ClearStochasticSoilModel(PipingInput input) { input.StochasticSoilModel = null; input.StochasticSoilProfile = null; return new[] { input }; } private static IEnumerable ClearStochasticSoilProfile(PipingInput input) { input.StochasticSoilProfile = null; return new[] { input }; } private static IEnumerable ClearHydraulicBoundaryLocation(PipingInput input) { if (input.HydraulicBoundaryLocation != null) { input.HydraulicBoundaryLocation = null; return new[] { input }; } return Enumerable.Empty(); } } }