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);
//}
}
}