I have two models for my form, a ViewModel going to it, and a ControlModel coming from it. The ControlModel has all the same field names and hierarchy, but all of the fields are a string data type.
How would you code AutoMapper to convert a string field to integer? I tried Int32.Parse(myString) but Int32 is not available within the expression (gives an error).
Mapper.CreateMap<SourceClass, DestinationClass>()
.ForMember(dest => dest.myInteger,
opt => opt.MapFrom(src => src.myString));
The types in the class and their corresponding conversion types:
string to int, int?, double, double?, DateTime, and bool
Additionally, is there any way to generalize mappings in a way that all integers in the target are parsed with that function? In other words, is there a way to create mappings for data types?
EDIT:
This looks promising:
AutoMapper.Mapper.CreateMap<string, int>()
.ConvertUsing(src => Convert.ToInt32(src));
EDIT:
This post is really helpful
I ended up doing something like this:
Mapper.CreateMap<string, int>().ConvertUsing<IntTypeConverter>();
Mapper.CreateMap<string, int?>().ConvertUsing<NullIntTypeConverter>();
Mapper.CreateMap<string, decimal?>().ConvertUsing<NullDecimalTypeConverter>();
Mapper.CreateMap<string, decimal>().ConvertUsing<DecimalTypeConverter>();
Mapper.CreateMap<string, bool?>().ConvertUsing<NullBooleanTypeConverter>();
Mapper.CreateMap<string, bool>().ConvertUsing<BooleanTypeConverter>();
Mapper.CreateMap<string, Int64?>().ConvertUsing<NullInt64TypeConverter>();
Mapper.CreateMap<string, Int64>().ConvertUsing<Int64TypeConverter>();
Mapper.CreateMap<string, DateTime?>().ConvertUsing<NullDateTimeTypeConverter>();
Mapper.CreateMap<string, DateTime>().ConvertUsing<DateTimeTypeConverter>();
Mapper.CreateMap<SourceClass, DestClass>();
Mapper.Map(mySourceObject, myDestinationObject);
And the classes it references (first draft):
// TODO: Boil down to two with Generics if possible
#region AutoMapTypeConverters
// Automap type converter definitions for
// int, int?, decimal, decimal?, bool, bool?, Int64, Int64?, DateTime
// Automapper string to int?
private class NullIntTypeConverter : TypeConverter<string, int?>
{ protected override int? ConvertCore(string source)
{ if (source == null)
return null;
else
{ int result;
return Int32.TryParse(source, out result) ? (int?) result : null;
} } }
// Automapper string to int
private class IntTypeConverter : TypeConverter<string, int>
{ protected override int ConvertCore(string source)
{ if (source == null)
throw new MappingException("null string value cannot convert to non-nullable return type.");
else
return Int32.Parse(source);
} }
// Automapper string to decimal?
private class NullDecimalTypeConverter : TypeConverter<string, decimal?>
{ protected override decimal? ConvertCore(string source)
{ if (source == null)
return null;
else
{ decimal result;
return Decimal.TryParse(source, out result) ? (decimal?) result : null;
} } }
// Automapper string to decimal
private class DecimalTypeConverter : TypeConverter<string, decimal>
{ protected override decimal ConvertCore(string source)
{ if (source == null)
throw new MappingException("null string value cannot convert to non-nullable return type.");
else
return Decimal.Parse(source);
} }
// Automapper string to bool?
private class NullBooleanTypeConverter : TypeConverter<string, bool?>
{ protected override bool? ConvertCore(string source)
{ if (source == null)
return null;
else
{ bool result;
return Boolean.TryParse(source, out result) ? (bool?) result : null;
} } }
// Automapper string to bool
private class BooleanTypeConverter : TypeConverter<string, bool>
{ protected override bool ConvertCore(string source)
{ if (source == null)
throw new MappingException("null string value cannot convert to non-nullable return type.");
else
return Boolean.Parse(source);
} }
// Automapper string to Int64?
private class NullInt64TypeConverter : TypeConverter<string, Int64?>
{ protected override Int64? ConvertCore(string source)
{ if (source == null)
return null;
else
{ Int64 result;
return Int64.TryParse(source, out result) ? (Int64?)result : null;
} } }
// Automapper string to Int64
private class Int64TypeConverter : TypeConverter<string, Int64>
{ protected override Int64 ConvertCore(string source)
{ if (source == null)
throw new MappingException("null string value cannot convert to non-nullable return type.");
else
return Int64.Parse(source);
} }
// Automapper string to DateTime?
// In our case, the datetime will be a JSON2.org datetime
// Example: "/Date(1288296203190)/"
private class NullDateTimeTypeConverter : TypeConverter<string, DateTime?>
{ protected override DateTime? ConvertCore(string source)
{ if (source == null)
return null;
else
{ DateTime result;
return DateTime.TryParse(source, out result) ? (DateTime?) result : null;
} } }
// Automapper string to DateTime
private class DateTimeTypeConverter : TypeConverter<string, DateTime>
{ protected override DateTime ConvertCore(string source)
{ if (source == null)
throw new MappingException("null string value cannot convert to non-nullable return type.");
else
return DateTime.Parse(source);
} }
#endregion
You could create properties in your source class that cast fields to the types as they exist in the destination. Then use AutoMapper in a plain vanilla manner.
public class source
{
public int _myfield;
public string MyField
{
get
{
return _myfield.ToString();
}
}
}
public class destination
{
public string MyField { get; set; }
}