Xamarin Public Jenkins (auto-signing) e79aa3c0ed Imported Upstream version 4.6.0.125
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
2016-08-03 10:59:49 +00:00

332 lines
14 KiB
C#

//------------------------------------------------------------------------------
// <copyright file="EnumConverter.cs" company="Microsoft">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
//------------------------------------------------------------------------------
/*
*/
namespace System.ComponentModel {
using Microsoft.Win32;
using System.Collections;
using System.Collections.Generic;
using System.ComponentModel.Design.Serialization;
using System.Globalization;
using System.Diagnostics;
using System.Reflection;
using System.Runtime.Serialization.Formatters;
using System.Runtime.Remoting;
using System.Runtime.InteropServices;
using System.Security.Permissions;
/// <devdoc>
/// <para>Provides a type converter to convert <see cref='System.Enum'/>
/// objects to and from various
/// other representations.</para>
/// </devdoc>
[HostProtection(SharedState = true)]
public class EnumConverter : TypeConverter {
/// <devdoc>
/// <para>
/// Provides a <see cref='System.ComponentModel.TypeConverter.StandardValuesCollection'/> that specifies the
/// possible values for the enumeration.
/// </para>
/// </devdoc>
private StandardValuesCollection values;
/// <devdoc>
/// <para>
/// Specifies
/// the
/// type of the enumerator this converter is
/// associated with.
/// </para>
/// </devdoc>
private Type type;
/// <devdoc>
/// <para>
/// Initializes a new instance of the <see cref='System.ComponentModel.EnumConverter'/> class for the given
/// type.
/// </para>
/// </devdoc>
public EnumConverter(Type type) {
this.type = type;
}
/// <devdoc>
/// <para>[To be supplied.]</para>
/// </devdoc>
protected Type EnumType {
get {
return type;
}
}
/// <devdoc>
/// <para>[To be supplied.]</para>
/// </devdoc>
protected StandardValuesCollection Values {
get {
return values;
}
set {
values = value;
}
}
/// <internalonly/>
/// <devdoc>
/// <para>Gets a value indicating whether this converter
/// can convert an object in the given source type to an enumeration object using
/// the specified context.</para>
/// </devdoc>
public override bool CanConvertFrom(ITypeDescriptorContext context, Type sourceType) {
if (sourceType == typeof(string) || sourceType == typeof(Enum[])) {
return true;
}
return base.CanConvertFrom(context, sourceType);
}
/// <devdoc>
/// <para>Gets a value indicating whether this converter can
/// convert an object to the given destination type using the context.</para>
/// </devdoc>
public override bool CanConvertTo(ITypeDescriptorContext context, Type destinationType) {
if (destinationType == typeof(InstanceDescriptor) || destinationType == typeof(Enum[])) {
return true;
}
return base.CanConvertTo(context, destinationType);
}
/// <devdoc>
/// <para>Gets an <see cref='System.Collections.IComparer'/>
/// interface that can
/// be used to sort the values of the enumerator.</para>
/// </devdoc>
protected virtual IComparer Comparer {
get {
return InvariantComparer.Default;
}
}
/// <internalonly/>
/// <devdoc>
/// <para>Converts the specified value object to an enumeration object.</para>
/// </devdoc>
public override object ConvertFrom(ITypeDescriptorContext context, CultureInfo culture, object value) {
if (value is string) {
try {
string strValue = (string)value;
if (strValue.IndexOf(',') != -1) {
long convertedValue = 0;
string[] values = strValue.Split(new char[] {','});
foreach(string v in values) {
convertedValue |= Convert.ToInt64((Enum)Enum.Parse(type, v, true), culture);
}
return Enum.ToObject(type, convertedValue);
}
else {
return Enum.Parse(type, strValue, true);
}
}
catch (Exception e) {
throw new FormatException(SR.GetString(SR.ConvertInvalidPrimitive, (string)value, type.Name), e);
}
}
else if (value is Enum[]) {
long finalValue = 0;
foreach(Enum e in (Enum[])value) {
finalValue |= Convert.ToInt64(e, culture);
}
return Enum.ToObject(type, finalValue);
}
return base.ConvertFrom(context, culture, value);
}
/// <internalonly/>
/// <devdoc>
/// <para>Converts the given
/// value object to the
/// specified destination type.</para>
/// </devdoc>
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Performance", "CA1803:AvoidCostlyCallsWherePossible")] // Keep call to Enum.IsDefined
public override object ConvertTo(ITypeDescriptorContext context, CultureInfo culture, object value, Type destinationType) {
if (destinationType == null) {
throw new ArgumentNullException("destinationType");
}
if (destinationType == typeof(string) && value != null) {
// Raise an argument exception if the value isn't defined and if
// the enum isn't a flags style.
//
Type underlyingType = Enum.GetUnderlyingType(type);
if (value is IConvertible && value.GetType() != underlyingType) {
value = ((IConvertible)value).ToType(underlyingType, culture);
}
if (!type.IsDefined(typeof(FlagsAttribute), false) && !Enum.IsDefined(type, value)) {
throw new ArgumentException(SR.GetString(SR.EnumConverterInvalidValue, value.ToString(), type.Name));
}
return Enum.Format(type, value, "G");
}
if (destinationType == typeof(InstanceDescriptor) && value != null) {
string enumName = ConvertToInvariantString(context, value);
if (type.IsDefined(typeof(FlagsAttribute), false) && enumName.IndexOf(',') != -1) {
// This is a flags enum, and there is no one flag
// that covers the value. Instead, convert the
// value to the underlying type and invoke
// a ToObject call on enum.
//
Type underlyingType = Enum.GetUnderlyingType(type);
if (value is IConvertible) {
object convertedValue = ((IConvertible)value).ToType(underlyingType, culture);
MethodInfo method = typeof(Enum).GetMethod("ToObject", new Type[] {typeof(Type), underlyingType});
if (method != null) {
return new InstanceDescriptor(method, new object[] {type, convertedValue});
}
}
}
else {
FieldInfo info = type.GetField(enumName);
if (info != null) {
return new InstanceDescriptor(info, null);
}
}
}
if (destinationType == typeof(Enum[]) && value != null) {
if (type.IsDefined(typeof(FlagsAttribute), false)) {
List<Enum> flagValues = new List<Enum>();
Array objValues = Enum.GetValues(type);
long[] ulValues = new long[objValues.Length];
for(int idx = 0; idx < objValues.Length; idx++) {
ulValues[idx] = Convert.ToInt64((Enum)objValues.GetValue(idx), culture);
}
long longValue = Convert.ToInt64((Enum)value, culture);
bool valueFound = true;
while(valueFound) {
valueFound = false;
foreach(long ul in ulValues) {
if ((ul != 0 && (ul & longValue) == ul) || ul == longValue) {
flagValues.Add((Enum)Enum.ToObject(type, ul));
valueFound = true;
longValue &= ~ul;
break;
}
}
if (longValue == 0) {
break;
}
}
if (!valueFound && longValue != 0) {
flagValues.Add((Enum)Enum.ToObject(type, longValue));
}
return flagValues.ToArray();
}
else {
return new Enum[] {(Enum)Enum.ToObject(type, value)};
}
}
return base.ConvertTo(context, culture, value, destinationType);
}
/// <internalonly/>
/// <devdoc>
/// <para>Gets a collection of standard values for the data type this validator is
/// designed for.</para>
/// </devdoc>
public override StandardValuesCollection GetStandardValues(ITypeDescriptorContext context) {
if (values == null) {
// We need to get the enum values in this rather round-about way so we can filter
// out fields marked Browsable(false). Note that if multiple fields have the same value,
// the behavior is undefined, since what we return are just enum values, not names.
Type reflectType = TypeDescriptor.GetReflectionType(type);
if (reflectType == null) {
reflectType = type;
}
FieldInfo[] fields = reflectType.GetFields(BindingFlags.Public | BindingFlags.Static);
ArrayList objValues = null;
if (fields != null && fields.Length > 0) {
objValues = new ArrayList(fields.Length);
}
if (objValues != null) {
foreach (FieldInfo field in fields) {
BrowsableAttribute browsableAttr = null;
foreach (Attribute attr in field.GetCustomAttributes(typeof(BrowsableAttribute), false)) {
browsableAttr = attr as BrowsableAttribute;
}
if (browsableAttr == null || browsableAttr.Browsable) {
object value = null;
try {
if (field.Name != null) {
value = Enum.Parse(type, field.Name);
}
}
catch (ArgumentException) {
// Hmm, for some reason, the parse threw. Let us ignore this value.
}
if (value != null) {
objValues.Add(value);
}
}
}
IComparer comparer = Comparer;
if (comparer != null) {
objValues.Sort(comparer);
}
}
Array arr = (objValues != null) ? objValues.ToArray() : null;
values = new StandardValuesCollection(arr);
}
return values;
}
/// <internalonly/>
/// <devdoc>
/// <para>Gets a value indicating whether the list of standard values returned from
/// <see cref='System.ComponentModel.TypeConverter.GetStandardValues'/>
/// is an exclusive list using the specified context.</para>
/// </devdoc>
public override bool GetStandardValuesExclusive(ITypeDescriptorContext context) {
return !type.IsDefined(typeof(FlagsAttribute), false);
}
/// <internalonly/>
/// <devdoc>
/// <para>Gets a value indicating
/// whether this object
/// supports a standard set of values that can be picked
/// from a list using the specified context.</para>
/// </devdoc>
public override bool GetStandardValuesSupported(ITypeDescriptorContext context) {
return true;
}
/// <internalonly/>
/// <devdoc>
/// <para>Gets a value indicating whether the given object value is valid for this type.</para>
/// </devdoc>
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Performance", "CA1803:AvoidCostlyCallsWherePossible")] // Keep call to Enum.IsDefined
public override bool IsValid(ITypeDescriptorContext context, object value) {
return Enum.IsDefined(type, value);
}
}
}