Files
QWERTYkez.Mensura/QWERTYkez.Mensura.Generator/UnitGenerator.cs

800 lines
55 KiB
C#
Raw Normal View History

2026-06-09 16:45:22 +07:00
namespace G;
2026-05-29 01:33:38 +07:00
[Generator]
2026-06-05 12:13:35 +07:00
public class UnitGenerator : IIncrementalGenerator
{
2026-06-05 12:13:35 +07:00
private const string AttributeName = "UnitGenerator";
2026-05-29 01:33:38 +07:00
public void Initialize(IncrementalGeneratorInitializationContext context)
{
2026-05-29 01:33:38 +07:00
// Генерируем атрибут
context.RegisterPostInitializationOutput(ctx =>
{
2026-05-29 01:33:38 +07:00
string attributeSource = @"
namespace QWERTYkez.Mensura
{
2026-05-29 01:33:38 +07:00
[System.AttributeUsage(System.AttributeTargets.Struct, AllowMultiple = false)]
2026-06-09 16:45:22 +07:00
internal sealed class UnitGeneratorAttribute : System.Attribute { }
2026-05-29 01:33:38 +07:00
}";
2026-06-05 12:13:35 +07:00
ctx.AddSource(".UnitGeneratorAttribute.g.cs", SourceText.From(attributeSource, Encoding.UTF8));
});
2026-05-29 01:33:38 +07:00
// Ищем все readonly partial record struct с атрибутом
var structsProvider = context.SyntaxProvider
.CreateSyntaxProvider(
2026-05-29 01:33:38 +07:00
predicate: static (node, _) => IsTargetStruct(node),
transform: static (ctx, _) => GetStructInfo(ctx))
.Where(info => info.HasValue)
.Select((info, _) => info!.Value)
.Collect();
2026-05-29 01:33:38 +07:00
context.RegisterSourceOutput(structsProvider, (spc, structs) =>
{
2026-06-09 16:45:22 +07:00
// Сначала генерируем сами структуры (Ваш оригинальный код без изменений)
2026-05-29 01:33:38 +07:00
foreach (var structInfo in structs)
{
string generatedCode = GeneratePartial(structInfo);
2026-06-05 12:13:35 +07:00
spc.AddSource($"{structInfo.TypeName}.g.cs", SourceText.From(generatedCode, Encoding.UTF8));
2026-05-29 01:33:38 +07:00
}
2026-06-09 16:45:22 +07:00
// --- ТОТ САМЫЙ КОД УСЛОВНОЙ ГЕНЕРАЦИИ БИНДЕРА ---
bool isMainAssembly = false;
var firstStruct = structs.FirstOrDefault();
if (firstStruct.TypeName != null)
{
// Если пространство имен содержит "Tests", флаг останется false
if (firstStruct.Namespace.StartsWith("QWERTYkez.Mensura") && !firstStruct.Namespace.Contains("Tests"))
{
isMainAssembly = true;
}
}
// Генерируем файл ТОЛЬКО для основного проекта
if (isMainAssembly)
{
var sb = new StringBuilder();
sb.AppendLine("namespace QWERTYkez.Mensura;");
sb.AppendLine("internal static partial class MensuraBinder");
sb.AppendLine("{");
sb.AppendLine(" [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.NoInlining)]");
sb.AppendLine(" static partial void AddGeneratedConverters(System.Collections.IList converters)");
sb.AppendLine(" {");
foreach (var structInfo in structs)
{
sb.AppendLine($" converters.Add(new QWERTYkez.Mensura.MensuraBinder.NewtonsoftRegistrar.NewtonsoftUnitConverter<QWERTYkez.Mensura.Units.{structInfo.TypeName}>());");
}
sb.AppendLine(" }");
sb.AppendLine("}");
spc.AddSource(".Units.MensuraBinder.g.cs", SourceText.From(sb.ToString(), Encoding.UTF8));
}
2026-05-29 01:33:38 +07:00
});
}
2026-05-29 01:33:38 +07:00
private static bool IsTargetStruct(SyntaxNode node)
{
2026-05-29 01:33:38 +07:00
if (node is not RecordDeclarationSyntax record)
return false;
// Должен быть record struct с модификаторами readonly и partial
if (!record.Modifiers.Any(SyntaxKind.ReadOnlyKeyword) ||
!record.Modifiers.Any(SyntaxKind.PartialKeyword) ||
!record.Keyword.IsKind(SyntaxKind.RecordKeyword) ||
!record.ClassOrStructKeyword.IsKind(SyntaxKind.StructKeyword))
return false;
2026-06-05 12:13:35 +07:00
// Проверяем наличие атрибута [UnitGenerator]
2026-05-29 01:33:38 +07:00
foreach (var attrList in record.AttributeLists)
foreach (var attr in attrList.Attributes)
{
string name = attr.Name.ToString();
if (name == AttributeName || name == AttributeName + "Attribute")
return true;
}
return false;
}
2026-05-29 01:33:38 +07:00
private static StructInfo? GetStructInfo(GeneratorSyntaxContext context)
{
var record = (RecordDeclarationSyntax)context.Node;
var semanticModel = context.SemanticModel;
2026-05-29 01:33:38 +07:00
if (semanticModel.GetDeclaredSymbol(record) is not INamedTypeSymbol typeSymbol)
return null;
2026-05-29 01:33:38 +07:00
string namespaceName = typeSymbol.ContainingNamespace?.ToString();
if (string.IsNullOrEmpty(namespaceName))
return null;
return new StructInfo(namespaceName, typeSymbol.Name);
}
private static string GeneratePartial(StructInfo info)
{
string typeName = info.TypeName;
string ns = info.Namespace;
// Здесь должен быть полный код из вашего файла XXXXXXXXXXXX.cs
2026-06-01 01:31:31 +07:00
// с заменой {typeName} на {typeName}. Для краткости приведён скелет.
2026-05-29 01:33:38 +07:00
// Вы должны скопировать сюда всё содержимое вашего второго файла,
2026-06-01 01:31:31 +07:00
// заменив {typeName} на {typeName}.
2026-05-29 01:33:38 +07:00
string skeleton = @"
2026-06-03 12:07:27 +07:00
global using {typeName}Extensions = QWERTYkez.Mensura.Units.{typeName}Extensions;
2026-05-29 16:45:24 +07:00
global using {typeName} = QWERTYkez.Mensura.Units.{typeName};
2026-06-03 12:07:27 +07:00
2026-05-29 01:33:38 +07:00
using System.Runtime.Serialization;
namespace QWERTYkez.Mensura.Units;
2026-06-09 16:45:22 +07:00
[JsonConverter(typeof(UnitJsonConverter<{typeName}>))]
2026-05-29 16:45:24 +07:00
public readonly partial record struct {typeName} : IMensuraUnit<{typeName}>, IEquatable<{typeName}>, IMensuraUnit
2026-05-29 01:33:38 +07:00
{
2026-06-04 12:03:39 +07:00
[JsonInclude, DataMember, JsonPropertyName(""v""), Obsolete] // для JSON / EF на случай сбоев, если пробелма с _Value
2026-05-29 01:33:38 +07:00
internal double Value { get => _Value; init => _Value = value; }
internal readonly double _Value;
internal {typeName}(double value) => _Value = value;
public override int GetHashCode() => _Value.GetHashCode();
public int CompareTo({typeName}? other) => _Value.CompareTo(other is null ? 0d : other.Value._Value);
public int CompareTo({typeName} other) => _Value.CompareTo(other._Value);
public bool Equals({typeName}? other) => _Value.Equals(other?._Value);
2026-06-08 12:00:10 +07:00
public static explicit operator {typeName}(double val) => Unsafe.As<double, {typeName}>(ref val);
public static explicit operator double({typeName} unit) => unit._Value;
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public {typeName} Abs() => new(Math.Abs(_Value));
[MethodImpl(MethodImplOptions.AggressiveInlining)]
internal R Pow2_Internal<R>() where R : struct, IMensuraUnit, IEquatable<R> => (_Value * _Value).ToUnit<R>();
[MethodImpl(MethodImplOptions.AggressiveInlining)]
internal R Sqrt_Internal<R>() where R : struct, IMensuraUnit, IEquatable<R> => Math.Sqrt(_Value).ToUnit<R>();
2026-05-29 01:33:38 +07:00
[JsonIgnore, IgnoreDataMember] public bool IsPositive => _Value >= 0;
[JsonIgnore, IgnoreDataMember] public bool IsGreaterThanZero => _Value > 0;
2026-06-02 12:28:46 +07:00
[JsonIgnore, IgnoreDataMember] public bool IsNegative => double.IsNegative(_Value);
2026-05-29 01:33:38 +07:00
[JsonIgnore, IgnoreDataMember] public bool IsZero => _Value == 0;
2026-06-02 12:28:46 +07:00
[JsonIgnore, IgnoreDataMember] public bool IsNaN => double.IsNaN(_Value);
[JsonIgnore, IgnoreDataMember] public bool IsFinite => double.IsFinite(_Value);
[JsonIgnore, IgnoreDataMember] public bool IsInfinity => double.IsInfinity(_Value);
[JsonIgnore, IgnoreDataMember] public bool IsPositiveInfinity => double.IsPositiveInfinity(_Value);
[JsonIgnore, IgnoreDataMember] public bool IsNegativeInfinity => double.IsNegativeInfinity(_Value);
2026-05-29 01:33:38 +07:00
public static {typeName} Zero { get; } = new(0d);
public static {typeName} Min { get; } = new(double.MinValue);
public static {typeName} Max { get; } = new(double.MaxValue);
public static {typeName} NegativeInfinity { get; } = new(double.NegativeInfinity);
public static {typeName} PositiveInfinity { get; } = new(double.PositiveInfinity);
2026-06-03 12:07:27 +07:00
public static bool operator ==({typeName}? T1, {typeName}? T2) => T1.Protected() == T2.Protected();
public static bool operator !=({typeName}? T1, {typeName}? T2) => T1.Protected() != T2.Protected();
2026-05-29 01:33:38 +07:00
2026-06-03 12:07:27 +07:00
public static bool operator <({typeName}? T1, {typeName}? T2) => T1.Protected() < T2.Protected();
public static bool operator <=({typeName}? T1, {typeName}? T2) => T1.Protected() <= T2.Protected();
public static bool operator >({typeName}? T1, {typeName}? T2) => T1.Protected() > T2.Protected();
public static bool operator >=({typeName}? T1, {typeName}? T2) => T1.Protected() >= T2.Protected();
2026-05-29 01:33:38 +07:00
public static {typeName} operator +({typeName} T2) => new(+T2._Value);
public static {typeName} operator +({typeName} T1, {typeName} T2) => new(T1._Value + T2._Value);
public static {typeName} operator -({typeName} T2) => new(-T2._Value);
public static {typeName} operator -({typeName} T1, {typeName} T2) => new(T1._Value - T2._Value);
// double
public static {typeName} operator *({typeName} T1, double T2) => new(T1._Value * T2);
public static {typeName} operator *({typeName} T1, double? T2) => T1 * (T2 ?? 0d);
public static {typeName} operator *(double T1, {typeName} T2) => new(T1 * T2._Value);
public static {typeName} operator *(double? T1, {typeName} T2) => (T1 ?? 0d) * T2;
public static {typeName} operator /({typeName} T1, double T2) => new(T1._Value / T2);
public static {typeName} operator /({typeName} T1, double? T2) => T1 / (T2 ?? 0d);
public static double operator /({typeName} T1, {typeName} T2) => T1._Value / T2._Value;
// sbyte
public static {typeName} operator *({typeName} T1, sbyte T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, sbyte? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(sbyte T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(sbyte? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, sbyte T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, sbyte? T2) => T1 / T2.ToDouble();
// short
public static {typeName} operator *({typeName} T1, short T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, short? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(short T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(short? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, short T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, short? T2) => T1 / T2.ToDouble();
// int
public static {typeName} operator *({typeName} T1, int T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, int? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(int T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(int? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, int T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, int? T2) => T1 / T2.ToDouble();
// long
public static {typeName} operator *({typeName} T1, long T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, long? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(long T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(long? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, long T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, long? T2) => T1 / T2.ToDouble();
// byte
public static {typeName} operator *({typeName} T1, byte T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, byte? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(byte T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(byte? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, byte T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, byte? T2) => T1 / T2.ToDouble();
// ushort
public static {typeName} operator *({typeName} T1, ushort T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, ushort? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(ushort T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(ushort? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, ushort T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, ushort? T2) => T1 / T2.ToDouble();
// uint
public static {typeName} operator *({typeName} T1, uint T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, uint? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(uint T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(uint? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, uint T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, uint? T2) => T1 / T2.ToDouble();
// ulong
public static {typeName} operator *({typeName} T1, ulong T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, ulong? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(ulong T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(ulong? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, ulong T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, ulong? T2) => T1 / T2.ToDouble();
// nint
public static {typeName} operator *({typeName} T1, nint T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, nint? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(nint T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(nint? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, nint T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, nint? T2) => T1 / T2.ToDouble();
// nuint
public static {typeName} operator *({typeName} T1, nuint T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, nuint? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(nuint T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(nuint? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, nuint T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, nuint? T2) => T1 / T2.ToDouble();
2026-06-10 16:05:42 +07:00
// Half
public static {typeName} operator *({typeName} T1, Half T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, Half? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(Half T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(Half? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, Half T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, Half? T2) => T1 / T2.ToDouble();
2026-05-29 01:33:38 +07:00
// float
public static {typeName} operator *({typeName} T1, float T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, float? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(float T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(float? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, float T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, float? T2) => T1 / T2.ToDouble();
// decimal
public static {typeName} operator *({typeName} T1, decimal T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, decimal? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(decimal T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(decimal? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, decimal T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, decimal? T2) => T1 / T2.ToDouble();
#if NET7_0_OR_GREATER
// Int128
public static {typeName} operator *({typeName} T1, Int128 T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, Int128? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(Int128 T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(Int128? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, Int128 T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, Int128? T2) => T1 / T2.ToDouble();
// UInt128
public static {typeName} operator *({typeName} T1, UInt128 T2) => T1 * T2.ToDouble();
public static {typeName} operator *({typeName} T1, UInt128? T2) => T1 * T2.ToDouble();
public static {typeName} operator *(UInt128 T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator *(UInt128? T1, {typeName} T2) => T1.ToDouble() * T2;
public static {typeName} operator /({typeName} T1, UInt128 T2) => T1 / T2.ToDouble();
public static {typeName} operator /({typeName} T1, UInt128? T2) => T1 / T2.ToDouble();
#endif
2026-06-10 16:05:42 +07:00
public static {typeName}[] operator +({typeName}[] T1, {typeName} T2) => T1.Plus(T2._Value);
public static {typeName}?[] operator +({typeName}?[] T1, {typeName} T2) => T1.Plus(T2._Value);
public static {typeName}[] operator +({typeName} T1, {typeName}[] T2) => T1._Value.Plus(T2);
public static {typeName}?[] operator +({typeName} T1, {typeName}?[] T2) => T1._Value.Plus(T2);
public static {typeName}[] operator +({typeName}[] T1, {typeName}? T2) => T1.Plus(T2.Protected());
public static {typeName}?[] operator +({typeName}?[] T1, {typeName}? T2) => T1.Plus(T2.Protected());
public static {typeName}[] operator +({typeName}? T1, {typeName}[] T2) => T1.Protected().Plus(T2);
public static {typeName}?[] operator +({typeName}? T1, {typeName}?[] T2) => T1.Protected().Plus(T2);
public static {typeName}[] operator -({typeName}[] T1, {typeName} T2) => T1.Minus(T2._Value);
public static {typeName}?[] operator -({typeName}?[] T1, {typeName} T2) => T1.Minus(T2._Value);
public static {typeName}[] operator -({typeName} T1, {typeName}[] T2) => T1._Value.Minus(T2);
public static {typeName}?[] operator -({typeName} T1, {typeName}?[] T2) => T1._Value.Minus(T2);
public static {typeName}[] operator -({typeName}[] T1, {typeName}? T2) => T1.Minus(T2.Protected());
public static {typeName}?[] operator -({typeName}?[] T1, {typeName}? T2) => T1.Minus(T2.Protected());
public static {typeName}[] operator -({typeName}? T1, {typeName}[] T2) => T1.Protected().Minus(T2);
public static {typeName}?[] operator -({typeName}? T1, {typeName}?[] T2) => T1.Protected().Minus(T2);
public static double[] operator /({typeName} T1, {typeName}[] T2) => T1.Div(T2);
public static double?[] operator /({typeName} T1, {typeName}?[] T2) => T1.Div(T2);
public static double[] operator /({typeName}[] T1, {typeName} T2) => T1.Div(T2);
public static double?[] operator /({typeName}?[] T1, {typeName} T2) => T1.Div(T2);
public static double[] operator /({typeName}? T1, {typeName}[] T2) => T1.ProtectedU().Div(T2);
public static double?[] operator /({typeName}? T1, {typeName}?[] T2) => T1.ProtectedU().Div(T2);
public static double[] operator /({typeName}[] T1, {typeName}? T2) => T1.Div(T2.ProtectedU());
public static double?[] operator /({typeName}?[] T1, {typeName}? T2) => T1.Div(T2.ProtectedU());
// double
public static {typeName}[] operator *({typeName} T1, double[] T2) => T1.Mul(T2);
public static {typeName}?[] operator *({typeName} T1, double?[] T2) => T1.Mul(T2);
public static {typeName}[] operator *({typeName}? T1, double[] T2) => T1.ProtectedU().Mul(T2);
public static {typeName}?[] operator *({typeName}? T1, double?[] T2) => T1.ProtectedU().Mul(T2);
public static {typeName}[] operator *(double[] T1, {typeName} T2) => T1.Mul(T2);
public static {typeName}?[] operator *(double?[] T1, {typeName} T2) => T1.Mul(T2);
public static {typeName}[] operator *(double[] T1, {typeName}? T2) => T1.Mul(T2.ProtectedU());
public static {typeName}?[] operator *(double?[] T1, {typeName}? T2) => T1.Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, double[] T2) => T1.Div(T2);
public static {typeName}?[] operator /({typeName} T1, double?[] T2) => T1.Div(T2);
public static {typeName}[] operator /({typeName}? T1, double[] T2) => T1.ProtectedU().Div(T2);
public static {typeName}?[] operator /({typeName}? T1, double?[] T2) => T1.ProtectedU().Div(T2);
// sbyte
public static {typeName}[] operator *({typeName} T1, sbyte[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, sbyte?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, sbyte[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, sbyte?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(sbyte[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(sbyte?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(sbyte[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(sbyte?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, sbyte[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, sbyte?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, sbyte[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, sbyte?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// short
public static {typeName}[] operator *({typeName} T1, short[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, short?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, short[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, short?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(short[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(short?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(short[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(short?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, short[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, short?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, short[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, short?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// int
public static {typeName}[] operator *({typeName} T1, int[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, int?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, int[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, int?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(int[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(int?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(int[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(int?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, int[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, int?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, int[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, int?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// long
public static {typeName}[] operator *({typeName} T1, long[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, long?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, long[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, long?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(long[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(long?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(long[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(long?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, long[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, long?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, long[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, long?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// byte
public static {typeName}[] operator *({typeName} T1, byte[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, byte?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, byte[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, byte?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(byte[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(byte?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(byte[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(byte?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, byte[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, byte?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, byte[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, byte?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// ushort
public static {typeName}[] operator *({typeName} T1, ushort[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, ushort?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, ushort[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, ushort?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(ushort[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(ushort?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(ushort[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(ushort?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, ushort[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, ushort?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, ushort[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, ushort?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// uint
public static {typeName}[] operator *({typeName} T1, uint[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, uint?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, uint[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, uint?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(uint[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(uint?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(uint[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(uint?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, uint[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, uint?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, uint[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, uint?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// ulong
public static {typeName}[] operator *({typeName} T1, ulong[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, ulong?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, ulong[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, ulong?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(ulong[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(ulong?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(ulong[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(ulong?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, ulong[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, ulong?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, ulong[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, ulong?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// nint
public static {typeName}[] operator *({typeName} T1, nint[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, nint?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, nint[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, nint?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(nint[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(nint?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(nint[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(nint?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, nint[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, nint?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, nint[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, nint?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// nuint
public static {typeName}[] operator *({typeName} T1, nuint[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, nuint?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, nuint[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, nuint?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(nuint[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(nuint?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(nuint[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(nuint?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, nuint[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, nuint?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, nuint[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, nuint?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// Half
public static {typeName}[] operator *({typeName} T1, Half[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, Half?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, Half[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, Half?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(Half[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(Half?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(Half[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(Half?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, Half[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, Half?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, Half[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, Half?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// float
public static {typeName}[] operator *({typeName} T1, float[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, float?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, float[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, float?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(float[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(float?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(float[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(float?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, float[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, float?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, float[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, float?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// decimal
public static {typeName}[] operator *({typeName} T1, decimal[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, decimal?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, decimal[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, decimal?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(decimal[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(decimal?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(decimal[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(decimal?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, decimal[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, decimal?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, decimal[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, decimal?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
#if NET7_0_OR_GREATER
// Int128
public static {typeName}[] operator *({typeName} T1, Int128[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, Int128?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, Int128[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, Int128?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(Int128[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(Int128?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(Int128[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(Int128?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, Int128[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, Int128?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, Int128[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, Int128?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
// UInt128
public static {typeName}[] operator *({typeName} T1, UInt128[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName} T1, UInt128?[] T2) => T1.Mul(T2.ToDouble());
public static {typeName}[] operator *({typeName}? T1, UInt128[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}?[] operator *({typeName}? T1, UInt128?[] T2) => T1.ProtectedU().Mul(T2.ToDouble());
public static {typeName}[] operator *(UInt128[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}?[] operator *(UInt128?[] T1, {typeName} T2) => T1.ToDouble().Mul(T2);
public static {typeName}[] operator *(UInt128[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}?[] operator *(UInt128?[] T1, {typeName}? T2) => T1.ToDouble().Mul(T2.ProtectedU());
public static {typeName}[] operator /({typeName} T1, UInt128[] T2) => T1.Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName} T1, UInt128?[] T2) => T1.Div(T2.ToDouble());
public static {typeName}[] operator /({typeName}? T1, UInt128[] T2) => T1.ProtectedU().Div(T2.ToDouble());
public static {typeName}?[] operator /({typeName}? T1, UInt128?[] T2) => T1.ProtectedU().Div(T2.ToDouble());
#endif
2026-05-29 01:33:38 +07:00
}
2026-06-03 12:07:27 +07:00
public static class {typeName}Extensions
{
2026-06-09 16:45:22 +07:00
internal static double Protected(this {typeName}? unit) => unit is null ? 0d : unit.Value._Value;
2026-06-10 16:05:42 +07:00
internal static {typeName} ProtectedU(this {typeName}? unit) => unit is null ? {typeName}.Zero : unit.Value;
2026-06-09 16:45:22 +07:00
// === ReadOnlySpan
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Div(this ReadOnlySpan<{typeName}> units,
double divisor, Span<{typeName}> destination) => units.Div<{typeName}>(divisor, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Div(this ReadOnlySpan<{typeName}?> units,
double divisor, Span<{typeName}?> destination) => units.Div<{typeName}>(divisor, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Div(this double dividend,
ReadOnlySpan<{typeName}> units, Span<{typeName}> destination) => dividend.Div<{typeName}>(units, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Div(this double dividend,
ReadOnlySpan<{typeName}?> units, Span<{typeName}?> destination) => dividend.Div<{typeName}>(units, destination);
// === Array ===
[MethodImpl(MethodImplOptions.AggressiveInlining)]public static {typeName}[] Div(
this {typeName}[] units, double divisor) => units.Div<{typeName}>(divisor);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static {typeName}?[] Div(
this {typeName}?[] units, double divisor) => units.Div<{typeName}>(divisor);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static {typeName}[] Div(
this double dividend, {typeName}[] units) => dividend.Div<{typeName}>(units);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static {typeName}?[] Div(
this double dividend, {typeName}?[] units) => dividend.Div<{typeName}>(units);
// === List<T> ===
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static List<{typeName}> Div(
this List<{typeName}> units, double divisor) => units.Div<{typeName}>(divisor);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static List<{typeName}?> Div(
this List<{typeName}?> units, double divisor) => units.Div<{typeName}>(divisor);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static List<{typeName}> Div(
this double dividend, List<{typeName}> units) => dividend.Div<{typeName}>(units);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static List<{typeName}?> Div(
this double dividend, List<{typeName}?> units) => dividend.Div<{typeName}>(units);
// === ICollection<T> ===
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Div(this ICollection<{typeName}> units,
double divisor, Span<{typeName}> destination) => units.Div<{typeName}>(divisor, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Div(this ICollection<{typeName}?> units,
double divisor, Span<{typeName}?> destination) => units.Div<{typeName}>(divisor, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Div(this double dividend,
ICollection<{typeName}> units, Span<{typeName}> destination) => dividend.Div<{typeName}>(units, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Div(this double dividend,
ICollection<{typeName}?> units, Span<{typeName}?> destination) => dividend.Div<{typeName}>(units, destination);
// === IReadOnlyCollection<T> ===
[MethodImpl(MethodImplOptions.AggressiveInlining)]public static void Div(this IReadOnlyCollection<{typeName}> units,
double divisor, Span<{typeName}> destination) => units.Div<{typeName}>(divisor, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)]public static void Div(this IReadOnlyCollection<{typeName}?> units,
double divisor, Span<{typeName}?> destination) => units.Div<{typeName}>(divisor, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Div(this double dividend,
IReadOnlyCollection<{typeName}> units, Span<{typeName}> destination) => dividend.Div<{typeName}>(units, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Div(this double dividend,
IReadOnlyCollection<{typeName}?> units, Span<{typeName}?> destination) => dividend.Div<{typeName}>(units, destination);
// === IEnumerable<T> ===
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static IEnumerable<{typeName}> Div(
this IEnumerable<{typeName}> units, double divisor) => units.Div<{typeName}>(divisor);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static IEnumerable<{typeName}?> Div(
this IEnumerable<{typeName}?> units, double divisor) => units.Div<{typeName}>(divisor);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static IEnumerable<{typeName}> Div(
this double dividend, IEnumerable<{typeName}> units) => dividend.Div<{typeName}>(units);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static IEnumerable<{typeName}?> Div(
this double dividend, IEnumerable<{typeName}?> units) => dividend.Div<{typeName}>(units);
// === ReadOnlySpan
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Mul(this ReadOnlySpan<{typeName}> units,
double multiplicator, Span<{typeName}> destination) => units.Mul<{typeName}>(multiplicator, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Mul(this ReadOnlySpan<{typeName}?> units,
double multiplicator, Span<{typeName}?> destination) => units.Mul<{typeName}>(multiplicator, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Mul(this double multiplicator,
ReadOnlySpan<{typeName}> units, Span<{typeName}> destination) => units.Mul<{typeName}>(multiplicator, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Mul(this double multiplicator,
ReadOnlySpan<{typeName}?> units, Span<{typeName}?> destination) => units.Mul<{typeName}>(multiplicator, destination);
// === Array ===
[MethodImpl(MethodImplOptions.AggressiveInlining)]public static {typeName}[] Mul(
this {typeName}[] units, double multiplicator) => units.Mul<{typeName}>(multiplicator);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static {typeName}?[] Mul(
this {typeName}?[] units, double multiplicator) => units.Mul<{typeName}>(multiplicator);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static {typeName}[] Mul(
this double multiplicator, {typeName}[] units) => units.Mul<{typeName}>(multiplicator);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static {typeName}?[] Mul(
this double multiplicator, {typeName}?[] units) => units.Mul<{typeName}>(multiplicator);
// === List<T> ===
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static List<{typeName}> Mul(
this List<{typeName}> units, double multiplicator) => units.Mul<{typeName}>(multiplicator);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static List<{typeName}?> Mul(
this List<{typeName}?> units, double multiplicator) => units.Mul<{typeName}>(multiplicator);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static List<{typeName}> Mul(
this double multiplicator, List<{typeName}> units) => units.Mul<{typeName}>(multiplicator);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static List<{typeName}?> Mul(
this double multiplicator, List<{typeName}?> units) => units.Mul<{typeName}>(multiplicator);
// === ICollection<T> ===
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Mul(this ICollection<{typeName}> units,
double multiplicator, Span<{typeName}> destination) => units.Mul<{typeName}>(multiplicator, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Mul(this ICollection<{typeName}?> units,
double multiplicator, Span<{typeName}?> destination) => units.Mul<{typeName}>(multiplicator, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Mul(this double multiplicator,
ICollection<{typeName}> units, Span<{typeName}> destination) => units.Mul<{typeName}>(multiplicator, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Mul(this double multiplicator,
ICollection<{typeName}?> units, Span<{typeName}?> destination) => units.Mul<{typeName}>(multiplicator, destination);
// === IReadOnlyCollection<T> ===
[MethodImpl(MethodImplOptions.AggressiveInlining)]public static void Mul(this IReadOnlyCollection<{typeName}> units,
double multiplicator, Span<{typeName}> destination) => units.Mul<{typeName}>(multiplicator, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)]public static void Mul(this IReadOnlyCollection<{typeName}?> units,
double multiplicator, Span<{typeName}?> destination) => units.Mul<{typeName}>(multiplicator, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Mul(this double multiplicator,
IReadOnlyCollection<{typeName}> units, Span<{typeName}> destination) => units.Mul<{typeName}>(multiplicator, destination);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static void Mul(this double multiplicator,
IReadOnlyCollection<{typeName}?> units, Span<{typeName}?> destination) => units.Mul<{typeName}>(multiplicator, destination);
// === IEnumerable<T> ===
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static IEnumerable<{typeName}> Mul(
this IEnumerable<{typeName}> units, double multiplicator) => units.Mul<{typeName}>(multiplicator);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static IEnumerable<{typeName}?> Mul(
this IEnumerable<{typeName}?> units, double multiplicator) => units.Mul<{typeName}>(multiplicator);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static IEnumerable<{typeName}> Mul(
this double multiplicator, IEnumerable<{typeName}> units) => units.Mul<{typeName}>(multiplicator);
[MethodImpl(MethodImplOptions.AggressiveInlining)] public static IEnumerable<{typeName}?> Mul(
this double multiplicator, IEnumerable<{typeName}?> units) => units.Mul<{typeName}>(multiplicator);
2026-06-10 16:05:42 +07:00
// Sum Avg Max Min (не nullable) ==========================================
2026-06-09 16:45:22 +07:00
// === ReadOnlySpan === SIMD
public static {typeName} Sum(this ReadOnlySpan<{typeName}> units) => AggregateUnitExtensions.Sum(units);
2026-06-10 16:05:42 +07:00
public static {typeName} Avg(this ReadOnlySpan<{typeName}> units) => AggregateUnitExtensions.Avg(units);
2026-06-09 16:45:22 +07:00
public static {typeName} Max(this ReadOnlySpan<{typeName}> units) => AggregateUnitExtensions.Max(units);
public static {typeName} Min(this ReadOnlySpan<{typeName}> units) => AggregateUnitExtensions.Min(units);
// === List<T> ===
public static {typeName} Sum(this List<{typeName}> list) => AggregateUnitExtensions.Sum(list);
2026-06-10 16:05:42 +07:00
public static {typeName} Avg(this List<{typeName}> list) => AggregateUnitExtensions.Avg(list);
2026-06-09 16:45:22 +07:00
public static {typeName} Max(this List<{typeName}> list) => AggregateUnitExtensions.Max(list);
public static {typeName} Min(this List<{typeName}> list) => AggregateUnitExtensions.Min(list);
// === ICollection<T> ===
public static {typeName} Sum(this ICollection<{typeName}> collection) => AggregateUnitExtensions.Sum(collection);
2026-06-10 16:05:42 +07:00
public static {typeName} Avg(this ICollection<{typeName}> collection) => AggregateUnitExtensions.Avg(collection);
2026-06-09 16:45:22 +07:00
public static {typeName} Max(this ICollection<{typeName}> collection) => AggregateUnitExtensions.Max(collection);
public static {typeName} Min(this ICollection<{typeName}> collection) => AggregateUnitExtensions.Min(collection);
// === IReadOnlyCollection<T> ===
public static {typeName} Sum(this IReadOnlyCollection<{typeName}> collection) => AggregateUnitExtensions.Sum(collection);
2026-06-10 16:05:42 +07:00
public static {typeName} Avg(this IReadOnlyCollection<{typeName}> collection) => AggregateUnitExtensions.Avg(collection);
2026-06-09 16:45:22 +07:00
public static {typeName} Max(this IReadOnlyCollection<{typeName}> collection) => AggregateUnitExtensions.Max(collection);
public static {typeName} Min(this IReadOnlyCollection<{typeName}> collection) => AggregateUnitExtensions.Min(collection);
// === IEnumerable<T> ===
public static {typeName} Sum(this IEnumerable<{typeName}> collection) => AggregateUnitExtensions.Sum(collection);
2026-06-10 16:05:42 +07:00
public static {typeName} Avg(this IEnumerable<{typeName}> collection) => AggregateUnitExtensions.Avg(collection);
2026-06-09 16:45:22 +07:00
public static {typeName} Max(this IEnumerable<{typeName}> collection) => AggregateUnitExtensions.Max(collection);
public static {typeName} Min(this IEnumerable<{typeName}> collection) => AggregateUnitExtensions.Min(collection);
2026-06-10 16:05:42 +07:00
// Sum Avg Max Min (nullable) ==========================================
2026-06-09 16:45:22 +07:00
// === ReadOnlySpan ===
public static {typeName} Sum(this ReadOnlySpan<{typeName}?> units) => AggregateUnitExtensions.Sum(units);
2026-06-10 16:05:42 +07:00
public static {typeName} Avg(this ReadOnlySpan<{typeName}?> units) => AggregateUnitExtensions.Avg(units);
2026-06-09 16:45:22 +07:00
public static {typeName} Max(this ReadOnlySpan<{typeName}?> units) => AggregateUnitExtensions.Max(units);
public static {typeName} Min(this ReadOnlySpan<{typeName}?> units) => AggregateUnitExtensions.Min(units);
// === List<T> ===
public static {typeName} Sum(this List<{typeName}?> list) => AggregateUnitExtensions.Sum(list);
2026-06-10 16:05:42 +07:00
public static {typeName} Avg(this List<{typeName}?> list) => AggregateUnitExtensions.Avg(list);
2026-06-09 16:45:22 +07:00
public static {typeName} Max(this List<{typeName}?> list) => AggregateUnitExtensions.Max(list);
public static {typeName} Min(this List<{typeName}?> list) => AggregateUnitExtensions.Min(list);
// === ICollection<T> ===
public static {typeName} Sum(this ICollection<{typeName}?> collection) => AggregateUnitExtensions.Sum(collection);
2026-06-10 16:05:42 +07:00
public static {typeName} Avg(this ICollection<{typeName}?> collection) => AggregateUnitExtensions.Avg(collection);
2026-06-09 16:45:22 +07:00
public static {typeName} Max(this ICollection<{typeName}?> collection) => AggregateUnitExtensions.Max(collection);
public static {typeName} Min(this ICollection<{typeName}?> collection) => AggregateUnitExtensions.Min(collection);
// === IReadOnlyCollection<T> ===
public static {typeName} Sum(this IReadOnlyCollection<{typeName}?> collection) => AggregateUnitExtensions.Sum(collection);
2026-06-10 16:05:42 +07:00
public static {typeName} Avg(this IReadOnlyCollection<{typeName}?> collection) => AggregateUnitExtensions.Avg(collection);
2026-06-09 16:45:22 +07:00
public static {typeName} Max(this IReadOnlyCollection<{typeName}?> collection) => AggregateUnitExtensions.Max(collection);
public static {typeName} Min(this IReadOnlyCollection<{typeName}?> collection) => AggregateUnitExtensions.Min(collection);
// === IEnumerable<T> ===
public static {typeName} Sum(this IEnumerable<{typeName}?> collection) => AggregateUnitExtensions.Sum(collection);
2026-06-10 16:05:42 +07:00
public static {typeName} Avg(this IEnumerable<{typeName}?> collection) => AggregateUnitExtensions.Avg(collection);
2026-06-09 16:45:22 +07:00
public static {typeName} Max(this IEnumerable<{typeName}?> collection) => AggregateUnitExtensions.Max(collection);
public static {typeName} Min(this IEnumerable<{typeName}?> collection) => AggregateUnitExtensions.Min(collection);
2026-06-03 12:07:27 +07:00
}
2026-05-29 01:33:38 +07:00
";
return skeleton.Replace("{typeName}", typeName).Replace("{ns}", ns);
}
2026-05-29 01:33:38 +07:00
private readonly struct StructInfo(string ns, string name)
{
public string Namespace { get; } = ns;
2026-05-29 01:33:38 +07:00
public string TypeName { get; } = name;
}
}