|
6 | 6 |
|
7 | 7 | namespace Limekuma.Utils; |
8 | 8 |
|
| 9 | +internal enum UnionValueState |
| 10 | +{ |
| 11 | + Empty, |
| 12 | + First, |
| 13 | + Second |
| 14 | +} |
| 15 | + |
9 | 16 | [JsonConverter(typeof(UnionJsonConverter))] |
10 | | -public class Union<TA, TB> |
| 17 | +public sealed class Union<T1, T2> |
11 | 18 | { |
12 | | - private readonly TA? _valueA; |
13 | | - private readonly TB? _valueB; |
| 19 | + private readonly T1? _value1; |
| 20 | + private readonly T2? _value2; |
| 21 | + private readonly UnionValueState _valueState = UnionValueState.Empty; |
14 | 22 |
|
15 | | - public Union(TA? value) => _valueA = value; |
16 | | - public Union(TB? value) => _valueB = value; |
| 23 | + public Union(T1? value) : this() |
| 24 | + { |
| 25 | + if (value is null) |
| 26 | + { |
| 27 | + return; |
| 28 | + } |
| 29 | + |
| 30 | + _valueState = UnionValueState.First; |
| 31 | + _value1 = value; |
| 32 | + } |
17 | 33 |
|
18 | | - public Union(object? obj) |
| 34 | + public Union(T2? value) : this() |
19 | 35 | { |
20 | | - if (obj is null) |
| 36 | + if (value is null) |
21 | 37 | { |
22 | 38 | return; |
23 | 39 | } |
24 | 40 |
|
25 | | - if (obj is TA valueA) |
| 41 | + _valueState = UnionValueState.Second; |
| 42 | + _value2 = value; |
| 43 | + } |
| 44 | + |
| 45 | + public Union(object? value) : this() |
| 46 | + { |
| 47 | + switch (value) |
| 48 | + { |
| 49 | + case null: |
| 50 | + return; |
| 51 | + case T1 value1: |
| 52 | + _valueState = UnionValueState.First; |
| 53 | + _value1 = value1; |
| 54 | + break; |
| 55 | + case T2 value2: |
| 56 | + _valueState = UnionValueState.Second; |
| 57 | + _value2 = value2; |
| 58 | + break; |
| 59 | + default: |
| 60 | + throw new ArgumentException($"Union value must be of type {typeof(T1).Name} or {typeof(T2).Name}.", nameof(value)); |
| 61 | + } |
| 62 | + } |
| 63 | + |
| 64 | + private Union() |
| 65 | + { |
| 66 | + if (typeof(T1).IsAssignableFrom(typeof(T2))) |
26 | 67 | { |
27 | | - _valueA = valueA; |
| 68 | + throw new ArgumentException($"Union value type {typeof(T1).Name} is assignable from type {typeof(T2).Name}."); |
28 | 69 | } |
29 | 70 |
|
30 | | - if (obj is TB valueB) |
| 71 | + if (typeof(T2).IsAssignableFrom(typeof(T1))) |
31 | 72 | { |
32 | | - _valueB = valueB; |
| 73 | + throw new ArgumentException($"Union value type {typeof(T2).Name} is assignable from type {typeof(T1).Name}."); |
33 | 74 | } |
34 | 75 | } |
35 | 76 |
|
36 | 77 | public object? Value |
37 | 78 | { |
38 | 79 | get |
39 | 80 | { |
40 | | - if (_valueA is not null) |
| 81 | + return _valueState switch |
41 | 82 | { |
42 | | - return _valueA; |
43 | | - } |
44 | | - |
45 | | - if (_valueB is not null) |
46 | | - { |
47 | | - return _valueB; |
48 | | - } |
49 | | - |
50 | | - return null; |
| 83 | + UnionValueState.Empty => null, |
| 84 | + UnionValueState.First => _value1, |
| 85 | + UnionValueState.Second => _value2, |
| 86 | + _ => throw new InvalidOperationException($"Union value type {_valueState} is not valid."), |
| 87 | + }; |
51 | 88 | } |
52 | 89 | } |
53 | 90 |
|
54 | | - public static implicit operator TA?(Union<TA, TB> o) => o._valueA; |
| 91 | + public static implicit operator T1?(Union<T1, T2> o) => o._value1; |
55 | 92 |
|
56 | | - public static implicit operator TB?(Union<TA, TB> o) => o._valueB; |
| 93 | + public static implicit operator T2?(Union<T1, T2> o) => o._value2; |
57 | 94 |
|
58 | | - public static implicit operator Union<TA, TB>(TA a) => new(a); |
| 95 | + public static implicit operator Union<T1, T2>(T1 a) => new(a); |
59 | 96 |
|
60 | | - public static implicit operator Union<TA, TB>(TB b) => new(b); |
| 97 | + public static implicit operator Union<T1, T2>(T2 b) => new(b); |
61 | 98 |
|
62 | | - public new Type? GetType() => Value?.GetType(); |
| 99 | + public Type? ValueType => Value?.GetType(); |
63 | 100 |
|
64 | | - public new string? ToString() => Value?.ToString(); |
| 101 | + public override string? ToString() => Value?.ToString(); |
65 | 102 |
|
66 | | - public new bool? Equals(object? obj) => Value?.Equals(obj); |
| 103 | + public override bool Equals(object? obj) |
| 104 | + { |
| 105 | + if (obj is Union<T1, T2> value) |
| 106 | + { |
| 107 | + if (value._valueState != _valueState) |
| 108 | + { |
| 109 | + return false; |
| 110 | + } |
| 111 | + |
| 112 | + return Value?.Equals(value.Value) ?? value.Value is null; |
| 113 | + } |
| 114 | + |
| 115 | + return Value?.Equals(obj) ?? obj is null; |
| 116 | + } |
67 | 117 |
|
68 | | - public new int? GetHashCode() => Value?.GetHashCode(); |
| 118 | + public override int GetHashCode() => Value?.GetHashCode() ?? 0; |
69 | 119 | } |
70 | 120 |
|
71 | 121 | public sealed class UnionJsonConverter : JsonConverterFactory |
|
0 commit comments