mirror of
https://gitee.com/ant-design-blazor/ant-design-blazor.git
synced 2024-12-12 11:55:24 +08:00
112 lines
4.3 KiB
C#
112 lines
4.3 KiB
C#
using System;
|
|
using System.Collections.Concurrent;
|
|
using System.Globalization;
|
|
|
|
namespace AntDesign.Docs.Routing
|
|
{
|
|
internal abstract class RouteConstraint
|
|
{
|
|
// note: the things that prevent this cache from growing unbounded is that
|
|
// we're the only caller to this code path, and the fact that there are only
|
|
// 8 possible instances that we create.
|
|
//
|
|
// The values passed in here for parsing are always static text defined in route attributes.
|
|
private static readonly ConcurrentDictionary<string, RouteConstraint> _cachedConstraints
|
|
= new ConcurrentDictionary<string, RouteConstraint>();
|
|
|
|
public abstract bool Match(string pathSegment, out object convertedValue);
|
|
|
|
public static RouteConstraint Parse(string template, string segment, string constraint)
|
|
{
|
|
if (string.IsNullOrEmpty(constraint))
|
|
{
|
|
throw new ArgumentException($"Malformed segment '{segment}' in route '{template}' contains an empty constraint.");
|
|
}
|
|
|
|
if (_cachedConstraints.TryGetValue(constraint, out var cachedInstance))
|
|
{
|
|
return cachedInstance;
|
|
}
|
|
else
|
|
{
|
|
var newInstance = CreateRouteConstraint(constraint);
|
|
if (newInstance != null)
|
|
{
|
|
// We've done to the work to create the constraint now, but it's possible
|
|
// we're competing with another thread. GetOrAdd can ensure only a single
|
|
// instance is returned so that any extra ones can be GC'ed.
|
|
return _cachedConstraints.GetOrAdd(constraint, newInstance);
|
|
}
|
|
else
|
|
{
|
|
throw new ArgumentException($"Unsupported constraint '{constraint}' in route '{template}'.");
|
|
}
|
|
}
|
|
}
|
|
|
|
private static RouteConstraint CreateRouteConstraint(string constraint)
|
|
{
|
|
switch (constraint)
|
|
{
|
|
case "bool":
|
|
return new TypeRouteConstraint<bool>(bool.TryParse);
|
|
|
|
case "datetime":
|
|
return new TypeRouteConstraint<DateTime>((string str, out DateTime result)
|
|
=> DateTime.TryParse(str, CultureInfo.InvariantCulture, DateTimeStyles.None, out result));
|
|
|
|
case "decimal":
|
|
return new TypeRouteConstraint<decimal>((string str, out decimal result)
|
|
=> decimal.TryParse(str, NumberStyles.Number, CultureInfo.InvariantCulture, out result));
|
|
|
|
case "double":
|
|
return new TypeRouteConstraint<double>((string str, out double result)
|
|
=> double.TryParse(str, NumberStyles.Number, CultureInfo.InvariantCulture, out result));
|
|
|
|
case "float":
|
|
return new TypeRouteConstraint<float>((string str, out float result)
|
|
=> float.TryParse(str, NumberStyles.Number, CultureInfo.InvariantCulture, out result));
|
|
|
|
case "guid":
|
|
return new TypeRouteConstraint<Guid>(Guid.TryParse);
|
|
|
|
case "int":
|
|
return new TypeRouteConstraint<int>((string str, out int result)
|
|
=> int.TryParse(str, NumberStyles.Integer, CultureInfo.InvariantCulture, out result));
|
|
|
|
case "long":
|
|
return new TypeRouteConstraint<long>((string str, out long result)
|
|
=> long.TryParse(str, NumberStyles.Integer, CultureInfo.InvariantCulture, out result));
|
|
|
|
default:
|
|
return null;
|
|
}
|
|
}
|
|
}
|
|
|
|
internal class TypeRouteConstraint<T> : RouteConstraint
|
|
{
|
|
public delegate bool TryParseDelegate(string str, out T result);
|
|
|
|
private readonly TryParseDelegate _parser;
|
|
|
|
public TypeRouteConstraint(TryParseDelegate parser)
|
|
{
|
|
_parser = parser;
|
|
}
|
|
|
|
public override bool Match(string pathSegment, out object convertedValue)
|
|
{
|
|
if (_parser(pathSegment, out var result))
|
|
{
|
|
convertedValue = result;
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
convertedValue = null;
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
} |