Mercurial > pub > bltoolkit
diff Source/Mapping/DictionaryListMapperT.cs @ 0:f990fcb411a9
Копия текущей версии из github
author | cin |
---|---|
date | Thu, 27 Mar 2014 21:46:09 +0400 (2014-03-27) |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/Source/Mapping/DictionaryListMapperT.cs Thu Mar 27 21:46:09 2014 +0400 @@ -0,0 +1,120 @@ +using System; +using System.Collections.Generic; +using System.Diagnostics; +using BLToolkit.Common; +using BLToolkit.Reflection; + +namespace BLToolkit.Mapping +{ + public class DictionaryListMapper<K,T> : IMapDataDestinationList + { + public DictionaryListMapper( + IDictionary<K,T> dic, + NameOrIndexParameter keyField, + ObjectMapper objectMapper) + { + _dic = dic; + _mapper = objectMapper; + _keyGetter = MapGetData<K>.I; + _fromSource = keyField.ByName && keyField.Name[0] == '@'; + _keyField = _fromSource? keyField.Name.Substring(1): keyField; + } + + private readonly IDictionary<K,T> _dic; + private readonly bool _fromSource; + private readonly MapGetData<K>.MB<K> _keyGetter; + private NameOrIndexParameter _keyField; + private int _index; + private ObjectMapper _mapper; + private object _newObject; + private bool _typeMismatch; + private K _keyValue; + + #region IMapDataDestinationList Members + + private void AddObject() + { + if (_newObject != null) + { + if (_typeMismatch) + _keyValue = _mapper.MappingSchema.ConvertTo<K,object>(_mapper[_index].GetValue(_newObject)); + else if (!_fromSource) + _keyValue = _keyGetter.From(_mapper, _newObject, _index); + + _dic[_keyValue] = (T)_newObject; + } + } + + public virtual void InitMapping(InitContext initContext) + { + var sm = _dic as ISupportMapping; + + if (sm != null) + { + sm.BeginMapping(initContext); + + if (_mapper != initContext.ObjectMapper) + _mapper = initContext.ObjectMapper; + } + + if (_fromSource) + _index = _keyField.ByName? initContext.DataSource.GetOrdinal(_keyField.Name): _keyField.Index; + else + { + _index = _keyField.ByName? _mapper.GetOrdinal(_keyField.Name, true): _keyField.Index; + + if (_index < 0) + throw new MappingException( + _keyField.ByName? + string.Format("Field '{0}' not found.", _keyField.Name): + string.Format("Index '{0}' is invalid.", _keyField.Index)); + + var mm = _mapper[_index]; + _typeMismatch = !TypeHelper.IsSameOrParent(typeof(K), mm.Type); + +#if !SILVERLIGHT + + Debug.WriteLineIf(_typeMismatch, string.Format( + "Member {0} type '{1}' does not match dictionary key type '{2}'.", + mm.Name, mm.Type.Name, (typeof(K).Name))); + +#endif + } + } + + [CLSCompliant(false)] + public virtual IMapDataDestination GetDataDestination(InitContext initContext) + { + return _mapper; + } + + static readonly char[] _trim = { ' ' }; + + public virtual object GetNextObject(InitContext initContext) + { + AddObject(); + + if (_fromSource) + { + _keyValue = _keyGetter.From(initContext.DataSource, initContext.SourceObject, _index); + + if (Common.Configuration.TrimDictionaryKey && _keyValue is string) + _keyValue = (K)(object)_keyValue.ToString().TrimEnd(_trim); + } + + return _newObject = _mapper.CreateInstance(initContext); + } + + public virtual void EndMapping(InitContext initContext) + { + AddObject(); + + ISupportMapping sm = _dic as ISupportMapping; + + if (sm != null) + sm.EndMapping(initContext); + } + + #endregion + } +}