using System; using System.Collections; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Linq; namespace ClaySharp.Implementation { public static class Arguments { public static INamedEnumerable FromT(IEnumerable arguments, IEnumerable names) { return new NamedEnumerable(arguments, names); } public static INamedEnumerable From(IEnumerable arguments, IEnumerable names) { return new NamedEnumerable(arguments, names); } class NamedEnumerable : INamedEnumerable { readonly IEnumerable _arguments; readonly IEnumerable _names; public NamedEnumerable(IEnumerable arguments, IEnumerable names) { if (arguments == null) { throw new ArgumentNullException("arguments"); } if (names == null) { throw new ArgumentNullException("names"); } if (arguments.Count() < names.Count()) { throw new ArgumentException("arguments.Count() < names.Count()"); } _arguments = arguments; _names = names; } IEnumerator IEnumerable.GetEnumerator() { return _arguments.GetEnumerator(); } IEnumerator IEnumerable.GetEnumerator() { return _arguments.GetEnumerator(); } IEnumerable INamedEnumerable.Positional { get { return _arguments.Take(_arguments.Count() - _names.Count()); } } IDictionary INamedEnumerable.Named { get { return new Named(_arguments, _names); } } class Named : IDictionary { private readonly IEnumerable _arguments; private readonly IEnumerable _names; private ICollection _argumentsCollection; private ICollection _namesCollection; public Named(IEnumerable arguments, IEnumerable names) { _arguments = arguments.Skip(arguments.Count() - names.Count()); _names = names; } IEnumerable> MakeEnumerable() { return _arguments.Zip(_names, (arg, name) => new KeyValuePair(name, arg)); } IEnumerator> IEnumerable>.GetEnumerator() { return MakeEnumerable().GetEnumerator(); } IEnumerator IEnumerable.GetEnumerator() { return MakeEnumerable().GetEnumerator(); } void ICollection>.Add(KeyValuePair item) { throw new NotImplementedException(); } void ICollection>.Clear() { throw new NotImplementedException(); } bool ICollection>.Contains(KeyValuePair item) { return MakeEnumerable().Contains(item); } void ICollection>.CopyTo(KeyValuePair[] array, int arrayIndex) { throw new NotImplementedException(); } bool ICollection>.Remove(KeyValuePair item) { throw new NotImplementedException(); } int ICollection>.Count { get { return _names.Count(); } } bool ICollection>.IsReadOnly { get { return true; } } bool IDictionary.ContainsKey(string key) { return _names.Contains(key); } void IDictionary.Add(string key, T value) { throw new NotImplementedException(); } bool IDictionary.Remove(string key) { throw new NotImplementedException(); } bool IDictionary.TryGetValue(string key, out T value) { var pair = MakeEnumerable().FirstOrDefault(kv => kv.Key == key); // pair is a value type. in case of key-miss, // will default to key=(string)null,value=(object)null value=pair.Value; return pair.Key != null; } //TBD T IDictionary.this[string key] { get { return MakeEnumerable() .Where(kv => kv.Key == key) .Select(kv => kv.Value) .FirstOrDefault(); } set { throw new NotImplementedException(); } } ICollection IDictionary.Keys { get { return _namesCollection = _namesCollection ?? _names as ICollection ?? _names.ToArray(); } } ICollection IDictionary.Values { get { return _argumentsCollection = _argumentsCollection ?? _arguments as ICollection ?? _arguments.ToArray(); } } } } public static INamedEnumerable Empty() { return From(Enumerable.Empty(), Enumerable.Empty()); } } }