// Copyright (C) Stichting Deltares 2017. 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.Collections; using System.Collections.Generic; using System.Drawing; using System.Linq; using System.Windows.Forms; using Core.Common.Base; using Core.Common.Controls.TreeView; using Core.Common.Gui; using Core.Common.Gui.ContextMenu; using Core.Common.Gui.Forms; using Core.Common.Gui.Forms.ProgressDialog; using Core.Common.Gui.Plugin; using Core.Common.Utils; using Ringtoets.Common.Data.AssessmentSection; using Ringtoets.Common.Data.Calculation; using Ringtoets.Common.Data.FailureMechanism; using Ringtoets.Common.Forms.ChangeHandlers; using Ringtoets.Common.Forms.ExportInfos; using Ringtoets.Common.Forms.Helpers; using Ringtoets.Common.Forms.ImportInfos; using Ringtoets.Common.Forms.PresentationObjects; using Ringtoets.Common.Forms.TreeNodeInfos; using Ringtoets.Common.IO.FileImporters.MessageProviders; using Ringtoets.Common.Service; using Ringtoets.MacroStabilityInwards.Data; using Ringtoets.MacroStabilityInwards.Forms; using Ringtoets.MacroStabilityInwards.Forms.PresentationObjects; using Ringtoets.MacroStabilityInwards.Forms.PropertyClasses; using Ringtoets.MacroStabilityInwards.Forms.Views; using Ringtoets.MacroStabilityInwards.IO.Exporters; using Ringtoets.MacroStabilityInwards.IO.Importers; using Ringtoets.MacroStabilityInwards.Plugin.FileImporter; using Ringtoets.MacroStabilityInwards.Plugin.Properties; using Ringtoets.MacroStabilityInwards.Primitives; using Ringtoets.MacroStabilityInwards.Service; using RingtoetsCommonFormsResources = Ringtoets.Common.Forms.Properties.Resources; using RingtoetsCommonDataResources = Ringtoets.Common.Data.Properties.Resources; using MacroStabilityInwardsDataResources = Ringtoets.MacroStabilityInwards.Data.Properties.Resources; using MacroStabilityInwardsFormsResources = Ringtoets.MacroStabilityInwards.Forms.Properties.Resources; namespace Ringtoets.MacroStabilityInwards.Plugin { /// /// The plug-in for the . /// public class MacroStabilityInwardsPlugin : PluginBase { public override IRibbonCommandHandler RibbonCommandHandler { get { return new PipingRibbon(); } } public override IEnumerable GetPropertyInfos() { yield return new PropertyInfo { CreateInstance = context => new MacroStabilityInwardsFailureMechanismContextProperties(context, new FailureMechanismPropertyChangeHandler()) }; yield return new PropertyInfo { CreateInstance = context => new PipingInputContextProperties(context, new ObservablePropertyChangeHandler(context.PipingCalculation, context.WrappedData)) }; yield return new PropertyInfo(); yield return new PropertyInfo { CreateInstance = context => new RingtoetsPipingSurfaceLineCollectionProperties(context.WrappedData) }; yield return new PropertyInfo(); yield return new PropertyInfo { CreateInstance = context => new StochasticSoilModelCollectionProperties(context.WrappedData) }; yield return new PropertyInfo(); yield return new PropertyInfo(); } public override IEnumerable GetImportInfos() { yield return new ImportInfo { Name = MacroStabilityInwardsDataResources.PipingSurfaceLineCollection_TypeDescriptor, Category = RingtoetsCommonFormsResources.Ringtoets_Category, Image = MacroStabilityInwardsFormsResources.PipingSurfaceLineIcon, FileFilterGenerator = RingtoetsPipingSurfaceLineFileFilter, IsEnabled = context => context.AssessmentSection.ReferenceLine != null, CreateFileImporter = (context, filePath) => PipingSurfaceLinesCsvImporter(context, filePath, new ImportMessageProvider(), new RingtoetsPipingSurfaceLineReplaceDataStrategy(context.FailureMechanism)), VerifyUpdates = context => VerifyPipingSurfaceLineUpdates(context, Resources.PipingPlugin_VerifyRingtoetsPipingSurfaceLineImport_When_importing_surfacelines_calculation_output_will_be_cleared_confirm) }; yield return new ImportInfo { Name = MacroStabilityInwardsDataResources.StochasticSoilModelCollection_TypeDescriptor, Category = RingtoetsCommonFormsResources.Ringtoets_Category, Image = MacroStabilityInwardsFormsResources.PipingSoilProfileIcon, FileFilterGenerator = StochasticSoilModelFileFilter, IsEnabled = context => context.AssessmentSection.ReferenceLine != null, CreateFileImporter = (context, filePath) => StochasticSoilModelImporter(context, filePath, new ImportMessageProvider(), new StochasticSoilModelReplaceDataStrategy(context.FailureMechanism)), VerifyUpdates = context => VerifyStochasticSoilModelUpdates(context, Resources.PipingPlugin_VerifyStochasticSoilModelImport_When_importing_StochasticSoilModels_calculation_output_will_be_cleared_confirm) }; yield return RingtoetsImportInfoFactory.CreateCalculationConfigurationImportInfo( (context, filePath) => new PipingCalculationConfigurationImporter( filePath, context.WrappedData, context.AvailableHydraulicBoundaryLocations, context.FailureMechanism)); } public override IEnumerable GetExportInfos() { yield return RingtoetsExportInfoFactory.CreateCalculationGroupConfigurationExportInfo( (context, filePath) => new PipingCalculationConfigurationExporter(context.WrappedData.Children, filePath), context => context.WrappedData.Children.Any()); yield return RingtoetsExportInfoFactory.CreateCalculationConfigurationExportInfo( (context, filePath) => new PipingCalculationConfigurationExporter(new[] { context.WrappedData }, filePath)); } public override IEnumerable GetUpdateInfos() { yield return new UpdateInfo { Name = MacroStabilityInwardsDataResources.PipingSurfaceLineCollection_TypeDescriptor, Category = RingtoetsCommonFormsResources.Ringtoets_Category, Image = MacroStabilityInwardsFormsResources.PipingSurfaceLineIcon, FileFilterGenerator = RingtoetsPipingSurfaceLineFileFilter, IsEnabled = context => context.WrappedData.SourcePath != null, CurrentPath = context => context.WrappedData.SourcePath, CreateFileImporter = (context, filePath) => PipingSurfaceLinesCsvImporter(context, filePath, new UpdateMessageProvider(), new RingtoetsPipingSurfaceLineUpdateDataStrategy(context.FailureMechanism)), VerifyUpdates = context => VerifyPipingSurfaceLineUpdates(context, Resources.PipingPlugin_VerifyRingtoetsPipingSurfaceLineUpdates_When_updating_surfacelines_definitions_assigned_to_calculation_output_will_be_cleared_confirm) }; yield return new UpdateInfo { Name = MacroStabilityInwardsDataResources.StochasticSoilModelCollection_TypeDescriptor, Category = RingtoetsCommonFormsResources.Ringtoets_Category, Image = MacroStabilityInwardsFormsResources.PipingSoilProfileIcon, FileFilterGenerator = StochasticSoilModelFileFilter, IsEnabled = context => context.WrappedData.SourcePath != null, CurrentPath = context => context.WrappedData.SourcePath, CreateFileImporter = (context, filePath) => StochasticSoilModelImporter(context, filePath, new UpdateMessageProvider(), new StochasticSoilModelUpdateDataStrategy(context.FailureMechanism)), VerifyUpdates = context => VerifyStochasticSoilModelUpdates(context, Resources.PipingPlugin_VerifyStochasticSoilModelUpdates_When_updating_StochasticSoilModel_definitions_assigned_to_calculation_output_will_be_cleared_confirm) }; } public override IEnumerable GetViewInfos() { yield return new ViewInfo { GetViewName = (view, mechanism) => mechanism.WrappedData.Name, Image = RingtoetsCommonFormsResources.CalculationIcon, CloseForData = ClosePipingFailureMechanismViewForData, AdditionalDataCheck = context => context.WrappedData.IsRelevant }; yield return new ViewInfo< FailureMechanismSectionResultContext, IEnumerable, PipingFailureMechanismResultView> { GetViewName = (view, results) => RingtoetsCommonFormsResources.FailureMechanism_AssessmentResult_DisplayName, Image = RingtoetsCommonFormsResources.FailureMechanismSectionResultIcon, CloseForData = CloseFailureMechanismResultViewForData, GetViewData = context => context.WrappedData, AfterCreate = (view, context) => view.FailureMechanism = context.FailureMechanism }; yield return new ViewInfo { GetViewData = context => context.WrappedData, GetViewName = (view, calculationGroup) => calculationGroup.Name, Image = RingtoetsCommonFormsResources.GeneralFolderIcon, AdditionalDataCheck = context => context.WrappedData == context.FailureMechanism.CalculationsGroup, CloseForData = ClosePipingCalculationsViewForData, AfterCreate = (view, context) => { view.AssessmentSection = context.AssessmentSection; view.MacroStabilityInwardsFailureMechanism = context.FailureMechanism; } }; yield return new ViewInfo { GetViewData = context => context.PipingCalculation, GetViewName = (view, input) => RingtoetsCommonFormsResources.Calculation_Input, Image = MacroStabilityInwardsFormsResources.PipingInputIcon, CloseForData = ClosePipingInputViewForData }; yield return new ViewInfo { GetViewData = context => context.WrappedData, GetViewName = (view, calculationGroup) => RingtoetsCommonFormsResources.Scenarios_DisplayName, Image = RingtoetsCommonFormsResources.ScenariosIcon, AdditionalDataCheck = context => context.WrappedData == context.ParentFailureMechanism.CalculationsGroup, CloseForData = ClosePipingScenariosViewForData, AfterCreate = (view, context) => { view.MacroStabilityInwardsFailureMechanism = context.ParentFailureMechanism; } }; } public override IEnumerable GetTreeNodeInfos() { yield return RingtoetsTreeNodeInfoFactory.CreateFailureMechanismContextTreeNodeInfo( FailureMechanismEnabledChildNodeObjects, FailureMechanismDisabledChildNodeObjects, FailureMechanismEnabledContextMenuStrip, FailureMechanismDisabledContextMenuStrip); yield return RingtoetsTreeNodeInfoFactory.CreateCalculationContextTreeNodeInfo( PipingCalculationContextChildNodeObjects, PipingCalculationContextContextMenuStrip, PipingCalculationContextOnNodeRemoved); yield return RingtoetsTreeNodeInfoFactory.CreateCalculationGroupContextTreeNodeInfo( PipingCalculationGroupContextChildNodeObjects, PipingCalculationGroupContextContextMenuStrip, PipingCalculationGroupContextOnNodeRemoved); yield return new TreeNodeInfo> { Text = context => RingtoetsCommonFormsResources.FailureMechanism_AssessmentResult_DisplayName, Image = context => RingtoetsCommonFormsResources.FailureMechanismSectionResultIcon, ContextMenuStrip = (nodeData, parentData, treeViewControl) => Gui.Get(nodeData, treeViewControl) .AddOpenItem() .Build() }; yield return new TreeNodeInfo { Text = pipingInputContext => RingtoetsCommonFormsResources.Calculation_Input, Image = pipingInputContext => MacroStabilityInwardsFormsResources.PipingInputIcon, ContextMenuStrip = (nodeData, parentData, treeViewControl) => Gui.Get(nodeData, treeViewControl) .AddOpenItem() .AddSeparator() .AddPropertiesItem() .Build() }; yield return new TreeNodeInfo { Text = ringtoetsPipingSurfaceLine => MacroStabilityInwardsDataResources.PipingSurfaceLineCollection_TypeDescriptor, Image = ringtoetsPipingSurfaceLine => RingtoetsCommonFormsResources.GeneralFolderIcon, ForeColor = ringtoetsPipingSurfaceLine => ringtoetsPipingSurfaceLine.WrappedData.Any() ? Color.FromKnownColor(KnownColor.ControlText) : Color.FromKnownColor(KnownColor.GrayText), ChildNodeObjects = ringtoetsPipingSurfaceLine => ringtoetsPipingSurfaceLine.WrappedData.Cast().ToArray(), ContextMenuStrip = RingtoetsPipingSurfaceLinesContextContextMenuStrip }; yield return new TreeNodeInfo { Text = pipingSurfaceLine => pipingSurfaceLine.Name, Image = pipingSurfaceLine => MacroStabilityInwardsFormsResources.PipingSurfaceLineIcon, ContextMenuStrip = (nodeData, parentData, treeViewControl) => Gui.Get(nodeData, treeViewControl) .AddSeparator() .AddPropertiesItem() .Build() }; yield return new TreeNodeInfo { Text = stochasticSoilModelContext => MacroStabilityInwardsDataResources.StochasticSoilModelCollection_TypeDescriptor, Image = stochasticSoilModelContext => RingtoetsCommonFormsResources.GeneralFolderIcon, ForeColor = stochasticSoilModelContext => stochasticSoilModelContext.WrappedData.Any() ? Color.FromKnownColor(KnownColor.ControlText) : Color.FromKnownColor(KnownColor.GrayText), ChildNodeObjects = stochasticSoilModelContext => stochasticSoilModelContext.WrappedData.Cast().ToArray(), ContextMenuStrip = StochasticSoilModelCollectionContextContextMenuStrip }; yield return new TreeNodeInfo { Text = stochasticSoilModel => stochasticSoilModel.Name, Image = stochasticSoilModel => MacroStabilityInwardsFormsResources.StochasticSoilModelIcon, ChildNodeObjects = stochasticSoilModel => stochasticSoilModel.StochasticSoilProfiles.Cast().ToArray(), ContextMenuStrip = (nodeData, parentData, treeViewControl) => Gui.Get(nodeData, treeViewControl) .AddCollapseAllItem() .AddExpandAllItem() .AddSeparator() .AddPropertiesItem() .Build() }; yield return new TreeNodeInfo { Text = pipingSoilProfile => pipingSoilProfile.SoilProfile != null ? pipingSoilProfile.SoilProfile.Name : "Profile", Image = pipingSoilProfile => MacroStabilityInwardsFormsResources.PipingSoilProfileIcon, ContextMenuStrip = (nodeData, parentData, treeViewControl) => Gui.Get(nodeData, treeViewControl) .AddPropertiesItem() .Build() }; yield return new TreeNodeInfo { Text = pipingOutput => RingtoetsCommonFormsResources.CalculationOutput_DisplayName, Image = pipingOutput => RingtoetsCommonFormsResources.GeneralOutputIcon, ContextMenuStrip = (nodeData, parentData, treeViewControl) => Gui.Get(nodeData, treeViewControl) .AddPropertiesItem() .Build() }; yield return new TreeNodeInfo { Text = context => RingtoetsCommonFormsResources.Scenarios_DisplayName, Image = context => RingtoetsCommonFormsResources.ScenariosIcon, ContextMenuStrip = (nodeData, parentData, treeViewControl) => Gui.Get(nodeData, treeViewControl) .AddOpenItem() .Build() }; yield return new TreeNodeInfo { Text = emptyPipingOutput => RingtoetsCommonFormsResources.CalculationOutput_DisplayName, Image = emptyPipingOutput => RingtoetsCommonFormsResources.GeneralOutputIcon, ForeColor = emptyPipingOutput => Color.FromKnownColor(KnownColor.GrayText), ContextMenuStrip = (nodeData, parentData, treeViewControl) => Gui.Get(nodeData, treeViewControl) .AddPropertiesItem() .Build() }; } private static FileFilterGenerator StochasticSoilModelFileFilter { get { return new FileFilterGenerator(Resources.Soil_file_Extension, Resources.Soil_file_Description); } } private static StochasticSoilModelImporter StochasticSoilModelImporter(StochasticSoilModelCollectionContext context, string filePath, IImporterMessageProvider messageProvider, IStochasticSoilModelUpdateModelStrategy updateStrategy) { return new StochasticSoilModelImporter(context.WrappedData, filePath, messageProvider, updateStrategy); } private bool VerifyStochasticSoilModelUpdates(StochasticSoilModelCollectionContext context, string query) { var changeHandler = new FailureMechanismCalculationChangeHandler(context.FailureMechanism, query, new DialogBasedInquiryHelper(Gui.MainWindow)); return !changeHandler.RequireConfirmation() || changeHandler.InquireConfirmation(); } #region MacroStabilityInwardsFailureMechanismView ViewInfo private static bool ClosePipingFailureMechanismViewForData(MacroStabilityInwardsFailureMechanismView view, object o) { var assessmentSection = o as IAssessmentSection; var pipingFailureMechanism = o as MacroStabilityInwardsFailureMechanism; var viewPipingFailureMechanismContext = (MacroStabilityInwardsFailureMechanismContext) view.Data; MacroStabilityInwardsFailureMechanism viewMacroStabilityInwardsFailureMechanism = viewPipingFailureMechanismContext.WrappedData; return assessmentSection != null ? ReferenceEquals(viewPipingFailureMechanismContext.Parent, assessmentSection) : ReferenceEquals(viewMacroStabilityInwardsFailureMechanism, pipingFailureMechanism); } #endregion #region FailureMechanismResultsView ViewInfo private static bool CloseFailureMechanismResultViewForData(PipingFailureMechanismResultView view, object o) { var assessmentSection = o as IAssessmentSection; var failureMechanism = o as MacroStabilityInwardsFailureMechanism; var failureMechanismContext = o as IFailureMechanismContext; if (assessmentSection != null) { return assessmentSection .GetFailureMechanisms() .OfType() .Any(fm => ReferenceEquals(view.Data, fm.SectionResults)); } if (failureMechanismContext != null) { failureMechanism = failureMechanismContext.WrappedData; } return failureMechanism != null && ReferenceEquals(view.Data, failureMechanism.SectionResults); } #endregion #region PipingCalculationsView ViewInfo private static bool ClosePipingCalculationsViewForData(PipingCalculationsView view, object o) { var assessmentSection = o as IAssessmentSection; var pipingFailureMechanism = o as MacroStabilityInwardsFailureMechanism; var pipingFailureMechanismContext = o as MacroStabilityInwardsFailureMechanismContext; if (pipingFailureMechanismContext != null) { pipingFailureMechanism = pipingFailureMechanismContext.WrappedData; } if (assessmentSection != null) { pipingFailureMechanism = assessmentSection.GetFailureMechanisms() .OfType() .FirstOrDefault(); } return pipingFailureMechanism != null && ReferenceEquals(view.Data, pipingFailureMechanism.CalculationsGroup); } #endregion endregion #region PipingScenariosView ViewInfo private static bool ClosePipingScenariosViewForData(PipingScenariosView view, object o) { var assessmentSection = o as IAssessmentSection; var pipingFailureMechanism = o as MacroStabilityInwardsFailureMechanism; var pipingFailureMechanismContext = o as MacroStabilityInwardsFailureMechanismContext; if (pipingFailureMechanismContext != null) { pipingFailureMechanism = pipingFailureMechanismContext.WrappedData; } if (assessmentSection != null) { pipingFailureMechanism = assessmentSection.GetFailureMechanisms() .OfType() .FirstOrDefault(); } return pipingFailureMechanism != null && ReferenceEquals(view.Data, pipingFailureMechanism.CalculationsGroup); } #endregion endregion #region PipingInputView ViewInfo private static bool ClosePipingInputViewForData(PipingInputView view, object o) { var pipingCalculationScenarioContext = o as PipingCalculationScenarioContext; if (pipingCalculationScenarioContext != null) { return ReferenceEquals(view.Data, pipingCalculationScenarioContext.WrappedData); } IEnumerable calculations = null; var pipingCalculationGroupContext = o as PipingCalculationGroupContext; if (pipingCalculationGroupContext != null) { calculations = pipingCalculationGroupContext.WrappedData.GetCalculations() .OfType(); } var failureMechanism = o as MacroStabilityInwardsFailureMechanism; var failureMechanismContext = o as MacroStabilityInwardsFailureMechanismContext; if (failureMechanismContext != null) { failureMechanism = failureMechanismContext.WrappedData; } var assessmentSection = o as IAssessmentSection; if (assessmentSection != null) { failureMechanism = assessmentSection.GetFailureMechanisms() .OfType() .FirstOrDefault(); } if (failureMechanism != null) { calculations = failureMechanism.CalculationsGroup.GetCalculations() .OfType(); } return calculations != null && calculations.Any(ci => ReferenceEquals(view.Data, ci)); } #endregion private void CalculateAll(MacroStabilityInwardsFailureMechanismContext failureMechanismContext) { IEnumerable calculations = GetAllPipingCalculations(failureMechanismContext.WrappedData); PipingProbabilityAssessmentInput assessmentInput = failureMechanismContext.WrappedData.PipingProbabilityAssessmentInput; double norm = failureMechanismContext.Parent.FailureMechanismContribution.Norm; double contribution = failureMechanismContext.WrappedData.Contribution; CalculateAll(calculations, assessmentInput, norm, contribution); } private void CalculateAll(CalculationGroup group, PipingCalculationGroupContext context) { PipingCalculation[] calculations = group.GetCalculations().OfType().ToArray(); PipingProbabilityAssessmentInput assessmentInput = context.FailureMechanism.PipingProbabilityAssessmentInput; double norm = context.AssessmentSection.FailureMechanismContribution.Norm; double contribution = context.FailureMechanism.Contribution; CalculateAll(calculations, assessmentInput, norm, contribution); } private static void ValidateAll(IEnumerable pipingCalculations) { foreach (PipingCalculation calculation in pipingCalculations) { PipingCalculationService.Validate(calculation); } } private void CalculateAll(IEnumerable calculations, PipingProbabilityAssessmentInput assessmentInput, double norm, double contribution) { ActivityProgressDialogRunner.Run( Gui.MainWindow, calculations .Select(pc => new PipingCalculationActivity(pc, assessmentInput, norm, contribution)) .ToList()); } private static string ValidateAllDataAvailableAndGetErrorMessage(IFailureMechanism failureMechanism) { if (failureMechanism.Contribution <= 0.0) { return RingtoetsCommonFormsResources.Contribution_of_failure_mechanism_zero; } return null; } #region RingtoetsPipingSurfaceLinesContext TreeNodeInfo private ContextMenuStrip RingtoetsPipingSurfaceLinesContextContextMenuStrip(RingtoetsPipingSurfaceLinesContext nodeData, object parentData, TreeViewControl treeViewControl) { return Gui.Get(nodeData, treeViewControl) .AddImportItem() .AddUpdateItem() .AddSeparator() .AddCollapseAllItem() .AddExpandAllItem() .AddSeparator() .AddPropertiesItem() .Build(); } #endregion #region StochasticSoilModelCollectionContext TreeNodeInfo private ContextMenuStrip StochasticSoilModelCollectionContextContextMenuStrip(StochasticSoilModelCollectionContext nodeData, object parentData, TreeViewControl treeViewControl) { return Gui.Get(nodeData, treeViewControl) .AddImportItem() .AddUpdateItem() .AddSeparator() .AddCollapseAllItem() .AddExpandAllItem() .AddSeparator() .AddPropertiesItem() .Build(); } #endregion #region MacroStabilityInwardsFailureMechanismContext TreeNodeInfo private ContextMenuStrip FailureMechanismEnabledContextMenuStrip(MacroStabilityInwardsFailureMechanismContext macroStabilityInwardsFailureMechanismContext, object parentData, TreeViewControl treeViewControl) { var builder = new RingtoetsContextMenuBuilder(Gui.Get(macroStabilityInwardsFailureMechanismContext, treeViewControl)); return builder.AddOpenItem() .AddSeparator() .AddToggleRelevancyOfFailureMechanismItem(macroStabilityInwardsFailureMechanismContext, RemoveAllViewsForItem) .AddSeparator() .AddValidateAllCalculationsInFailureMechanismItem( macroStabilityInwardsFailureMechanismContext, ValidateAll, ValidateAllDataAvailableAndGetErrorMessage) .AddPerformAllCalculationsInFailureMechanismItem( macroStabilityInwardsFailureMechanismContext, CalculateAll, ValidateAllDataAvailableAndGetErrorMessage) .AddSeparator() .AddClearAllCalculationOutputInFailureMechanismItem(macroStabilityInwardsFailureMechanismContext.WrappedData) .AddSeparator() .AddCollapseAllItem() .AddExpandAllItem() .AddSeparator() .AddPropertiesItem() .Build(); } private void RemoveAllViewsForItem(MacroStabilityInwardsFailureMechanismContext failureMechanismContext) { Gui.ViewCommands.RemoveAllViewsForItem(failureMechanismContext); } private static void ValidateAll(MacroStabilityInwardsFailureMechanismContext context) { ValidateAll(context.WrappedData.Calculations.OfType()); } private ContextMenuStrip FailureMechanismDisabledContextMenuStrip(MacroStabilityInwardsFailureMechanismContext macroStabilityInwardsFailureMechanismContext, object parentData, TreeViewControl treeViewControl) { var builder = new RingtoetsContextMenuBuilder(Gui.Get(macroStabilityInwardsFailureMechanismContext, treeViewControl)); return builder.AddToggleRelevancyOfFailureMechanismItem(macroStabilityInwardsFailureMechanismContext, RemoveAllViewsForItem) .AddSeparator() .AddCollapseAllItem() .AddExpandAllItem() .Build(); } private static string ValidateAllDataAvailableAndGetErrorMessage(MacroStabilityInwardsFailureMechanismContext context) { return ValidateAllDataAvailableAndGetErrorMessage(context.WrappedData); } private static IEnumerable GetAllPipingCalculations(MacroStabilityInwardsFailureMechanism failureMechanism) { return failureMechanism.Calculations.OfType(); } private static object[] FailureMechanismEnabledChildNodeObjects(MacroStabilityInwardsFailureMechanismContext macroStabilityInwardsFailureMechanismContext) { MacroStabilityInwardsFailureMechanism wrappedData = macroStabilityInwardsFailureMechanismContext.WrappedData; return new object[] { new CategoryTreeFolder(RingtoetsCommonFormsResources.FailureMechanism_Inputs_DisplayName, GetInputs(wrappedData, macroStabilityInwardsFailureMechanismContext.Parent), TreeFolderCategory.Input), new PipingCalculationGroupContext(wrappedData.CalculationsGroup, wrappedData.SurfaceLines, wrappedData.StochasticSoilModels, wrappedData, macroStabilityInwardsFailureMechanismContext.Parent), new CategoryTreeFolder(RingtoetsCommonFormsResources.FailureMechanism_Outputs_DisplayName, GetOutputs(wrappedData), TreeFolderCategory.Output) }; } private static object[] FailureMechanismDisabledChildNodeObjects(MacroStabilityInwardsFailureMechanismContext macroStabilityInwardsFailureMechanismContext) { return new object[] { macroStabilityInwardsFailureMechanismContext.WrappedData.NotRelevantComments }; } private static IList GetInputs(MacroStabilityInwardsFailureMechanism failureMechanism, IAssessmentSection assessmentSection) { return new ArrayList { new FailureMechanismSectionsContext(failureMechanism, assessmentSection), new RingtoetsPipingSurfaceLinesContext(failureMechanism.SurfaceLines, failureMechanism, assessmentSection), new StochasticSoilModelCollectionContext(failureMechanism.StochasticSoilModels, failureMechanism, assessmentSection), failureMechanism.InputComments }; } private static IList GetOutputs(MacroStabilityInwardsFailureMechanism failureMechanism) { return new ArrayList { new PipingScenariosContext(failureMechanism.CalculationsGroup, failureMechanism), new FailureMechanismSectionResultContext( failureMechanism.SectionResults, failureMechanism), failureMechanism.OutputComments }; } #endregion #region PipingCalculationScenarioContext TreeNodeInfo private ContextMenuStrip PipingCalculationContextContextMenuStrip(PipingCalculationScenarioContext nodeData, object parentData, TreeViewControl treeViewControl) { var builder = new RingtoetsContextMenuBuilder(Gui.Get(nodeData, treeViewControl)); PipingCalculation calculation = nodeData.WrappedData; StrictContextMenuItem updateEntryAndExitPoint = CreateUpdateEntryAndExitPointItem(nodeData); return builder.AddExportItem() .AddSeparator() .AddRenameItem() .AddCustomItem(updateEntryAndExitPoint) .AddSeparator() .AddValidateCalculationItem( nodeData, Validate, ValidateAllDataAvailableAndGetErrorMessage) .AddPerformCalculationItem( calculation, nodeData, PerformCalculation, ValidateAllDataAvailableAndGetErrorMessage) .AddSeparator() .AddClearCalculationOutputItem(calculation) .AddDeleteItem() .AddSeparator() .AddCollapseAllItem() .AddExpandAllItem() .AddSeparator() .AddPropertiesItem() .Build(); } private static object[] PipingCalculationContextChildNodeObjects(PipingCalculationScenarioContext pipingCalculationScenarioContext) { PipingCalculationScenario pipingCalculationScenario = pipingCalculationScenarioContext.WrappedData; var childNodes = new List { pipingCalculationScenario.Comments, new PipingInputContext(pipingCalculationScenario.InputParameters, pipingCalculationScenario, pipingCalculationScenarioContext.AvailablePipingSurfaceLines, pipingCalculationScenarioContext.AvailableStochasticSoilModels, pipingCalculationScenarioContext.FailureMechanism, pipingCalculationScenarioContext.AssessmentSection) }; if (pipingCalculationScenario.HasOutput) { childNodes.Add(new PipingOutputContext( pipingCalculationScenario.Output, pipingCalculationScenario.SemiProbabilisticOutput)); } else { childNodes.Add(new EmptyPipingOutput()); } return childNodes.ToArray(); } private static void PipingCalculationContextOnNodeRemoved(PipingCalculationScenarioContext pipingCalculationScenarioContext, object parentNodeData) { var calculationGroupContext = parentNodeData as PipingCalculationGroupContext; if (calculationGroupContext != null) { bool successfullyRemovedData = calculationGroupContext.WrappedData.Children.Remove(pipingCalculationScenarioContext.WrappedData); if (successfullyRemovedData) { calculationGroupContext.NotifyObservers(); } } } private static void Validate(PipingCalculationScenarioContext context) { PipingCalculationService.Validate(context.WrappedData); } private static string ValidateAllDataAvailableAndGetErrorMessage(PipingCalculationScenarioContext context) { return ValidateAllDataAvailableAndGetErrorMessage(context.FailureMechanism); } private void PerformCalculation(PipingCalculation calculation, PipingCalculationScenarioContext context) { ActivityProgressDialogRunner.Run(Gui.MainWindow, new PipingCalculationActivity(calculation, context.FailureMechanism.PipingProbabilityAssessmentInput, context.AssessmentSection.FailureMechanismContribution.Norm, context.FailureMechanism.Contribution)); } private StrictContextMenuItem CreateUpdateEntryAndExitPointItem(PipingCalculationScenarioContext context) { var contextMenuEnabled = true; string toolTipMessage = Resources.PipingPlugin_CreateUpdateEntryAndExitPointItem_Update_calculation_with_characteristic_points_ToolTip; if (context.WrappedData.InputParameters.SurfaceLine == null) { contextMenuEnabled = false; toolTipMessage = Resources.PipingPlugin_CreateUpdateEntryAndExitPointItem_Update_calculation_no_surface_line_ToolTip; } else if (context.WrappedData.InputParameters.IsEntryAndExitPointInputSynchronized) { contextMenuEnabled = false; toolTipMessage = RingtoetsCommonFormsResources.CalculationItem_No_changes_to_update_ToolTip; } return new StrictContextMenuItem( Resources.PipingPlugin_CreateUpdateEntryAndExitPointItem_Update_entry_and_exit_point, toolTipMessage, RingtoetsCommonFormsResources.UpdateItemIcon, (o, args) => { UpdatedSurfaceLineDependentDataOfCalculation(context.WrappedData); }) { Enabled = contextMenuEnabled }; } private void UpdatedSurfaceLineDependentDataOfCalculation(PipingCalculation scenario) { string message = RingtoetsCommonFormsResources.VerifyUpdate_Confirm_calculation_output_cleared; if (VerifyEntryAndExitPointUpdates(new[] { scenario }, message)) { UpdateSurfaceLineDependentData(scenario); } } private static void UpdateSurfaceLineDependentData(PipingCalculation scenario) { scenario.InputParameters.SynchronizeEntryAndExitPointInput(); var affectedObjects = new List { scenario.InputParameters }; affectedObjects.AddRange(RingtoetsCommonDataSynchronizationService.ClearCalculationOutput(scenario)); foreach (IObservable affectedObject in affectedObjects) { affectedObject.NotifyObservers(); } } #endregion #region PipingCalculationGroupContext TreeNodeInfo private static object[] PipingCalculationGroupContextChildNodeObjects(PipingCalculationGroupContext nodeData) { var childNodeObjects = new List(); foreach (ICalculationBase item in nodeData.WrappedData.Children) { var calculation = item as PipingCalculationScenario; var group = item as CalculationGroup; if (calculation != null) { childNodeObjects.Add(new PipingCalculationScenarioContext(calculation, nodeData.AvailablePipingSurfaceLines, nodeData.AvailableStochasticSoilModels, nodeData.FailureMechanism, nodeData.AssessmentSection)); } else if (group != null) { childNodeObjects.Add(new PipingCalculationGroupContext(group, nodeData.AvailablePipingSurfaceLines, nodeData.AvailableStochasticSoilModels, nodeData.FailureMechanism, nodeData.AssessmentSection)); } else { childNodeObjects.Add(item); } } return childNodeObjects.ToArray(); } private ContextMenuStrip PipingCalculationGroupContextContextMenuStrip(PipingCalculationGroupContext nodeData, object parentData, TreeViewControl treeViewControl) { CalculationGroup group = nodeData.WrappedData; var builder = new RingtoetsContextMenuBuilder(Gui.Get(nodeData, treeViewControl)); bool isNestedGroup = parentData is PipingCalculationGroupContext; StrictContextMenuItem generateCalculationsItem = CreateGeneratePipingCalculationsItem(nodeData); StrictContextMenuItem updateEntryAndExitPointsItem = CreateUpdateEntryAndExitPointItem(nodeData); if (!isNestedGroup) { builder.AddOpenItem() .AddSeparator(); } builder.AddImportItem() .AddExportItem() .AddSeparator(); if (!isNestedGroup) { builder.AddCustomItem(generateCalculationsItem) .AddSeparator(); } builder.AddCreateCalculationGroupItem(group) .AddCreateCalculationItem(nodeData, AddCalculationScenario) .AddSeparator(); if (isNestedGroup) { builder.AddRenameItem(); } builder.AddCustomItem(updateEntryAndExitPointsItem) .AddSeparator() .AddValidateAllCalculationsInGroupItem( nodeData, ValidateAll, ValidateAllDataAvailableAndGetErrorMessage) .AddPerformAllCalculationsInGroupItem( group, nodeData, CalculateAll, ValidateAllDataAvailableAndGetErrorMessage) .AddSeparator() .AddClearAllCalculationOutputInGroupItem(group); if (isNestedGroup) { builder.AddDeleteItem(); } else { builder.AddRemoveAllChildrenItem(); } return builder.AddSeparator() .AddCollapseAllItem() .AddExpandAllItem() .AddSeparator() .AddPropertiesItem() .Build(); } private static void ValidateAll(PipingCalculationGroupContext context) { ValidateAll(context.WrappedData.GetCalculations().OfType()); } private static void AddCalculationScenario(PipingCalculationGroupContext nodeData) { var calculation = new PipingCalculationScenario(nodeData.FailureMechanism.GeneralInput) { Name = NamingHelper.GetUniqueName(nodeData.WrappedData.Children, RingtoetsCommonDataResources.Calculation_DefaultName, c => c.Name) }; nodeData.WrappedData.Children.Add(calculation); nodeData.WrappedData.NotifyObservers(); } private StrictContextMenuItem CreateGeneratePipingCalculationsItem(PipingCalculationGroupContext nodeData) { bool surfaceLineAvailable = nodeData.AvailablePipingSurfaceLines.Any() && nodeData.AvailableStochasticSoilModels.Any(); string pipingCalculationGroupGeneratePipingCalculationsToolTip = surfaceLineAvailable ? MacroStabilityInwardsFormsResources.PipingCalculationGroup_Generate_PipingCalculations_ToolTip : MacroStabilityInwardsFormsResources.PipingCalculationGroup_Generate_PipingCalculations_NoSurfaceLinesOrSoilModels_ToolTip; return new StrictContextMenuItem( RingtoetsCommonFormsResources.CalculationGroup_Generate_Scenarios, pipingCalculationGroupGeneratePipingCalculationsToolTip, RingtoetsCommonFormsResources.GenerateScenariosIcon, (o, args) => { ShowSurfaceLineSelectionDialog(nodeData); }) { Enabled = surfaceLineAvailable }; } private static string ValidateAllDataAvailableAndGetErrorMessage(PipingCalculationGroupContext context) { return ValidateAllDataAvailableAndGetErrorMessage(context.FailureMechanism); } private void ShowSurfaceLineSelectionDialog(PipingCalculationGroupContext nodeData) { using (var view = new PipingSurfaceLineSelectionDialog(Gui.MainWindow, nodeData.AvailablePipingSurfaceLines)) { view.ShowDialog(); GeneratePipingCalculations(nodeData.WrappedData, view.SelectedItems, nodeData.AvailableStochasticSoilModels, nodeData.FailureMechanism.GeneralInput); } nodeData.NotifyObservers(); } private static void GeneratePipingCalculations(CalculationGroup target, IEnumerable surfaceLines, IEnumerable soilModels, GeneralPipingInput generalInput) { foreach (ICalculationBase group in PipingCalculationConfigurationHelper.GenerateCalculationItemsStructure(surfaceLines, soilModels, generalInput)) { target.Children.Add(group); } } private static void PipingCalculationGroupContextOnNodeRemoved(PipingCalculationGroupContext nodeData, object parentNodeData) { var parentGroupContext = (PipingCalculationGroupContext) parentNodeData; parentGroupContext.WrappedData.Children.Remove(nodeData.WrappedData); parentGroupContext.NotifyObservers(); } private StrictContextMenuItem CreateUpdateEntryAndExitPointItem(PipingCalculationGroupContext nodeData) { var contextMenuEnabled = true; string toolTipMessage = Resources.PipingPlugin_CreateUpdateEntryAndExitPointItem_Update_all_calculations_with_surface_line_ToolTip; IList calculationsToUpdate = nodeData.WrappedData.GetCalculations() .OfType() .Where(c => c.InputParameters.SurfaceLine != null && !c.InputParameters.IsEntryAndExitPointInputSynchronized) .ToList(); if (!calculationsToUpdate.Any()) { contextMenuEnabled = false; toolTipMessage = RingtoetsCommonFormsResources.CreateUpdateContextMenuItem_No_calculations_to_update_ToolTip; } return new StrictContextMenuItem( Resources.PipingPlugin_CreateUpdateEntryAndExitPointItem_Update_all_entry_and_exit_points, toolTipMessage, RingtoetsCommonFormsResources.UpdateItemIcon, (sender, args) => UpdateEntryAndExitPointsOfAllCalculations(calculationsToUpdate)) { Enabled = contextMenuEnabled }; } private void UpdateEntryAndExitPointsOfAllCalculations(IList calculations) { string message = RingtoetsCommonFormsResources.VerifyUpdate_Confirm_calculation_outputs_cleared; if (VerifyEntryAndExitPointUpdates(calculations, message)) { foreach (PipingCalculationScenario calculation in calculations) { UpdateSurfaceLineDependentData(calculation); } } } private bool VerifyEntryAndExitPointUpdates(IEnumerable calculations, string query) { var changeHandler = new CalculationChangeHandler(calculations, query, new DialogBasedInquiryHelper(Gui.MainWindow)); return !changeHandler.RequireConfirmation() || changeHandler.InquireConfirmation(); } #endregion #region Ringtoets piping surface line importer private static PipingSurfaceLinesCsvImporter PipingSurfaceLinesCsvImporter(RingtoetsPipingSurfaceLinesContext context, string filePath, IImporterMessageProvider messageProvider, ISurfaceLineUpdateDataStrategy strategy) { return new PipingSurfaceLinesCsvImporter(context.WrappedData, context.AssessmentSection.ReferenceLine, filePath, messageProvider, strategy); } private static FileFilterGenerator RingtoetsPipingSurfaceLineFileFilter { get { return new FileFilterGenerator( RingtoetsCommonFormsResources.DataTypeDisplayName_csv_file_filter_Extension, $"{MacroStabilityInwardsDataResources.PipingSurfaceLineCollection_TypeDescriptor} {RingtoetsCommonFormsResources.DataTypeDisplayName_csv_file_filter_Description}"); } } private bool VerifyPipingSurfaceLineUpdates(RingtoetsPipingSurfaceLinesContext context, string query) { var changeHandler = new FailureMechanismCalculationChangeHandler(context.FailureMechanism, query, new DialogBasedInquiryHelper(Gui.MainWindow)); return !changeHandler.RequireConfirmation() || changeHandler.InquireConfirmation(); } #endregion } }