File: JSComponents\JSComponentConfigurationStore.cs
Web Access
Project: src\src\Components\Web\src\Microsoft.AspNetCore.Components.Web.csproj (Microsoft.AspNetCore.Components.Web)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System.Diagnostics.CodeAnalysis;
using Microsoft.AspNetCore.Components.RenderTree;
using Microsoft.AspNetCore.Components.Web.Infrastructure;
using Microsoft.AspNetCore.Internal;
 
namespace Microsoft.AspNetCore.Components.Web;
 
/// <summary>
/// Specifies options for use when enabling JS component support.
/// This type is not normally used directly from application code. In most cases, applications should
/// call methods on the <see cref="IJSComponentConfiguration" /> on their application host builder.
/// </summary>
public sealed class JSComponentConfigurationStore
{
    // Everything's internal here, and can only be operated upon via the extension methods on
    // IJSComponentConfiguration. This is so that, in the future, we can add any additional
    // configuration APIs (as further extension methods) and/or storage (as internal members here)
    // without needing any changes on the downstream code that implements IJSComponentConfiguration,
    // and without exposing any of the configuration storage across layers.
 
    private readonly Dictionary<string, Type> _jsComponentTypesByIdentifier = new(StringComparer.Ordinal);
    internal Dictionary<string, JSComponentParameter[]> JSComponentParametersByIdentifier { get; } = new(StringComparer.Ordinal);
    internal Dictionary<string, List<string>> JSComponentIdentifiersByInitializer { get; } = new(StringComparer.Ordinal);
 
    internal void Add([DynamicallyAccessedMembers(LinkerFlags.Component)] Type componentType, string identifier)
    {
        var parameterTypes = JSComponentInterop.GetComponentParameters(componentType).ParameterInfoByName;
        var parameters = new JSComponentParameter[parameterTypes.Count];
        var index = 0;
        foreach (var (name, type) in parameterTypes)
        {
            parameters[index++] = new JSComponentParameter(name, type.Type);
        }
 
        _jsComponentTypesByIdentifier.Add(identifier, componentType);
        JSComponentParametersByIdentifier.Add(identifier, parameters);
    }
 
    [UnconditionalSuppressMessage("Trimming", "IL2067",
        Justification = "Types added to dictionary are always correctly annotated.")]
    internal bool TryGetComponentType(
        string identifier,
        [NotNullWhen(true)][DynamicallyAccessedMembers(LinkerFlags.Component)] out Type? componentType)
    {
        return _jsComponentTypesByIdentifier.TryGetValue(identifier, out componentType);
    }
 
    [DynamicDependency(DynamicallyAccessedMemberTypes.PublicProperties, typeof(JSComponentParameter))]
    [DynamicDependency(nameof(WebRenderer.WebRendererInteropMethods.AddRootComponent), typeof(WebRenderer.WebRendererInteropMethods))]
    [DynamicDependency(nameof(WebRenderer.WebRendererInteropMethods.SetRootComponentParameters), typeof(WebRenderer.WebRendererInteropMethods))]
    [DynamicDependency(nameof(WebRenderer.WebRendererInteropMethods.RemoveRootComponent), typeof(WebRenderer.WebRendererInteropMethods))]
    internal void Add([DynamicallyAccessedMembers(LinkerFlags.Component)] Type componentType, string identifier, string javaScriptInitializer)
    {
        Add(componentType, identifier);
 
        ArgumentException.ThrowIfNullOrEmpty(javaScriptInitializer);
 
        // Since it has a JS initializer, prepare the metadata we'll supply to JS code
        if (!JSComponentIdentifiersByInitializer.TryGetValue(javaScriptInitializer, out var identifiersForInitializer))
        {
            identifiersForInitializer = new();
            JSComponentIdentifiersByInitializer.Add(javaScriptInitializer, identifiersForInitializer);
        }
 
        identifiersForInitializer.Add(identifier);
    }
 
    internal readonly struct JSComponentParameter
    {
        public readonly string Name { get; }
        public readonly string Type { get; }
 
        public JSComponentParameter(string name, Type dotNetType)
        {
            Name = name;
            Type = GetJSType(dotNetType);
        }
 
        private static string GetJSType(Type dotNetType) => dotNetType switch
        {
            var x when x == typeof(string) => "string",
            var x when x == typeof(bool) => "boolean",
            var x when x == typeof(bool?) => "boolean?",
            var x when x == typeof(decimal) => "number",
            var x when x == typeof(decimal?) => "number?",
            var x when x == typeof(double) => "number",
            var x when x == typeof(double?) => "number?",
            var x when x == typeof(float) => "number",
            var x when x == typeof(float?) => "number?",
            var x when x == typeof(int) => "number",
            var x when x == typeof(int?) => "number?",
            var x when x == typeof(long) => "number",
            var x when x == typeof(long?) => "number?",
            var x when JSComponentInterop.IsEventCallbackType(x) => "eventcallback",
            _ => "object"
        };
    }
}