#region License Information
/* HeuristicLab
 * Copyright (C) Heuristic and Evolutionary Algorithms Laboratory (HEAL)
 *
 * This file is part of HeuristicLab.
 *
 * HeuristicLab 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.
 *
 * HeuristicLab 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 HeuristicLab. If not, see <http://www.gnu.org/licenses/>.
 */
#endregion

using System;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HeuristicLab.Data;
using HeuristicLab.Operators;
using HeuristicLab.Optimization;
using HeuristicLab.Optimization.Operators;
using HeuristicLab.Parameters;
using HEAL.Attic;

namespace HeuristicLab.Analysis {
  /// <summary>
  /// An operator which analyzes the minimum, average and maximum of a value in the scope tree.
  /// </summary>
  [Item("MinAverageMaxValueAnalyzer", "An operator which analyzes the minimum, average and maximum of a value in the scope tree.")]
  [StorableType("D305BB40-F613-4DB5-8778-C0278551CAA3")]
  public sealed class MinAverageMaxValueAnalyzer : AlgorithmOperator, IAnalyzer {
    #region Parameter properties
    public ScopeTreeLookupParameter<DoubleValue> ValueParameter {
      get { return (ScopeTreeLookupParameter<DoubleValue>)Parameters["Value"]; }
    }
    public ValueLookupParameter<DoubleValue> MinValueParameter {
      get { return (ValueLookupParameter<DoubleValue>)Parameters["MinValue"]; }
    }
    public ValueLookupParameter<DoubleValue> AverageValueParameter {
      get { return (ValueLookupParameter<DoubleValue>)Parameters["AverageValue"]; }
    }
    public ValueLookupParameter<DoubleValue> MaxValueParameter {
      get { return (ValueLookupParameter<DoubleValue>)Parameters["MaxValue"]; }
    }
    public ValueLookupParameter<DataTable> ValuesParameter {
      get { return (ValueLookupParameter<DataTable>)Parameters["Values"]; }
    }
    public ValueLookupParameter<VariableCollection> ResultsParameter {
      get { return (ValueLookupParameter<VariableCollection>)Parameters["Results"]; }
    }
    public ValueParameter<BoolValue> CollectMinValueInResultsParameter {
      get { return (ValueParameter<BoolValue>)Parameters["CollectMinValueInResults"]; }
    }
    public ValueParameter<BoolValue> CollectMaxValueInResultsParameter {
      get { return (ValueParameter<BoolValue>)Parameters["CollectMaxValueInResults"]; }
    }
    public ValueParameter<BoolValue> CollectAverageValueInResultsParameter {
      get { return (ValueParameter<BoolValue>)Parameters["CollectAverageValueInResults"]; }
    }
    #endregion

    #region Properties
    public bool EnabledByDefault {
      get { return true; }
    }
    private MinAverageMaxValueCalculator MinAverageMaxValueCalculator {
      get { return (MinAverageMaxValueCalculator)OperatorGraph.InitialOperator; }
    }
    private BoolValue CollectMinValueInResults {
      get { return CollectMinValueInResultsParameter.Value; }
    }
    private BoolValue CollectMaxValueInResults {
      get { return CollectMaxValueInResultsParameter.Value; }
    }
    private BoolValue CollectAverageValueInResults {
      get { return CollectAverageValueInResultsParameter.Value; }
    }
    #endregion

    [Storable]
    private ResultsCollector resultsCollector;

    #region Storing & Cloning
    [StorableConstructor]
    private MinAverageMaxValueAnalyzer(StorableConstructorFlag _) : base(_) { }
    private MinAverageMaxValueAnalyzer(MinAverageMaxValueAnalyzer original, Cloner cloner)
      : base(original, cloner) {
      resultsCollector = cloner.Clone(original.resultsCollector);
      RegisterEventHandlers();
    }
    public override IDeepCloneable Clone(Cloner cloner) {
      return new MinAverageMaxValueAnalyzer(this, cloner);
    }
    #endregion
    public MinAverageMaxValueAnalyzer()
      : base() {
      #region Create parameters
      Parameters.Add(new ScopeTreeLookupParameter<DoubleValue>("Value", "The value contained in the scope tree which should be analyzed."));
      Parameters.Add(new ValueLookupParameter<DoubleValue>("MinValue", "The minimum of the value."));
      Parameters.Add(new ValueParameter<BoolValue>("CollectMinValueInResults", "Determines if the minimum of the value should also be stored in the results.", new BoolValue(true)));
      Parameters.Add(new ValueLookupParameter<DoubleValue>("AverageValue", "The average of the value."));
      Parameters.Add(new ValueParameter<BoolValue>("CollectAverageValueInResults", "Determines if the average of the value should also be stored in the results.", new BoolValue(true)));
      Parameters.Add(new ValueLookupParameter<DoubleValue>("MaxValue", "The maximum of the value."));
      Parameters.Add(new ValueParameter<BoolValue>("CollectMaxValueInResults", "Determines if the maximum of the value should also be stored in the results.", new BoolValue(true)));
      Parameters.Add(new ValueLookupParameter<DataTable>("Values", "The data table to store the minimum, average and maximum of the value."));
      Parameters.Add(new ValueLookupParameter<VariableCollection>("Results", "The results collection where the analysis values should be stored."));
      #endregion

      #region Create operators
      MinAverageMaxValueCalculator minAverageMaxValueCalculator = new MinAverageMaxValueCalculator();
      DataTableValuesCollector dataTableValuesCollector = new DataTableValuesCollector();
      resultsCollector = new ResultsCollector();

      minAverageMaxValueCalculator.AverageValueParameter.ActualName = AverageValueParameter.Name;
      minAverageMaxValueCalculator.MaxValueParameter.ActualName = MaxValueParameter.Name;
      minAverageMaxValueCalculator.MinValueParameter.ActualName = MinValueParameter.Name;
      minAverageMaxValueCalculator.ValueParameter.ActualName = ValueParameter.Name;
      minAverageMaxValueCalculator.ValueParameter.Depth = ValueParameter.Depth;

      dataTableValuesCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("MinValue", null, MinValueParameter.Name));
      dataTableValuesCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("AverageValue", null, AverageValueParameter.Name));
      dataTableValuesCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("MaxValue", null, MaxValueParameter.Name));
      dataTableValuesCollector.DataTableParameter.ActualName = ValuesParameter.Name;

      if (CollectMinValueInResults.Value)
        resultsCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("MinValue", null, MinValueParameter.Name));
      if (CollectAverageValueInResults.Value)
        resultsCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("AverageValue", null, AverageValueParameter.Name));
      if (CollectMaxValueInResults.Value)
        resultsCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("MaxValue", null, MaxValueParameter.Name));
      resultsCollector.CollectedValues.Add(new LookupParameter<DataTable>(ValuesParameter.Name));
      resultsCollector.ResultsParameter.ActualName = ResultsParameter.Name;
      #endregion

      #region Create operator graph
      OperatorGraph.InitialOperator = minAverageMaxValueCalculator;
      minAverageMaxValueCalculator.Successor = dataTableValuesCollector;
      dataTableValuesCollector.Successor = resultsCollector;
      resultsCollector.Successor = null;
      #endregion

      RegisterEventHandlers();
    }

    [StorableHook(HookType.AfterDeserialization)]
    private void AfterDeserialization() {
      RegisterEventHandlers();
    }

    private void RegisterEventHandlers() {
      ValueParameter.DepthChanged += new EventHandler(ValueParameter_DepthChanged);
      CollectMinValueInResultsParameter.ValueChanged += new EventHandler(CollectMinValueInResultsParameter_ValueChanged);
      CollectMinValueInResultsParameter.Value.ValueChanged += new EventHandler(CollectMinValueInResultsParameter_Value_ValueChanged);
      CollectMaxValueInResultsParameter.ValueChanged += new EventHandler(CollectMaxValueInResultsParameter_ValueChanged);
      CollectMaxValueInResultsParameter.Value.ValueChanged += new EventHandler(CollectMaxValueInResultsParameter_Value_ValueChanged);
      CollectAverageValueInResultsParameter.ValueChanged += new EventHandler(CollectAverageValueInResultsParameter_ValueChanged);
      CollectAverageValueInResultsParameter.Value.ValueChanged += new EventHandler(CollectAverageValueInResultsParameter_Value_ValueChanged);
    }

    private void CollectAverageValueInResultsParameter_ValueChanged(object sender, EventArgs e) {
      IParameter avgValueParameter;
      resultsCollector.CollectedValues.TryGetValue("AverageValue", out avgValueParameter);
      if (CollectAverageValueInResults.Value && avgValueParameter == null)
        resultsCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("AverageValue", null, AverageValueParameter.Name));
      else if (!CollectAverageValueInResults.Value && avgValueParameter != null)
        resultsCollector.CollectedValues.Remove(avgValueParameter);
      CollectAverageValueInResultsParameter.Value.ValueChanged += new EventHandler(CollectAverageValueInResultsParameter_Value_ValueChanged);
    }

    private void CollectAverageValueInResultsParameter_Value_ValueChanged(object sender, EventArgs e) {
      IParameter avgValueParameter;
      resultsCollector.CollectedValues.TryGetValue("AverageValue", out avgValueParameter);
      if (CollectAverageValueInResults.Value && avgValueParameter == null)
        resultsCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("AverageValue", null, AverageValueParameter.Name));
      else if (!CollectAverageValueInResults.Value && avgValueParameter != null)
        resultsCollector.CollectedValues.Remove(avgValueParameter);
    }

    private void CollectMaxValueInResultsParameter_ValueChanged(object sender, EventArgs e) {
      IParameter maxValueParameter;
      resultsCollector.CollectedValues.TryGetValue("MaxValue", out maxValueParameter);
      if (CollectMaxValueInResults.Value && maxValueParameter == null)
        resultsCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("MaxValue", null, MaxValueParameter.Name));
      else if (!CollectMaxValueInResults.Value && maxValueParameter != null)
        resultsCollector.CollectedValues.Remove(maxValueParameter);
      CollectMaxValueInResultsParameter.Value.ValueChanged += new EventHandler(CollectMaxValueInResultsParameter_Value_ValueChanged);
    }

    private void CollectMaxValueInResultsParameter_Value_ValueChanged(object sender, EventArgs e) {
      IParameter maxValueParameter;
      resultsCollector.CollectedValues.TryGetValue("MaxValue", out maxValueParameter);
      if (CollectMaxValueInResults.Value && maxValueParameter == null)
        resultsCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("MaxValue", null, MaxValueParameter.Name));
      else if (!CollectMaxValueInResults.Value && maxValueParameter != null)
        resultsCollector.CollectedValues.Remove(maxValueParameter);
    }

    private void CollectMinValueInResultsParameter_ValueChanged(object sender, EventArgs e) {
      IParameter minValueParameter;
      resultsCollector.CollectedValues.TryGetValue("MinValue", out minValueParameter);
      if (CollectMinValueInResults.Value && minValueParameter == null)
        resultsCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("MinValue", null, MinValueParameter.Name));
      else if (!CollectMinValueInResults.Value && minValueParameter != null)
        resultsCollector.CollectedValues.Remove(minValueParameter);
      CollectMinValueInResultsParameter.Value.ValueChanged += new EventHandler(CollectMinValueInResultsParameter_Value_ValueChanged);
    }

    private void CollectMinValueInResultsParameter_Value_ValueChanged(object sender, EventArgs e) {
      IParameter minValueParameter;
      resultsCollector.CollectedValues.TryGetValue("MinValue", out minValueParameter);
      if (CollectMinValueInResults.Value && minValueParameter == null)
        resultsCollector.CollectedValues.Add(new LookupParameter<DoubleValue>("MinValue", null, MinValueParameter.Name));
      else if (!CollectMinValueInResults.Value && minValueParameter != null)
        resultsCollector.CollectedValues.Remove(minValueParameter);
    }

    private void ValueParameter_DepthChanged(object sender, System.EventArgs e) {
      MinAverageMaxValueCalculator.ValueParameter.Depth = ValueParameter.Depth;
    }
  }
}