diff --git a/CHANGELOG.md b/CHANGELOG.md index be3544c4..2c5b6387 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,6 +7,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] - 2021-02-11 ### Changed +- Update CreateOperatorCodeFixProvider - Use Span.Fill - CeilPow2 AggressiveInlining diff --git a/Source/AtCoderAnalyzer/CreateOperatorCodeFixProvider.cs b/Source/AtCoderAnalyzer/CreateOperatorCodeFixProvider.cs index 7d7892da..50d52422 100644 --- a/Source/AtCoderAnalyzer/CreateOperatorCodeFixProvider.cs +++ b/Source/AtCoderAnalyzer/CreateOperatorCodeFixProvider.cs @@ -70,7 +70,7 @@ public override async Task RegisterCodeFixesAsync(CodeFixContext context) var writtenTypeSyntax = writtenTypeSyntaxes[i]; var originalType = originalTypes[i]; var constraintTypes = originalType.ConstraintTypes; - var writtenType = writtenTypes[i] as INamedTypeSymbol; + var writtenType = writtenTypes[i] as ITypeSymbol; if (!constraintTypes .OfType() diff --git a/Source/AtCoderAnalyzer/Helpers/SimplifyTypeSyntaxRewriter.cs b/Source/AtCoderAnalyzer/Helpers/SimplifyTypeSyntaxRewriter.cs index 165d08a5..de22c3d3 100644 --- a/Source/AtCoderAnalyzer/Helpers/SimplifyTypeSyntaxRewriter.cs +++ b/Source/AtCoderAnalyzer/Helpers/SimplifyTypeSyntaxRewriter.cs @@ -10,15 +10,12 @@ internal class SimplifyTypeSyntaxRewriter : CSharpSyntaxRewriter private ImmutableHashSet Usings { get; } public SimplifyTypeSyntaxRewriter(ImmutableHashSet usings) => Usings = usings; - public override SyntaxNode Visit(SyntaxNode node) + public override SyntaxNode VisitQualifiedName(QualifiedNameSyntax node) { - if (node is QualifiedNameSyntax qualified) - { - if (Usings.Contains(qualified.Left.ToString())) - return qualified.Right; - return qualified; - } - return base.Visit(node); + var right = (SimpleNameSyntax)Visit(node.Right); + if (Usings.Contains(node.Left.ToString())) + return right; + return SyntaxFactory.QualifiedName(node.Left, right); } } } diff --git a/Test/AtCoderAnalyzer.Test/CreateOperatorCodeFixProviderTest.cs b/Test/AtCoderAnalyzer.Test/CreateOperatorCodeFixProviderTest.cs index 32203c8d..67953df0 100644 --- a/Test/AtCoderAnalyzer.Test/CreateOperatorCodeFixProviderTest.cs +++ b/Test/AtCoderAnalyzer.Test/CreateOperatorCodeFixProviderTest.cs @@ -654,7 +654,7 @@ struct Op : System.Collections.Generic.IComparer public int Compare(short x, short y) => default; }"; await VerifyCS.VerifyCodeFixAsync(source, - VerifyCS.Diagnostic().WithSpan(6, 5, 6, 23).WithArguments("Op"), + VerifyCS.Diagnostic("AC0008").WithSpan(6, 5, 6, 23).WithArguments("Op"), fixedSource); } @@ -716,7 +716,7 @@ public void Init(ref (int, long) val, out bool success, params int[] nums) public (int, long) Prop2 { set; get; } }"; await VerifyCS.VerifyCodeFixAsync(source, - VerifyCS.Diagnostic().WithSpan(12, 5, 12, 29).WithArguments("Op"), + VerifyCS.Diagnostic("AC0008").WithSpan(12, 5, 12, 29).WithArguments("Op"), fixedSource); } @@ -770,8 +770,100 @@ public void Init() public (int n, long m) Prop2 { set; get; } }"; await VerifyCS.VerifyCodeFixAsync(source, - VerifyCS.Diagnostic().WithSpan(14, 9, 14, 31).WithArguments("Op"), + VerifyCS.Diagnostic("AC0008").WithSpan(14, 9, 14, 31).WithArguments("Op"), fixedSource); } + + [Fact] + public async Task Array() + { + var source = @" +using AtCoder; +using System.Runtime.CompilerServices; + +[IsOperator] +public interface IAny { + T Prop { get; } +} +class Program +{ + static void M() where TOp : struct, IAny {} + static void Run() + { + M(); + } +} +"; + var fixedSource = @" +using AtCoder; +using System.Runtime.CompilerServices; + +[IsOperator] +public interface IAny { + T Prop { get; } +} +class Program +{ + static void M() where TOp : struct, IAny {} + static void Run() + { + M(); + } +} + +struct BigOp : IAny +{ + public System.Numerics.BigInteger[] Prop => default; +}"; + await VerifyCS.VerifyCodeFixAsync(source, + VerifyCS.Diagnostic("AC0008").WithSpan(14, 9, 14, 47).WithArguments("BigOp"), + fixedSource); + } + + [Fact] + public async Task Generic() + { + var source = @" +using AtCoder; +using System.Runtime.CompilerServices; + +[IsOperator] +public interface IAny { + T Prop { get; } +} +class Program +{ + static void M() where TOp : struct, IAny {} + static void Run() + { + M, ModOp>(); + } +} +"; + var fixedSource = @" +using AtCoder; +using System.Runtime.CompilerServices; + +[IsOperator] +public interface IAny { + T Prop { get; } +} +class Program +{ + static void M() where TOp : struct, IAny {} + static void Run() + { + M, ModOp>(); + } +} + +struct ModOp : IAny> +{ + public StaticModInt Prop => default; +}"; + await VerifyCS.VerifyCodeFixAsync(source, + VerifyCS.Diagnostic("AC0008").WithSpan(14, 9, 14, 46).WithArguments("ModOp"), + fixedSource); + } } }