246 lines
7.4 KiB
C#
Raw Normal View History

//
// System.ComponentModel.EnumConverter.cs
//
// Authors:
// Gonzalo Paniagua Javier (gonzalo@ximian.com)
// Andreas Nahr (ClassDevelopment@A-SoftTech.com)
// Ivan N. Zlatev (contact@i-nz.net)
//
// (C) 2002 Ximian, Inc (http://www.ximian.com)
// (C) 2003 Andreas Nahr
// (C) 2007 Ivan N. Zlatev
//
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//
using System;
using System.Collections;
using System.Globalization;
using System.Reflection;
using System.ComponentModel.Design.Serialization;
namespace System.ComponentModel
{
public class EnumConverter : TypeConverter
{
private Type type;
private StandardValuesCollection stdValues;
public EnumConverter (Type type)
{
this.type = type;
}
public override bool CanConvertTo (ITypeDescriptorContext context, Type destinationType)
{
if (destinationType == typeof (InstanceDescriptor))
return true;
if (destinationType == typeof (Enum[]))
return true;
return base.CanConvertTo (context, destinationType);
}
public override object ConvertTo (ITypeDescriptorContext context,
CultureInfo culture,
object value,
Type destinationType)
{
if (destinationType == typeof (string) && value != null) {
// we need to be able to convert enum names,
// integral values and other enum fields
if (value is IConvertible) {
Type underlyingType = Enum.GetUnderlyingType (type);
if (underlyingType != value.GetType ()) {
value = ((IConvertible) value).ToType (
underlyingType, culture);
}
}
if (!IsFlags && !IsValid (context, value))
throw CreateValueNotValidException (value);
return Enum.Format (type, value, "G");
} else if (destinationType == typeof (InstanceDescriptor) && value != null) {
string enumString = ConvertToString (context, culture, value);
if (IsFlags && enumString.IndexOf (",") != -1) {
if (value is IConvertible) {
Type underlyingType = Enum.GetUnderlyingType (type);
object enumValue = ((IConvertible)value).ToType (underlyingType, culture);
MethodInfo toObjectMethod = typeof (Enum).GetMethod ("ToObject", new Type[] { typeof (Type),
underlyingType });
return new InstanceDescriptor (toObjectMethod, new object[] { type, enumValue });
}
} else {
FieldInfo f = type.GetField (enumString);
if (f != null)
return new InstanceDescriptor (f, null);
}
} else if (destinationType == typeof (Enum[]) && value != null) {
if (!IsFlags) {
return new Enum[] { (Enum) Enum.ToObject (type, value) };
} else {
long valueLong = Convert.ToInt64 (
(Enum) value, culture);
Array enumValArray = Enum.GetValues (type);
long[] enumValues = new long[enumValArray.Length];
for (int i=0; i < enumValArray.Length; i++)
enumValues[i] = Convert.ToInt64 (enumValArray.GetValue (i));
ArrayList enums = new ArrayList ();
bool interrupt = false;
while (!interrupt) {
// interrupt the loop unless we find a match to avoid
// looping indefinitely
interrupt = true;
foreach (long val in enumValues) {
if ((val != 0 && ((val & valueLong) == val)) || val == valueLong) {
enums.Add (Enum.ToObject (type, val));
valueLong &= (~val);
interrupt = false;
}
}
if (valueLong == 0) // nothing left to do
interrupt = true;
}
// add item for remainder
if (valueLong != 0)
enums.Add (Enum.ToObject (type, valueLong));
return enums.ToArray (typeof(Enum));
}
}
return base.ConvertTo (context, culture, value, destinationType);
}
public override bool CanConvertFrom (ITypeDescriptorContext context, Type sourceType)
{
if (sourceType == typeof (string))
return true;
if (sourceType == typeof (Enum[]))
return true;
return base.CanConvertFrom (context, sourceType);
}
public override object ConvertFrom (ITypeDescriptorContext context,
CultureInfo culture,
object value)
{
if (value is string) {
string name = value as string;
try {
if (name.IndexOf (',') == -1)
return Enum.Parse (type, name, true);
long val = 0;
string [] names = name.Split (',');
foreach (string n in names) {
Enum e = (Enum) Enum.Parse (type, n, true);
val |= Convert.ToInt64 (e, culture);
}
return Enum.ToObject (type, val);
} catch (Exception ex) {
throw new FormatException (name + " is " +
"not a valid value for " +
type.Name, ex);
}
} else if (value is Enum[]) {
long val = 0;
foreach (Enum e in (Enum[])value)
val |= Convert.ToInt64 (e, culture);
return Enum.ToObject (type, val);
}
return base.ConvertFrom (context, culture, value);
}
public override bool IsValid (ITypeDescriptorContext context, object value)
{
return Enum.IsDefined (type, value);
}
public override bool GetStandardValuesSupported (ITypeDescriptorContext context)
{
return true;
}
public override bool GetStandardValuesExclusive (ITypeDescriptorContext context)
{
return !IsFlags;
}
public override StandardValuesCollection GetStandardValues (ITypeDescriptorContext context)
{
if (stdValues == null) {
Array values = Enum.GetValues (type);
Array.Sort (values);
stdValues = new StandardValuesCollection (values);
}
return stdValues;
}
protected virtual IComparer Comparer {
get { return new EnumConverter.EnumComparer (); }
}
protected Type EnumType {
get { return type; }
}
protected TypeConverter.StandardValuesCollection Values {
get { return stdValues; }
set { stdValues = value; }
}
ArgumentException CreateValueNotValidException (object value)
{
string msg = string.Format (CultureInfo.InvariantCulture,
"The value '{0}' is not a valid value for the " +
"enum '{1}'", value, type.Name);
return new ArgumentException (msg);
}
bool IsFlags {
get {
return (type.IsDefined (typeof (FlagsAttribute),
false));
}
}
private class EnumComparer : IComparer
{
int IComparer.Compare (object compareObject1, object compareObject2)
{
string CompareString1 = (compareObject1 as string);
string CompareString2 = (compareObject2 as string);
if ((CompareString1 == null) || (CompareString2 == null))
return Collections.Comparer.Default.Compare (compareObject1, compareObject2);
return CultureInfo.InvariantCulture.CompareInfo.Compare (CompareString1, CompareString2);
}
}
}
}