//----------------------------------------------------------------------------- // Copyright (c) Microsoft Corporation. All rights reserved. //----------------------------------------------------------------------------- namespace System.Runtime.Collections { using System; using System.Collections; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Runtime; class NullableKeyDictionary : IDictionary { bool isNullKeyPresent; TValue nullKeyValue; IDictionary innerDictionary; public NullableKeyDictionary() : base() { this.innerDictionary = new Dictionary(); } public int Count { get { return this.innerDictionary.Count + (this.isNullKeyPresent ? 1 : 0); } } public bool IsReadOnly { get { return false; } } public ICollection Keys { get { return new NullKeyDictionaryKeyCollection(this); } } public ICollection Values { get { return new NullKeyDictionaryValueCollection(this); } } public TValue this[TKey key] { get { if (key == null) { if (this.isNullKeyPresent) { return this.nullKeyValue; } else { throw Fx.Exception.AsError(new KeyNotFoundException()); } } else { return this.innerDictionary[key]; } } set { if (key == null) { this.isNullKeyPresent = true; this.nullKeyValue = value; } else { this.innerDictionary[key] = value; } } } public void Add(TKey key, TValue value) { if (key == null) { if (this.isNullKeyPresent) { throw Fx.Exception.Argument("key", InternalSR.NullKeyAlreadyPresent); } this.isNullKeyPresent = true; this.nullKeyValue = value; } else { this.innerDictionary.Add(key, value); } } public bool ContainsKey(TKey key) { return key == null ? this.isNullKeyPresent : this.innerDictionary.ContainsKey(key); } public bool Remove(TKey key) { if (key == null) { bool result = this.isNullKeyPresent; this.isNullKeyPresent = false; this.nullKeyValue = default(TValue); return result; } else { return this.innerDictionary.Remove(key); } } public bool TryGetValue(TKey key, out TValue value) { if (key == null) { if (this.isNullKeyPresent) { value = this.nullKeyValue; return true; } else { value = default(TValue); return false; } } else { return this.innerDictionary.TryGetValue(key, out value); } } public void Add(KeyValuePair item) { Add(item.Key, item.Value); } public void Clear() { this.isNullKeyPresent = false; this.nullKeyValue = default(TValue); this.innerDictionary.Clear(); } public bool Contains(KeyValuePair item) { if (item.Key == null) { if (this.isNullKeyPresent) { return item.Value == null ? this.nullKeyValue == null : item.Value.Equals(this.nullKeyValue); } else { return false; } } else { return this.innerDictionary.Contains(item); } } public void CopyTo(KeyValuePair[] array, int arrayIndex) { this.innerDictionary.CopyTo(array, arrayIndex); if (this.isNullKeyPresent) { array[arrayIndex + this.innerDictionary.Count] = new KeyValuePair(default(TKey), this.nullKeyValue); } } public bool Remove(KeyValuePair item) { if (item.Key == null) { if (this.Contains(item)) { this.isNullKeyPresent = false; this.nullKeyValue = default(TValue); return true; } else { return false; } } else { return this.innerDictionary.Remove(item); } } public IEnumerator> GetEnumerator() { IEnumerator> innerEnumerator = this.innerDictionary.GetEnumerator() as IEnumerator>; while (innerEnumerator.MoveNext()) { yield return innerEnumerator.Current; } if (this.isNullKeyPresent) { yield return new KeyValuePair(default(TKey), this.nullKeyValue); } } IEnumerator IEnumerable.GetEnumerator() { return ((IEnumerable>)this).GetEnumerator(); } class NullKeyDictionaryKeyCollection : ICollection { NullableKeyDictionary nullKeyDictionary; public NullKeyDictionaryKeyCollection(NullableKeyDictionary nullKeyDictionary) { this.nullKeyDictionary = nullKeyDictionary; } public int Count { get { int count = this.nullKeyDictionary.innerDictionary.Keys.Count; if (this.nullKeyDictionary.isNullKeyPresent) { count++; } return count; } } public bool IsReadOnly { get { return true; } } public void Add(TypeKey item) { throw Fx.Exception.AsError(new NotSupportedException(InternalSR.KeyCollectionUpdatesNotAllowed)); } public void Clear() { throw Fx.Exception.AsError(new NotSupportedException(InternalSR.KeyCollectionUpdatesNotAllowed)); } public bool Contains(TypeKey item) { return item == null ? this.nullKeyDictionary.isNullKeyPresent : this.nullKeyDictionary.innerDictionary.Keys.Contains(item); } public void CopyTo(TypeKey[] array, int arrayIndex) { this.nullKeyDictionary.innerDictionary.Keys.CopyTo(array, arrayIndex); if (this.nullKeyDictionary.isNullKeyPresent) { array[arrayIndex + this.nullKeyDictionary.innerDictionary.Keys.Count] = default(TypeKey); } } public bool Remove(TypeKey item) { throw Fx.Exception.AsError(new NotSupportedException(InternalSR.KeyCollectionUpdatesNotAllowed)); } public IEnumerator GetEnumerator() { foreach (TypeKey item in this.nullKeyDictionary.innerDictionary.Keys) { yield return item; } if (this.nullKeyDictionary.isNullKeyPresent) { yield return default(TypeKey); } } IEnumerator IEnumerable.GetEnumerator() { return ((IEnumerable)this).GetEnumerator(); } } class NullKeyDictionaryValueCollection : ICollection { NullableKeyDictionary nullKeyDictionary; public NullKeyDictionaryValueCollection(NullableKeyDictionary nullKeyDictionary) { this.nullKeyDictionary = nullKeyDictionary; } public int Count { get { int count = this.nullKeyDictionary.innerDictionary.Values.Count; if (this.nullKeyDictionary.isNullKeyPresent) { count++; } return count; } } public bool IsReadOnly { get { return true; } } public void Add(TypeValue item) { throw Fx.Exception.AsError(new NotSupportedException(InternalSR.ValueCollectionUpdatesNotAllowed)); } public void Clear() { throw Fx.Exception.AsError(new NotSupportedException(InternalSR.ValueCollectionUpdatesNotAllowed)); } public bool Contains(TypeValue item) { return this.nullKeyDictionary.innerDictionary.Values.Contains(item) || (this.nullKeyDictionary.isNullKeyPresent && this.nullKeyDictionary.nullKeyValue.Equals(item)); } public void CopyTo(TypeValue[] array, int arrayIndex) { this.nullKeyDictionary.innerDictionary.Values.CopyTo(array, arrayIndex); if (this.nullKeyDictionary.isNullKeyPresent) { array[arrayIndex + this.nullKeyDictionary.innerDictionary.Values.Count] = this.nullKeyDictionary.nullKeyValue; } } public bool Remove(TypeValue item) { throw Fx.Exception.AsError(new NotSupportedException(InternalSR.ValueCollectionUpdatesNotAllowed)); } public IEnumerator GetEnumerator() { foreach (TypeValue item in this.nullKeyDictionary.innerDictionary.Values) { yield return item; } if (this.nullKeyDictionary.isNullKeyPresent) { yield return this.nullKeyDictionary.nullKeyValue; } } IEnumerator IEnumerable.GetEnumerator() { return ((IEnumerable)this).GetEnumerator(); } } } }