Free cookie consent management tool by TermsFeed Policy Generator

source: trunk/sources/HeuristicLab.Parameters/3.3/OptionalValueParameter.cs @ 4825

Last change on this file since 4825 was 4825, checked in by swagner, 14 years ago

Fixed memory leak when using parameters whose values are not deeply cloned (#1268)

File size: 8.1 KB
Line 
1#region License Information
2/* HeuristicLab
3 * Copyright (C) 2002-2010 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
4 *
5 * This file is part of HeuristicLab.
6 *
7 * HeuristicLab is free software: you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation, either version 3 of the License, or
10 * (at your option) any later version.
11 *
12 * HeuristicLab is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with HeuristicLab. If not, see <http://www.gnu.org/licenses/>.
19 */
20#endregion
21
22using System;
23using System.Drawing;
24using HeuristicLab.Common;
25using HeuristicLab.Core;
26using HeuristicLab.Persistence.Default.CompositeSerializers.Storable;
27
28namespace HeuristicLab.Parameters {
29  /// <summary>
30  /// A parameter whose value is defined in the parameter itself or is null.
31  /// </summary>
32  [Item("OptionalValueParameter", "A parameter whose value is defined in the parameter itself or is null.")]
33  [StorableClass]
34  public class OptionalValueParameter<T> : Parameter, IValueParameter<T> where T : class, IItem {
35    public override Image ItemImage {
36      get {
37        if (value != null) return value.ItemImage;
38        else return base.ItemImage;
39      }
40    }
41
42    [Storable]
43    private T value;
44    public virtual T Value {
45      get { return this.value; }
46      set {
47        if (value != this.value) {
48          DeregisterValueEvents();
49          this.value = value;
50          RegisterValueEvents();
51          OnValueChanged();
52        }
53      }
54    }
55    IItem IValueParameter.Value {
56      get { return Value; }
57      set {
58        T val = value as T;
59        if ((value != null) && (val == null))
60          throw new InvalidOperationException(
61            string.Format("Type mismatch. Value is not a \"{0}\".",
62                          typeof(T).GetPrettyName())
63          );
64        Value = val;
65      }
66    }
67
68    [Storable(DefaultValue = true)]
69    private bool getsCollected;
70    public bool GetsCollected {
71      get { return getsCollected; }
72      set {
73        if (value != getsCollected) {
74          getsCollected = value;
75          OnGetsCollectedChanged();
76        }
77      }
78    }
79
80    [Storable(DefaultValue = true)]
81    private bool reactOnValueToStringChangedAndValueItemImageChanged;
82    /// <summary>
83    ///   True if this parameter should react on the ToStringChanged and ItemImageChanged events of its value, otherwise false.
84    /// </summary>
85    /// <remarks>
86    ///   In some rare cases when the value of the parameter is not deeply cloned, this property has to be set to false
87    ///   to avoid a memory leak (cf. ticket #1268). In all other cases this property should always be true.
88    /// </remarks>
89    public bool ReactOnValueToStringChangedAndValueItemImageChanged {
90      get { return reactOnValueToStringChangedAndValueItemImageChanged; }
91      set {
92        if (value != reactOnValueToStringChangedAndValueItemImageChanged) {
93          reactOnValueToStringChangedAndValueItemImageChanged = value;
94          if (reactOnValueToStringChangedAndValueItemImageChanged)
95            RegisterValueEvents();
96          else
97            DeregisterValueEvents();
98        }
99      }
100    }
101
102    #region Constructors
103    [StorableConstructor]
104    protected OptionalValueParameter(bool deserializing) : base(deserializing) { }
105    protected OptionalValueParameter(OptionalValueParameter<T> original, Cloner cloner)
106      : base(original, cloner) {
107      value = cloner.Clone(original.value);
108      getsCollected = original.getsCollected;
109      reactOnValueToStringChangedAndValueItemImageChanged = original.reactOnValueToStringChangedAndValueItemImageChanged;
110      RegisterValueEvents();
111    }
112    public OptionalValueParameter()
113      : base("Anonymous", typeof(T)) {
114      this.getsCollected = true;
115      this.reactOnValueToStringChangedAndValueItemImageChanged = true;
116    }
117    public OptionalValueParameter(string name)
118      : base(name, typeof(T)) {
119      this.getsCollected = true;
120      this.reactOnValueToStringChangedAndValueItemImageChanged = true;
121    }
122    public OptionalValueParameter(string name, bool getsCollected)
123      : base(name, typeof(T)) {
124      this.getsCollected = getsCollected;
125      this.reactOnValueToStringChangedAndValueItemImageChanged = true;
126    }
127    public OptionalValueParameter(string name, T value)
128      : base(name, typeof(T)) {
129      this.value = value;
130      this.getsCollected = true;
131      this.reactOnValueToStringChangedAndValueItemImageChanged = true;
132      RegisterValueEvents();
133    }
134    public OptionalValueParameter(string name, T value, bool getsCollected)
135      : base(name, typeof(T)) {
136      this.value = value;
137      this.getsCollected = getsCollected;
138      this.reactOnValueToStringChangedAndValueItemImageChanged = true;
139      RegisterValueEvents();
140    }
141    public OptionalValueParameter(string name, string description)
142      : base(name, description, typeof(T)) {
143      this.getsCollected = true;
144      this.reactOnValueToStringChangedAndValueItemImageChanged = true;
145    }
146    public OptionalValueParameter(string name, string description, bool getsCollected)
147      : base(name, description, typeof(T)) {
148      this.getsCollected = getsCollected;
149      this.reactOnValueToStringChangedAndValueItemImageChanged = true;
150    }
151    public OptionalValueParameter(string name, string description, T value)
152      : base(name, description, typeof(T)) {
153      this.value = value;
154      this.getsCollected = true;
155      this.reactOnValueToStringChangedAndValueItemImageChanged = true;
156      RegisterValueEvents();
157    }
158    public OptionalValueParameter(string name, string description, T value, bool getsCollected)
159      : base(name, description, typeof(T)) {
160      this.value = value;
161      this.getsCollected = getsCollected;
162      this.reactOnValueToStringChangedAndValueItemImageChanged = true;
163      RegisterValueEvents();
164    }
165    #endregion
166
167    [StorableHook(HookType.AfterDeserialization)]
168    private void AfterDeserialization() {
169      RegisterValueEvents();
170    }
171
172    public override IDeepCloneable Clone(Cloner cloner) {
173      return new OptionalValueParameter<T>(this, cloner);
174    }
175
176    public override string ToString() {
177      if (reactOnValueToStringChangedAndValueItemImageChanged)
178        return Name + ": " + (Value != null ? Value.ToString() : "null");
179      else
180        return Name;
181    }
182
183    protected override IItem GetActualValue() {
184      return Value;
185    }
186    protected override void SetActualValue(IItem value) {
187      ((IValueParameter)this).Value = value;
188    }
189
190    public event EventHandler ValueChanged;
191    protected virtual void OnValueChanged() {
192      EventHandler handler = ValueChanged;
193      if (handler != null) handler(this, EventArgs.Empty);
194      OnItemImageChanged();
195      OnToStringChanged();
196    }
197    public event EventHandler GetsCollectedChanged;
198    protected virtual void OnGetsCollectedChanged() {
199      EventHandler handler = GetsCollectedChanged;
200      if (handler != null) handler(this, EventArgs.Empty);
201    }
202
203    private void RegisterValueEvents() {
204      if ((value != null) && reactOnValueToStringChangedAndValueItemImageChanged) {
205        value.ItemImageChanged += new EventHandler(Value_ItemImageChanged);
206        value.ToStringChanged += new EventHandler(Value_ToStringChanged);
207      }
208    }
209    private void DeregisterValueEvents() {
210      if (value != null) {
211        value.ItemImageChanged -= new EventHandler(Value_ItemImageChanged);
212        value.ToStringChanged -= new EventHandler(Value_ToStringChanged);
213      }
214    }
215    private void Value_ItemImageChanged(object sender, EventArgs e) {
216      OnItemImageChanged();
217    }
218    private void Value_ToStringChanged(object sender, EventArgs e) {
219      OnToStringChanged();
220    }
221  }
222}
Note: See TracBrowser for help on using the repository browser.