using System; using System.Collections.Generic; using System.Globalization; using System.Linq; namespace NTwain.Data { /// /// Contains extension methods for converting (possibly bad) integer values to enum values. /// public static class ValueExtensions { /// /// Casts a list of objects to a list of specified enum. /// /// The type of the enum. /// The list. /// public static IList CastToEnum(this IEnumerable list) where TEnum : struct, IConvertible { return list.CastToEnum(true); } /// /// Casts a list of objects to a list of specified enum. /// /// The type of the enum. /// The list. /// set to true for working with bad values. /// public static IList CastToEnum(this IEnumerable list, bool tryUpperWord) where TEnum : struct, IConvertible { return list.Select(o => o.ConvertToEnum(tryUpperWord)).ToList(); } /// /// Casts an objects to the specified enum. /// /// The type of the enum. /// The value. /// public static TEnum ConvertToEnum(this object value) where TEnum : struct, IConvertible { return ConvertToEnum(value, true); } /// /// Casts an objects to the specified enum. /// /// The type of the enum. /// The value. /// if set to true [try upper word]. /// public static TEnum ConvertToEnum(this object value, bool tryUpperWord) where TEnum : struct, IConvertible { if (value != null) { var returnType = typeof(TEnum); // standard int values if (returnType.IsEnum) { var rawType = Enum.GetUnderlyingType(returnType); if (tryUpperWord) { // small routine to work with bad sources that may put // 16bit value in the upper word instead of lower word (as per the twain spec). if (typeof(ushort).IsAssignableFrom(rawType)) { var intVal = Convert.ToUInt32(value, CultureInfo.InvariantCulture); var enumVal = GetLowerWord(intVal); if (!Enum.IsDefined(returnType, enumVal)) { return (TEnum)Enum.ToObject(returnType, GetUpperWord(intVal)); } } } // old method: // return (TEnum)Enum.ToObject(returnType, value); // new method: // try to convert to enum's underlying type first then cast to the enum return (TEnum)Convert.ChangeType(value, rawType, CultureInfo.InvariantCulture); } else if (typeof(IConvertible).IsAssignableFrom(returnType)) { // for regular integers and whatnot return (TEnum)Convert.ChangeType(value, returnType, CultureInfo.InvariantCulture); } // return as-is from cap. if caller made a mistake then there should be exceptions return (TEnum)value; } return default(TEnum); } static ushort GetLowerWord(uint value) { return (ushort)(value & 0xffff); } static uint GetUpperWord(uint value) { return (ushort)(value >> 16); } /// /// Tries to convert a value to if possible. /// /// The value. /// public static TWFix32 ConvertToFix32(this object value) { if (value != null) { if (value is TWFix32) { return (TWFix32)value; } return (TWFix32)Convert.ToSingle(value, CultureInfo.InvariantCulture); } return default(TWFix32); } /// /// Tries to cast a value to if possible. /// /// The value. /// public static TWFrame ConvertToFrame(this object value) { if (value != null) { if (value is TWFrame) { return (TWFrame)value; } } return default(TWFrame); } /// /// Converts object to string. /// /// The value. /// public static string ConvertToString(this object value) { if (value == null) { return null; } return value.ToString(); } ///// ///// Routine that does nothing. ///// ///// The value. ///// //public static object NoConvertRoutine(object value) //{ // return value; //} ///// ///// Predefined routine for ///// ///// The value. ///// //public static string ConvertToString(object value) //{ // if (value != null) // { // return value.ToString(); // } // return default(string); //} } }