|
| 1 | +// Copyright 2020 ONIXLabs |
| 2 | +// |
| 3 | +// Licensed under the Apache License, Version 2.0 (the "License"); |
| 4 | +// you may not use this file except in compliance with the License. |
| 5 | +// You may obtain a copy of the License at |
| 6 | +// |
| 7 | +// http://www.apache.org/licenses/LICENSE-2.0 |
| 8 | +// |
| 9 | +// Unless required by applicable law or agreed to in writing, software |
| 10 | +// distributed under the License is distributed on an "AS IS" BASIS, |
| 11 | +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 12 | +// See the License for the specific language governing permissions and |
| 13 | +// limitations under the License. |
| 14 | + |
| 15 | +using System; |
| 16 | +using System.Collections.Generic; |
| 17 | +using System.Text; |
| 18 | +using OnixLabs.Core.Text; |
| 19 | + |
| 20 | +namespace OnixLabs.Core.Reflection; |
| 21 | + |
| 22 | +/// <summary> |
| 23 | +/// Represents a formatter for C# type declarations. |
| 24 | +/// <remarks> |
| 25 | +/// There are some limitations as to what this formatter is capable of producing; for example, nullability state information |
| 26 | +/// for nullable reference types, and <see cref="ValueTuple"/> custom names are not available within a <see cref="Type"/> |
| 27 | +/// instance, and therefore cannot be produced by this type declaration formatter. |
| 28 | +/// </remarks> |
| 29 | +/// </summary> |
| 30 | +internal static class CSharpTypeDeclarationFormatter |
| 31 | +{ |
| 32 | + private const char NullableTypeIdentifier = '?'; |
| 33 | + private const char GenericTypeIdentifierMarker = '`'; |
| 34 | + private const char GenericTypeOpenBracket = '<'; |
| 35 | + private const char GenericTypeCloseBracket = '>'; |
| 36 | + private const char ValueTupleOpenParenthesis = '('; |
| 37 | + private const char ValueTupleCloseParenthesis = ')'; |
| 38 | + private const string TypeSeparator = ", "; |
| 39 | + private const string ValueTupleItemName = " Item"; |
| 40 | + private const string TypeNullExceptionMessage = "Type must not be null."; |
| 41 | + |
| 42 | + private static readonly Dictionary<Type, string> TypeAliases = new() |
| 43 | + { |
| 44 | + [typeof(byte)] = "byte", |
| 45 | + [typeof(sbyte)] = "sbyte", |
| 46 | + [typeof(short)] = "short", |
| 47 | + [typeof(ushort)] = "ushort", |
| 48 | + [typeof(int)] = "int", |
| 49 | + [typeof(uint)] = "uint", |
| 50 | + [typeof(long)] = "long", |
| 51 | + [typeof(ulong)] = "ulong", |
| 52 | + [typeof(nint)] = "nint", |
| 53 | + [typeof(nuint)] = "nuint", |
| 54 | + [typeof(float)] = "float", |
| 55 | + [typeof(double)] = "double", |
| 56 | + [typeof(decimal)] = "decimal", |
| 57 | + [typeof(object)] = "object", |
| 58 | + [typeof(bool)] = "bool", |
| 59 | + [typeof(char)] = "char", |
| 60 | + [typeof(string)] = "string", |
| 61 | + [typeof(void)] = "void" |
| 62 | + }; |
| 63 | + |
| 64 | + /// <summary> |
| 65 | + /// Gets the type declaration for the current <see cref="Type"/> instance. |
| 66 | + /// <remarks> |
| 67 | + /// Depending on the specified <see cref="TypeDeclarationFlags"/>, this method is capable or returning type declarations including |
| 68 | + /// simple type names, namespace qualified types names, aliased types names, nullable shorthand notation, generic arguments, and value tuples. |
| 69 | + /// </remarks> |
| 70 | + /// </summary> |
| 71 | + /// <param name="type">The current <see cref="Type"/> instance from which to obtain the type declaration.</param> |
| 72 | + /// <param name="flags">The flags that specify how the type declaration should be formatted.</param> |
| 73 | + /// <returns>Returns the type declaration for the current <see cref="Type"/> instance.</returns> |
| 74 | + public static string GetTypeDeclaration(Type type, TypeDeclarationFlags flags) |
| 75 | + { |
| 76 | + RequireNotNull(type, TypeNullExceptionMessage, nameof(type)); |
| 77 | + |
| 78 | + Type unwrappedType = ConditionallyUnwrapNullableType(type, flags); |
| 79 | + StringBuilder builder = new(); |
| 80 | + |
| 81 | + if (CanFormatValueTupleType(unwrappedType, flags)) |
| 82 | + FormatValueTupleType(unwrappedType, builder, flags); |
| 83 | + |
| 84 | + else if (CanFormatGenericType(unwrappedType, flags)) |
| 85 | + FormatGenericType(unwrappedType, builder, flags); |
| 86 | + |
| 87 | + else FormatTypeName(unwrappedType, builder, flags); |
| 88 | + |
| 89 | + FormatNullableShorthandNotation(type, builder, flags); |
| 90 | + |
| 91 | + return builder.ToString(); |
| 92 | + } |
| 93 | + |
| 94 | + /// <summary> |
| 95 | + /// Conditionally unwraps a <see cref="Nullable{T}"/> type, if the <see cref="TypeDeclarationFlags.UseNullableShorthandTypeNames"/> flag is set. |
| 96 | + /// </summary> |
| 97 | + /// <param name="type">The potential <see cref="Nullable{T}"/> type to unwrap.</param> |
| 98 | + /// <param name="flags">The flags that specify how the type declaration should be formatted.</param> |
| 99 | + /// <returns> |
| 100 | + /// Returns the underlying type, if the specified type is <see cref="Nullable{T}"/>, and the |
| 101 | + /// <see cref="TypeDeclarationFlags.UseNullableShorthandTypeNames"/> flag is set; otherwise, returns the current <see cref="Type"/>. |
| 102 | + /// </returns> |
| 103 | + private static Type ConditionallyUnwrapNullableType(Type type, TypeDeclarationFlags flags) |
| 104 | + { |
| 105 | + if ((flags & TypeDeclarationFlags.UseNullableShorthandTypeNames) is not 0) |
| 106 | + return Nullable.GetUnderlyingType(type) ?? type; |
| 107 | + |
| 108 | + return type; |
| 109 | + } |
| 110 | + |
| 111 | + /// <summary> |
| 112 | + /// Determines whether the specified type is a generic type, and whether the <see cref="TypeDeclarationFlags.UseGenericTypeArguments"/> flag is set. |
| 113 | + /// </summary> |
| 114 | + /// <param name="type">The type to check is potentially generic.</param> |
| 115 | + /// <param name="flags">The flags that specify how the type declaration should be formatted.</param> |
| 116 | + /// <returns>Returns true if the specified type is a generic type, and the <see cref="TypeDeclarationFlags.UseGenericTypeArguments"/> flag is set; otherwise, false.</returns> |
| 117 | + private static bool CanFormatGenericType(Type type, TypeDeclarationFlags flags) |
| 118 | + { |
| 119 | + bool useGenericTypeArguments = (flags & TypeDeclarationFlags.UseGenericTypeArguments) is not 0; |
| 120 | + return useGenericTypeArguments && type.IsGenericType; |
| 121 | + } |
| 122 | + |
| 123 | + /// <summary> |
| 124 | + /// Determines whether the specified type is a multi-argument value-tuple type, and whether the |
| 125 | + /// <see cref="TypeDeclarationFlags.UseValueTupleSyntax"/> or <see cref="TypeDeclarationFlags.UseValueTupleNames"/> flag is set. |
| 126 | + /// </summary> |
| 127 | + /// <param name="type">The type to check is potentially a multi-argument value-tuple type.</param> |
| 128 | + /// <param name="flags">The flags that specify how the type declaration should be formatted.</param> |
| 129 | + /// <returns> |
| 130 | + /// Returns true if the specified type is a multi-argument value-tuple type, and whether the |
| 131 | + /// <see cref="TypeDeclarationFlags.UseValueTupleSyntax"/> or <see cref="TypeDeclarationFlags.UseValueTupleNames"/> flag is set; otherwise, false. |
| 132 | + /// </returns> |
| 133 | + private static bool CanFormatValueTupleType(Type type, TypeDeclarationFlags flags) |
| 134 | + { |
| 135 | + bool useTupleSyntax = (flags & TypeDeclarationFlags.UseValueTupleSyntax) is not 0; |
| 136 | + bool useTupleNames = (flags & TypeDeclarationFlags.UseValueTupleNames) is not 0; |
| 137 | + |
| 138 | + return (useTupleSyntax || useTupleNames) |
| 139 | + && type.Name.StartsWith(nameof(ValueTuple)) |
| 140 | + && type.GenericTypeArguments.Length > 1; |
| 141 | + } |
| 142 | + |
| 143 | + /// <summary> |
| 144 | + /// Formats the specified type as a value-tuple. |
| 145 | + /// </summary> |
| 146 | + /// <param name="type">The type to format.</param> |
| 147 | + /// <param name="builder">The <see cref="StringBuilder"/> to which the type information will be appended.</param> |
| 148 | + /// <param name="flags">The flags that specify how the type declaration should be formatted.</param> |
| 149 | + private static void FormatValueTupleType(Type type, StringBuilder builder, TypeDeclarationFlags flags) |
| 150 | + { |
| 151 | + bool useTupleNames = (flags & TypeDeclarationFlags.UseValueTupleNames) is not 0; |
| 152 | + |
| 153 | + builder.Append(ValueTupleOpenParenthesis); |
| 154 | + FormatTypeArguments(type.GetGenericArguments(), builder, flags, useTupleNames); |
| 155 | + builder.Append(ValueTupleCloseParenthesis); |
| 156 | + } |
| 157 | + |
| 158 | + /// <summary> |
| 159 | + /// Formats the specified type as a generic type. |
| 160 | + /// </summary> |
| 161 | + /// <param name="type">The type to format.</param> |
| 162 | + /// <param name="builder">The <see cref="StringBuilder"/> to which the type information will be appended.</param> |
| 163 | + /// <param name="flags">The flags that specify how the type declaration should be formatted.</param> |
| 164 | + private static void FormatGenericType(Type type, StringBuilder builder, TypeDeclarationFlags flags) |
| 165 | + { |
| 166 | + FormatTypeName(type, builder, flags); |
| 167 | + builder.Append(GenericTypeOpenBracket); |
| 168 | + FormatTypeArguments(type.GetGenericArguments(), builder, flags, useTupleNames: false); |
| 169 | + builder.Append(GenericTypeCloseBracket); |
| 170 | + } |
| 171 | + |
| 172 | + /// <summary> |
| 173 | + /// Formats the specified type as a type alias, namespace qualified name, or simple name. |
| 174 | + /// </summary> |
| 175 | + /// <param name="type">The type to format.</param> |
| 176 | + /// <param name="builder">The <see cref="StringBuilder"/> to which the type information will be appended.</param> |
| 177 | + /// <param name="flags">The flags that specify how the type declaration should be formatted.</param> |
| 178 | + private static void FormatTypeName(Type type, StringBuilder builder, TypeDeclarationFlags flags) |
| 179 | + { |
| 180 | + bool useAliasedTypeNames = (flags & TypeDeclarationFlags.UseAliasedTypeNames) is not 0; |
| 181 | + bool useNamespaceQualifiedTypeNames = (flags & TypeDeclarationFlags.UseNamespaceQualifiedTypeNames) is not 0; |
| 182 | + |
| 183 | + if (useAliasedTypeNames && TypeAliases.TryGetValue(type, out string? alias)) |
| 184 | + { |
| 185 | + builder.Append(alias); |
| 186 | + return; |
| 187 | + } |
| 188 | + |
| 189 | + if (useNamespaceQualifiedTypeNames) |
| 190 | + { |
| 191 | + builder.Append((type.FullName ?? type.Name).SubstringBeforeFirst(GenericTypeIdentifierMarker)); |
| 192 | + return; |
| 193 | + } |
| 194 | + |
| 195 | + builder.Append(type.Name.SubstringBeforeFirst(GenericTypeIdentifierMarker)); |
| 196 | + } |
| 197 | + |
| 198 | + /// <summary> |
| 199 | + /// Formats the specified type's generic argument types. |
| 200 | + /// </summary> |
| 201 | + /// <param name="arguments">The argument types to format.</param> |
| 202 | + /// <param name="builder">The <see cref="StringBuilder"/> to which the type information will be appended.</param> |
| 203 | + /// <param name="flags">The flags that specify how the type declaration should be formatted.</param> |
| 204 | + /// <param name="useTupleNames">Specifies whether tuple names should be formatted.</param> |
| 205 | + private static void FormatTypeArguments(Type[] arguments, StringBuilder builder, TypeDeclarationFlags flags, bool useTupleNames) |
| 206 | + { |
| 207 | + for (int index = 0; index < arguments.Length; index++) |
| 208 | + { |
| 209 | + builder.Append(GetTypeDeclaration(arguments[index], flags)); |
| 210 | + |
| 211 | + if (useTupleNames) |
| 212 | + builder |
| 213 | + .Append(ValueTupleItemName) |
| 214 | + .Append(index + 1); |
| 215 | + |
| 216 | + builder.Append(TypeSeparator); |
| 217 | + } |
| 218 | + |
| 219 | + builder.TrimEnd(TypeSeparator); |
| 220 | + } |
| 221 | + |
| 222 | + /// <summary> |
| 223 | + /// Formats the type using nullable shorthand notation. |
| 224 | + /// </summary> |
| 225 | + /// <param name="type">The type to format.</param> |
| 226 | + /// <param name="builder">The <see cref="StringBuilder"/> to which the type information will be appended.</param> |
| 227 | + /// <param name="flags">The flags that specify how the type declaration should be formatted.</param> |
| 228 | + private static void FormatNullableShorthandNotation(Type type, StringBuilder builder, TypeDeclarationFlags flags) |
| 229 | + { |
| 230 | + if ((flags & TypeDeclarationFlags.UseNullableShorthandTypeNames) is not 0 && Nullable.GetUnderlyingType(type) is not null) |
| 231 | + builder.Append(NullableTypeIdentifier); |
| 232 | + } |
| 233 | +} |
0 commit comments