// 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.Linq;
using Core.Common.Base;
using Ringtoets.Piping.Data;
using Ringtoets.Piping.Service;
namespace Ringtoets.Piping.Plugin.FileImporter
{
///
/// Strategy for updating the current stochastic soil models with the imported stochastic soil models.
///
public class StochasticSoilModelUpdateData : IStochasticSoilModelUpdateStrategy
{
private readonly PipingFailureMechanism failureMechanism;
///
/// Creates a new instance of .
///
/// The failure mechanism in which the models are updated.
public StochasticSoilModelUpdateData(PipingFailureMechanism failureMechanism)
{
if (failureMechanism == null)
{
throw new ArgumentNullException(nameof(failureMechanism));
}
this.failureMechanism = failureMechanism;
}
///
/// Updates the .
/// Updates stochastic soil models in that are part of
/// .
/// Adds stochastic soil models that are not in , but are part of
/// .
/// Removes stochastic soil models that are in , but are not part
/// of .
///
/// The imported stochastic soil models.
/// The file path from which the
/// were imported.
/// The current collection of .
/// Thrown when
/// contains multiple with the same ,
/// and also contains a with
/// the same name.
///
/// Thrown when any parameter is null.
/// List of updated instances.
public IEnumerable UpdateModelWithImportedData(
IEnumerable readStochasticSoilModels,
string sourceFilePath,
StochasticSoilModelCollection targetCollection)
{
if (readStochasticSoilModels == null)
{
throw new ArgumentNullException(nameof(readStochasticSoilModels));
}
if (targetCollection == null)
{
throw new ArgumentNullException(nameof(targetCollection));
}
return ModifyModelCollection(readStochasticSoilModels, targetCollection, sourceFilePath);
}
private IEnumerable ModifyModelCollection(IEnumerable readStochasticSoilModels, StochasticSoilModelCollection targetCollection, string sourceFilePath)
{
var affectedObjects = new List
{
targetCollection
};
var readModelList = readStochasticSoilModels.ToList();
var addedModels = GetAddedReadModels(targetCollection, readModelList).ToList();
var updatedModels = GetUpdatedExistingModels(targetCollection, readModelList).ToList();
var removedModels = GetRemovedExistingModels(targetCollection, readModelList).ToList();
affectedObjects.AddRange(UpdateModels(updatedModels, readModelList));
affectedObjects.AddRange(RemoveModels(removedModels));
targetCollection.Clear();
targetCollection.AddRange(addedModels.Union(updatedModels), sourceFilePath);
return affectedObjects;
}
private static IEnumerable GetAddedReadModels(IEnumerable existingCollection, IEnumerable readStochasticSoilModels)
{
return readStochasticSoilModels.Except(existingCollection, new SoilModelNameEqualityComparer());
}
private static IEnumerable GetUpdatedExistingModels(IEnumerable existingCollection, IEnumerable readStochasticSoilModels)
{
return existingCollection.Intersect(readStochasticSoilModels, new SoilModelNameEqualityComparer());
}
private static IEnumerable GetRemovedExistingModels(IEnumerable existingCollection, IEnumerable readStochasticSoilModels)
{
return existingCollection.Except(readStochasticSoilModels, new SoilModelNameEqualityComparer());
}
private IEnumerable RemoveModels(IEnumerable removedModels)
{
var affectedObjects = new List();
foreach (var model in removedModels)
{
affectedObjects.AddRange(ClearStochasticSoilModelDependentData(model));
}
return affectedObjects;
}
private IEnumerable UpdateModels(IList updatedModels, IList readModels)
{
var affectedObjects = new List();
foreach (StochasticSoilModel updatedModel in updatedModels)
{
StochasticSoilModel readModel = readModels.Single(r => r.Name.Equals(updatedModel.Name));
affectedObjects.AddRange(UpdateStochasticSoilModel(updatedModel, readModel));
}
return affectedObjects;
}
private IEnumerable ClearStochasticSoilModelDependentData(StochasticSoilModel removedModel)
{
return PipingDataSynchronizationService.RemoveStochasticSoilModel(failureMechanism, removedModel);
}
private IEnumerable UpdateStochasticSoilModel(StochasticSoilModel existingModel, StochasticSoilModel readModel)
{
StochasticSoilModelProfileDifference difference = existingModel.Update(readModel);
var affectedObjects = new List();
foreach (StochasticSoilProfile removedProfile in difference.RemovedProfiles)
{
affectedObjects.AddRange(PipingDataSynchronizationService.RemoveStochasticSoilProfileFromInput(failureMechanism, removedProfile));
}
foreach (StochasticSoilProfile updatedProfile in difference.UpdatedProfiles)
{
affectedObjects.AddRange(PipingDataSynchronizationService.UpdateStochasticSoilProfileForInput(failureMechanism, updatedProfile));
}
return affectedObjects;
}
///
/// Class for comparing by just the name.
///
private class SoilModelNameEqualityComparer : IEqualityComparer
{
public bool Equals(StochasticSoilModel x, StochasticSoilModel y)
{
return x.Name == y.Name;
}
public int GetHashCode(StochasticSoilModel obj)
{
return obj.Name.GetHashCode();
}
}
}
}