1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
|
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Google.Protobuf
{
public class EnumHelper<T> where T : struct, IComparable, IFormattable
{
// TODO(jonskeet): For snmall enums, use something lighter-weight than a dictionary?
private static readonly Dictionary<int, T> _byNumber;
private static readonly Dictionary<string, T> _byName;
private const long UnknownValueBase = 0x100000000L;
static EnumHelper()
{
if (!typeof(T).IsEnum)
{
throw new InvalidOperationException(string.Format("{0} is not an enum type", typeof(T).FullName));
}
if (Enum.GetUnderlyingType(typeof(T)) != typeof(long))
{
throw new InvalidOperationException(string.Format("{0} does not have long as an underlying type", typeof(T).FullName));
}
// It will actually be a T[], but the CLR will let us convert.
long[] array = (long[])Enum.GetValues(typeof(T));
_byNumber = new Dictionary<int, T>(array.Length);
foreach (long i in array)
{
_byNumber[(int) i] = (T)(object)i;
}
string[] names = (string[])Enum.GetNames(typeof(T));
_byName = new Dictionary<string, T>();
foreach (var name in names)
{
_byName[name] = (T) Enum.Parse(typeof(T), name, false);
}
}
/// <summary>
/// Converts the given 32-bit integer into a value of the enum type.
/// If the integer is a known, named value, that is returned; otherwise,
/// a value which is outside the range of 32-bit integers but which preserves
/// the original integer is returned.
/// </summary>
public static T FromInt32(int value)
{
T validValue;
return _byNumber.TryGetValue(value, out validValue) ? validValue : FromRawValue((long) value | UnknownValueBase);
}
/// <summary>
/// Converts a value of the enum type to a 32-bit integer. This reverses
/// <see cref="Int32"/>.
/// </summary>
public static int ToInt32(T value)
{
return (int)GetRawValue(value);
}
public static bool IsKnownValue(T value)
{
long raw = GetRawValue(value);
if (raw >= int.MinValue && raw <= int.MaxValue)
{
return _byNumber.ContainsKey((int)raw);
}
return false;
}
private static long GetRawValue(T value)
{
// TODO(jonskeet): Try using expression trees to get rid of the boxing here.
return (long)(object)value;
}
private static T FromRawValue(long value)
{
// TODO(jonskeet): Try using expression trees to get rid of the boxing here.
return (T)(object)value;
}
}
}
|