File: src\Analyzers\CSharp\Tests\AddParameter\AddParameterTests.cs
Web Access
Project: src\src\Features\CSharpTest\Microsoft.CodeAnalysis.CSharp.Features.UnitTests.csproj (Microsoft.CodeAnalysis.CSharp.Features.UnitTests)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
 
using System.Collections.Immutable;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.CodeActions;
using Microsoft.CodeAnalysis.CodeFixes;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.CSharp.AddParameter;
using Microsoft.CodeAnalysis.CSharp.Test.Utilities;
using Microsoft.CodeAnalysis.Diagnostics;
using Microsoft.CodeAnalysis.Editor.CSharp.UnitTests.Diagnostics;
using Microsoft.CodeAnalysis.Test.Utilities;
using Roslyn.Test.Utilities;
using Xunit;
using Xunit.Abstractions;
 
namespace Microsoft.CodeAnalysis.Editor.CSharp.UnitTests.AddParameter;
 
[Trait(Traits.Feature, Traits.Features.CodeActionsAddParameter)]
public sealed class AddParameterTests(ITestOutputHelper logger)
    : AbstractCSharpDiagnosticProviderBasedUserDiagnosticTest_NoEditor(logger)
{
    internal override (DiagnosticAnalyzer?, CodeFixProvider) CreateDiagnosticProviderAndFixer(Workspace workspace)
        => (null, new CSharpAddParameterCodeFixProvider());
 
    protected override ImmutableArray<CodeAction> MassageActions(ImmutableArray<CodeAction> actions)
        => FlattenActions(actions);
 
    [Fact]
    public async Task TestMissingWithImplicitConstructor()
    {
        await TestMissingAsync(
            """
            class C
            {
            }
 
            class D
            {
                void M()
                {
                    new [|C|](1);
                }
            }
            """);
    }
 
    [Fact]
    public async Task TestOnEmptyConstructor()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C() { }
            }
 
            class D
            {
                void M()
                {
                    new [|C|](1);
                }
            }
            """,
            """
            class C
            {
                public C(int v) { }
            }
 
            class D
            {
                void M()
                {
                    new C(1);
                }
            }
            """);
    }
 
    [Fact]
    public async Task TestNamedArg()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C() { }
            }
 
            class D
            {
                void M()
                {
                    new C([|p|]: 1);
                }
            }
            """,
            """
            class C
            {
                public C(int p) { }
            }
 
            class D
            {
                void M()
                {
                    new C(p: 1);
                }
            }
            """);
    }
 
    [Fact]
    public async Task TestMissingWithConstructorWithSameNumberOfParams()
    {
        await TestMissingAsync(
            """
            class C
            {
                public C(bool b) { }
            }
 
            class D
            {
                void M()
                {
                    new [|C|](1);
                }
            }
            """);
    }
 
    [Fact]
    public async Task TestAddBeforeMatchingArg()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(int i) { }
            }
 
            class D
            {
                void M()
                {
                    new [|C|](true, 1);
                }
            }
            """,
            """
            class C
            {
                public C(bool v, int i) { }
            }
 
            class D
            {
                void M()
                {
                    new C(true, 1);
                }
            }
            """);
    }
 
    [Fact]
    public async Task TestAddAfterMatchingConstructorParam()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(int i) { }
            }
 
            class D
            {
                void M()
                {
                    new [|C|](1, true);
                }
            }
            """,
            """
            class C
            {
                public C(int i, bool v) { }
            }
 
            class D
            {
                void M()
                {
                    new C(1, true);
                }
            }
            """);
    }
 
    [Fact]
    public async Task TestParams1()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(params int[] i) { }
            }
 
            class D
            {
                void M()
                {
                    new C([|true|], 1);
                }
            }
            """,
            """
            class C
            {
                public C(bool v, params int[] i) { }
            }
 
            class D
            {
                void M()
                {
                    new C(true, 1);
                }
            }
            """);
    }
 
    [Fact]
    public async Task TestParams2()
    {
        await TestMissingAsync(
            """
            class C
            {
                public C(params int[] i) { }
            }
 
            class D
            {
                void M()
                {
                    new [|C|](1, true);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/20708")]
    public async Task TestMultiLineParameters1()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(int i,
                         /* goo */ int j)
                {
 
                }
 
                private void Goo()
                {
                    new [|C|](true, 0, 0);
                }
            }
            """,
            """
            class C
            {
                public C(bool v,
                         int i,
                         /* goo */ int j)
                {
 
                }
 
                private void Goo()
                {
                    new C(true, 0, 0);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/20708")]
    public async Task TestMultiLineParameters2()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(int i,
                         /* goo */ int j)
                {
 
                }
 
                private void Goo()
                {
                    new [|C|](0, true, 0);
                }
            }
            """,
            """
            class C
            {
                public C(int i,
                         bool v,
                         /* goo */ int j)
                {
 
                }
 
                private void Goo()
                {
                    new C(0, true, 0);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/20708")]
    public async Task TestMultiLineParameters3()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(int i,
                         /* goo */ int j)
                {
 
                }
 
                private void Goo()
                {
                    new [|C|](0, 0, true);
                }
            }
            """,
            """
            class C
            {
                public C(int i,
                         /* goo */ int j,
                         bool v)
                {
 
                }
 
                private void Goo()
                {
                    new C(0, 0, true);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/20708")]
    public async Task TestMultiLineParameters4()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(
                    int i,
                    /* goo */ int j)
                {
 
                }
 
                private void Goo()
                {
                    new [|C|](true, 0, 0);
                }
            }
            """,
            """
            class C
            {
                public C(
                    bool v,
                    int i,
                    /* goo */ int j)
                {
 
                }
 
                private void Goo()
                {
                    new C(true, 0, 0);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/20708")]
    public async Task TestMultiLineParameters5()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(
                    int i,
                    /* goo */ int j)
                {
 
                }
 
                private void Goo()
                {
                    new [|C|](0, true, 0);
                }
            }
            """,
            """
            class C
            {
                public C(
                    int i,
                    bool v,
                    /* goo */ int j)
                {
 
                }
 
                private void Goo()
                {
                    new C(0, true, 0);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/20708")]
    public async Task TestMultiLineParameters6()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(
                    int i,
                    /* goo */ int j)
                {
 
                }
 
                private void Goo()
                {
                    new [|C|](0, 0, true);
                }
            }
            """,
            """
            class C
            {
                public C(
                    int i,
                    /* goo */ int j,
                    bool v)
                {
 
                }
 
                private void Goo()
                {
                    new C(0, 0, true);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/20973")]
    public async Task TestNullArg1()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(int i) { }
            }
 
            class D
            {
                void M()
                {
                    new [|C|](null, 1);
                }
            }
            """,
            """
            class C
            {
                public C(object value, int i) { }
            }
 
            class D
            {
                void M()
                {
                    new C(null, 1);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/20973")]
    public async Task TestNullArg2()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(string s) { }
            }
 
            class D
            {
                void M()
                {
                    new [|C|](null, 1);
                }
            }
            """,
            """
            class C
            {
                public C(string s, int v) { }
            }
 
            class D
            {
                void M()
                {
                    new C(null, 1);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/20973")]
    public async Task TestDefaultArg1()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(int i) { }
            }
 
            class D
            {
                void M()
                {
                    new [|C|](default, 1);
                }
            }
            """,
            """
            class C
            {
                public C(int i, int v) { }
            }
 
            class D
            {
                void M()
                {
                    new C(default, 1);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/20973")]
    public async Task TestDefaultArg2()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                public C(string s) { }
            }
 
            class D
            {
                void M()
                {
                    new [|C|](default, 1);
                }
            }
            """,
            """
            class C
            {
                public C(string s, int v) { }
            }
 
            class D
            {
                void M()
                {
                    new C(default, 1);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationInstanceMethod1()
    {
        await TestInRegularAndScriptAsync(
            """
            class C
            {
                void M1()
                {
                }
                void M2()
                {
                    int i=0;
                    [|M1|](i);
                }
            }
            """,
            """
            class C
            {
                void M1(int i)
                {
                }
                void M2()
                {
                    int i=0;
                    M1(i);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationInheritedMethodGetFixed()
    {
        await TestInRegularAndScriptAsync(
            """
            class Base
            {
                protected void M1()
                {
                }
            }
            class C1 : Base
            {
                void M2()
                {
                    int i = 0;
                    [|M1|](i);
                }
            }
            """,
            """
            class Base
            {
                protected void M1(int i)
                {
                }
            }
            class C1 : Base
            {
                void M2()
                {
                    int i = 0;
                    M1(i);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationInheritedMethodInMetadatGetsNotFixed()
    {
        await TestMissingAsync(
            """
            class C1
            {
                void M2()
                {
                    int i = 0;
                    [|GetHashCode|](i);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationLocalFunction()
    {
        await TestInRegularAndScriptAsync(
            """
            class C1
            {
                void M1()
                {
                    int Local() => 1;
                    [|Local|](2);
                }
            }
            """,
            """
            class C1
            {
                void M1()
                {
                    int Local(int v) => 1;
                    Local(2);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    [Trait("TODO", "Fix broken")]
    public async Task TestInvocationLambda1()
    {
        await TestMissingInRegularAndScriptAsync(
            """
            using System;
            class C1
            {
                void M1()
                {
                    Action a = () => { };
                    [|a|](2);
                }
            }
            """);
        //Should be Action<int> a = (int v) => { };
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationStaticMethod()
    {
        await TestInRegularAndScriptAsync(
            """
            class C1
            {
                static void M1()
                {
                }
                void M2()
                {
                    [|M1|](1);
                }
            }
            """,
            """
            class C1
            {
                static void M1(int v)
                {
                }
                void M2()
                {
                    M1(1);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationExtensionMethod()
    {
        var code =
            """
            namespace N {
            static class Extensions
            {
                public static void ExtensionM1(this object o)
                {
                }
            }
            class C1
            {
                void M1()
                {
                    new object().[|ExtensionM1|](1);
                }
            }}
            """;
        var fix =
            """
            namespace N {
            static class Extensions
            {
                public static void ExtensionM1(this object o, int v)
                {
                }
            }
            class C1
            {
                void M1()
                {
                    new object().ExtensionM1(1);
                }
            }}
            """;
        await TestInRegularAndScriptAsync(code, fix);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationExtensionMethod_StaticInvocationStyle()
    {
        // error CS1501: No overload for method 'ExtensionM1' takes 2 arguments
        var code =
            """
            namespace N {
            static class Extensions
            {
                public static void ExtensionM1(this object o)
                {
                }
            }
            class C1
            {
                void M1()
                {
                    Extensions.[|ExtensionM1|](new object(), 1);
                }
            }}
            """;
        var fix =
            """
            namespace N {
            static class Extensions
            {
                public static void ExtensionM1(this object o, int v)
                {
                }
            }
            class C1
            {
                void M1()
                {
                    Extensions.ExtensionM1(new object(), 1);
                }
            }}
            """;
        await TestInRegularAndScriptAsync(code, fix);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationOverride()
    {
        var code = """
            class Base
            {
                protected virtual void M1() { }
            }
            class C1 : Base
            {
                protected override void M1() { }
                void M2()
                {
                    [|M1|](1);
                }
            }
            """;
        var fix_DeclarationOnly = """
            class Base
            {
                protected virtual void M1() { }
            }
            class C1 : Base
            {
                protected override void M1(int v) { }
                void M2()
                {
                    M1(1);
                }
            }
            """;
        var fix_All = """
            class Base
            {
                protected virtual void M1(int v) { }
            }
            class C1 : Base
            {
                protected override void M1(int v) { }
                void M2()
                {
                    M1(1);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix_DeclarationOnly, index: 0);
        await TestInRegularAndScriptAsync(code, fix_All, index: 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationExplicitInterface()
    {
        var code = """
            interface I1
            {
                void M1();
            }
            class C1 : I1
            {
                void I1.M1() { }
                void M2()
                {
                    ((I1)this).[|M1|](1);
                }
            }
            """;
        var fix_DeclarationOnly = """
            interface I1
            {
                void M1(int v);
            }
            class C1 : I1
            {
                void I1.M1() { }
                void M2()
                {
                    ((I1)this).M1(1);
                }
            }
            """;
        var fix_All = """
            interface I1
            {
                void M1(int v);
            }
            class C1 : I1
            {
                void I1.M1(int v) { }
                void M2()
                {
                    ((I1)this).M1(1);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix_DeclarationOnly, index: 0);
        await TestInRegularAndScriptAsync(code, fix_All, index: 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationImplicitInterface()
    {
        var code =
            """
            interface I1
            {
                void M1();
            }
            class C1 : I1
            {
                public void M1() { }
                void M2()
                {
                    [|M1|](1);
                }
            }
            """;
        var fix_DeclarationOnly = """
            interface I1
            {
                void M1();
            }
            class C1 : I1
            {
                public void M1(int v) { }
                void M2()
                {
                    M1(1);
                }
            }
            """;
        var fix_All = """
            interface I1
            {
                void M1(int v);
            }
            class C1 : I1
            {
                public void M1(int v) { }
                void M2()
                {
                    M1(1);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix_DeclarationOnly, index: 0);
        await TestInRegularAndScriptAsync(code, fix_All, index: 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationImplicitInterfaces()
    {
        var code =
            """
            interface I1
            {
                void M1();
            }
            interface I2
            {
                void M1();
            }
            class C1 : I1, I2
            {
                public void M1() { }
                void M2()
                {
                    [|M1|](1);
                }
            }
            """;
        var fix_DeclarationOnly = """
            interface I1
            {
                void M1();
            }
            interface I2
            {
                void M1();
            }
            class C1 : I1, I2
            {
                public void M1(int v) { }
                void M2()
                {
                    M1(1);
                }
            }
            """;
        var fix_All = """
            interface I1
            {
                void M1(int v);
            }
            interface I2
            {
                void M1(int v);
            }
            class C1 : I1, I2
            {
                public void M1(int v) { }
                void M2()
                {
                    M1(1);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix_DeclarationOnly, index: 0);
        await TestInRegularAndScriptAsync(code, fix_All, index: 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    [Trait("TODO", "Fix broken")]
    public async Task TestInvocationGenericMethod()
    {
        await TestInRegularAndScriptAsync(
            """
            class C1
            {
                void M1<T>(T arg) { }
                void M2()
                {
                    [|M1|](1, 2);
                }
            }
            """,
            """
            class C1
            {
                void M1<T>(T arg, int v) { }
                void M2()
                {
                    M1(1, 2);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationRecursion()
    {
        await TestInRegularAndScriptAsync(
            """
            class C1
            {
                void M1()
                {
                    [|M1|](1);
                }
            }
            """,
            """
            class C1
            {
                void M1(int v)
                {
                    M1(1);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationOverloads1()
    {
        var code =
            """
            class C1
            {
                void M1(string s) { }
                void M1(int i) { }
                void M2()
                {
                    [|M1|](1, 2);
                }
            }
            """;
        var fix0 =
            """
            class C1
            {
                void M1(string s) { }
                void M1(int i, int v) { }
                void M2()
                {
                    M1(1, 2);
                }
            }
            """;
        var fix1 =
            """
            class C1
            {
                void M1(int v, string s) { }
                void M1(int i) { }
                void M2()
                {
                    M1(1, 2);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, 0);
        await TestInRegularAndScriptAsync(code, fix1, 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationOverloads2()
    {
        var code =
            """
            class C1
            {
                void M1(string s1, string s2) { }
                void M1(string s) { }
                void M1(int i) { }
                void M2()
                {
                    M1(1, [|2|]);
                }
            }
            """;
        var fix0 =
            """
            class C1
            {
                void M1(string s1, string s2) { }
                void M1(string s) { }
                void M1(int i, int v) { }
                void M2()
                {
                    M1(1, 2);
                }
            }
            """;
        var fix1 =
            """
            class C1
            {
                void M1(string s1, string s2) { }
                void M1(int v, string s) { }
                void M1(int i) { }
                void M2()
                {
                    M1(1, 2);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, 0);
        await TestInRegularAndScriptAsync(code, fix1, 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationTuple1()
    {
        var code =
            """
            class C1
            {
                void M1((int, int) t1)
                {
                }
                void M2()
                {
                    [|M1|]((0, 0), (1, "1"));
                }
            }
            """;
        var fix0 =
            """
            class C1
            {
                void M1((int, int) t1, (int, string) value)
                {
                }
                void M2()
                {
                    M1((0, 0), (1, "1"));
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationTuple2()
    {
        var code =
            """
            class C1
            {
                void M1((int, int) t1)
                {
                }
                void M2()
                {
                    var tup = (1, "1");
                    [|M1|]((0, 0), tup);
                }
            }
            """;
        var fix0 =
            """
            class C1
            {
                void M1((int, int) t1, (int, string) tup)
                {
                }
                void M2()
                {
                    var tup = (1, "1");
                    M1((0, 0), tup);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationTuple3()
    {
        var code =
            """
            class C1
            {
                void M1((int, int) t1)
                {
                }
                void M2()
                {
                    var tup = (i: 1, s: "1");
                    [|M1|]((0, 0), tup);
                }
            }
            """;
        var fix0 =
            """
            class C1
            {
                void M1((int, int) t1, (int i, string s) tup)
                {
                }
                void M2()
                {
                    var tup = (i: 1, s: "1");
                    M1((0, 0), tup);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Missing_TypeArguments_AddingTypeArgumentAndParameter()
    {
        // error CS0305: Using the generic method 'C1.M1<T>(T)' requires 1 type arguments
        var code =
            """
            class C1
            {
                void M1<T>(T i) { }
                void M2()
                {
                    [|M1|]<int, bool>(1, true);
                }
            }
            """;
        // Could be fixed as void M1<T, T1>(T i, T1 v) { }
        await TestMissingInRegularAndScriptAsync(code);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Missing_TypeArguments_AddingTypeArgument()
    {
        // error CS0308: The non-generic method 'C1.M1(int)' cannot be used with type arguments
        var code =
            """
            class C1
            {
                void M1(int i) { }
                void M2()
                {
                    [|M1<bool>|](1, true);
                }
            }
            """;
        // Could be fixed as void M1<T>(int i, T v) { }
        await TestMissingInRegularAndScriptAsync(code);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    [Trait("TODO", "Fix missing")]
    public async Task TestInvocation_Missing_ExplicitInterfaceImplementation()
    {
        // error CS0539: 'C1.M1(int)' in explicit interface declaration is not a member of interface
        var code =
            """
            interface I1
            {
                void M1();
            }
            class C1 : I1
            {
                    void I1.M1() { }
                    void I1.[|M1|](int i) { }
            }
            """;
        // Could apply argument to interface method: void M1(int i);
        await TestMissingAsync(code);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_OverloadResolutionFailure()
    {
        // error CS1503: Argument 1: cannot convert from 'double' to 'int'
        var code =
            """
            class C1
            {
                void M1(int i1, int i2) { }
                void M1(double d) { }
                void M2()
                {
                    M1([|1.0|], 1);
                }
            }
            """;
        var fix0 =
            """
            class C1
            {
                void M1(int i1, int i2) { }
                void M1(double d, int v) { }
                void M2()
                {
                    M1(1.0, 1);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_LambdaExpressionParameter()
    {
        // error CS1660: Cannot convert lambda expression to type 'int' because it is not a delegate type
        var code =
            """
            class C1
            {
                void M1(int i1, int i2) { }
                void M1(System.Action a) { }
                void M2()
                {
                    M1([|()=> { }|], 1);
                }
            }
            """;
        var fix =
            """
            class C1
            {
                void M1(int i1, int i2) { }
                void M1(System.Action a, int v) { }
                void M2()
                {
                    M1(()=> { }, 1);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_NamedParameter()
    {
        // error CS1739: The best overload for 'M1' does not have a parameter named 'i2'
        var code =
            """
            class C1
            {
                void M1(int i1) { }
                void M2()
                {
                    M1([|i2|]: 1);
                }
            }
            """;
        var fix =
            """
            class C1
            {
                void M1(int i1, int i2) { }
                void M2()
                {
                    M1(i2: 1);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationAddTypeParameter_AddTypeParameterIfUserSpecifiesOne_OnlyTypeArgument()
    {
        var code =
            """
            class C1
            {
                void M1() { }
                void M2()
                {
                    [|M1|]<bool>();
                }
            }
            """;
        // Could be fixed as void M1<T>() { }
        await TestMissingInRegularAndScriptAsync(code);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocationAddTypeParameter_AddTypeParameterIfUserSpecifiesOne_TypeArgumentAndParameterArgument()
    {
        var code =
            """
            class C1
            {
                void M1() { }
                void M2()
                {
                    [|M1|]<bool>(true);
                }
            }
            """;
        // Could be fixed to void M1<T>(T v) { }
        await TestMissingInRegularAndScriptAsync(code);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_ExisitingTypeArgumentIsNotGeneralized()
    {
        var code =
            """
            class C1
            {
                void M1<T>(T v) { }
                void M2()
                {
                    [|M1|](true, true);
                }
            }
            """;
        var fix0 =
            """
            class C1
            {
                void M1<T>(T v, bool v1) { }
                void M2()
                {
                    M1(true, true);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_AddParameterToMethodWithParams()
    {
        // error CS1503: Argument 1: cannot convert from 'bool' to 'int'
        var code =
            """
            class C1
            {
                static void M1(params int[] nums) { }
                static void M2()
                {
                    M1([|true|], 4);
                }
            }
            """;
        var fix0 =
            """
            class C1
            {
                static void M1(bool v, params int[] nums) { }
                static void M2()
                {
                    M1(true, 4);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Cascading_FixingVirtualFixesOverrideToo()
    {
        // error CS1501: No overload for method 'M1' takes 1 arguments
        var code =
            """
            class BaseClass
            {
                protected virtual void M1() { }
            }
            class Derived1: BaseClass
            {
                protected override void M1() { }
            }
            class Test: BaseClass
            {
                void M2() 
                {
                    [|M1|](1);
                }
            }
            """;
        var fix_DeclarationOnly =
            """
            class BaseClass
            {
                protected virtual void M1(int v) { }
            }
            class Derived1: BaseClass
            {
                protected override void M1() { }
            }
            class Test: BaseClass
            {
                void M2() 
                {
                    M1(1);
                }
            }
            """;
        var fix_All =
            """
            class BaseClass
            {
                protected virtual void M1(int v) { }
            }
            class Derived1: BaseClass
            {
                protected override void M1(int v) { }
            }
            class Test: BaseClass
            {
                void M2() 
                {
                    M1(1);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix_DeclarationOnly, index: 0);
        await TestInRegularAndScriptAsync(code, fix_All, index: 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Cascading_PartialMethods()
    {
        var code =
            """
            <Workspace>
                <Project Language="C#" AssemblyName="Assembly1" CommonReferences="true">
                    <Document>
            namespace N1
            {
                partial class C1
                {
                    partial void PartialM();
                }
            }
                    </Document>
                    <Document>
            namespace N1
            {
                partial class C1
                {
                    partial void PartialM() { }
                    void M1()
                    {
                        [|PartialM|](1);
                    }
                }
            }
                    </Document>
                </Project>
            </Workspace>
            """;
        var fix0 =
            """
            <Workspace>
                <Project Language="C#" AssemblyName="Assembly1" CommonReferences="true">
                    <Document>
            namespace N1
            {
                partial class C1
                {
                    partial void PartialM(int v);
                }
            }
                    </Document>
                    <Document>
            namespace N1
            {
                partial class C1
                {
                    partial void PartialM(int v) { }
                    void M1()
                    {
                        PartialM(1);
                    }
                }
            }
                    </Document>
                </Project>
            </Workspace>
            """;
        await TestInRegularAndScriptAsync(code, fix0);
    }
 
    [Fact]
    public async Task TestInvocation_Cascading_ExtendedPartialMethods()
    {
        var code =
            """
            <Workspace>
                <Project Language="C#" AssemblyName="Assembly1" CommonReferences="true">
                    <Document>
            namespace N1
            {
                partial class C1
                {
                    public partial void PartialM();
                }
            }
                    </Document>
                    <Document>
            namespace N1
            {
                partial class C1
                {
                    public partial void PartialM() { }
                    void M1()
                    {
                        [|PartialM|](1);
                    }
                }
            }
                    </Document>
                </Project>
            </Workspace>
            """;
        var fix0 =
            """
            <Workspace>
                <Project Language="C#" AssemblyName="Assembly1" CommonReferences="true">
                    <Document>
            namespace N1
            {
                partial class C1
                {
                    public partial void PartialM(int v);
                }
            }
                    </Document>
                    <Document>
            namespace N1
            {
                partial class C1
                {
                    public partial void PartialM(int v) { }
                    void M1()
                    {
                        PartialM(1);
                    }
                }
            }
                    </Document>
                </Project>
            </Workspace>
            """;
        await TestInRegularAndScriptAsync(code, fix0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Cascading_PartialMethodsInSameDocument()
    {
        var code =
            """
            namespace N1
            {
                partial class C1
                {
                    partial void PartialM();
                }
                partial class C1
                {
                    partial void PartialM() { }
                    void M1()
                    {
                        [|PartialM|](1);
                    }
                }
            }
            """;
        var fix0 =
            """
            namespace N1
            {
                partial class C1
                {
                    partial void PartialM(int v);
                }
                partial class C1
                {
                    partial void PartialM(int v) { }
                    void M1()
                    {
                        PartialM(1);
                    }
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Cascading_BaseNotInSource()
    {
        // error CS1501: No overload for method 'M' takes 1 arguments
        var code =
            """
            <Workspace>
                <Project Language="C#" CommonReferences="true">
                    <MetadataReferenceFromSource Language="C#" CommonReferences="true">
                        <Document FilePath="ReferencedDocument">
            namespace N
            {
                public class BaseClass
                {
                    public virtual void M() { }
                }
            }
                        </Document>
                    </MetadataReferenceFromSource>
                    <Document FilePath="TestDocument">
            namespace N
            {
                public class Derived: BaseClass
                {
                    public void M2()
                    {
                        [|M|](1);
                    }
                }
            }
                    </Document>
                </Project>
            </Workspace>
            """;
        await TestMissingAsync(code);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Cascading_RootNotInSource()
    {
        // error CS1501: No overload for method 'M' takes 1 arguments
        var code =
            """
            <Workspace>
                <Project Language="C#" CommonReferences="true">
                    <MetadataReferenceFromSource Language="C#" CommonReferences="true">
                        <Document FilePath="ReferencedDocument">namespace N
            {
                public class BaseClass
                {
                    public virtual void M() { }
                }
            }</Document>
                    </MetadataReferenceFromSource>
                    <Document FilePath="TestDocument">namespace N
            {
                public class Derived: BaseClass
                {
                    public override void M() { }
                }
                public class DerivedDerived: Derived
                {
                    public void M2()
                    {
                        [|M|](1);
                    }
                }
            }</Document>
                </Project>
            </Workspace>
            """;
        var fixedDocumentWithoutConflictAnnotation = """
            namespace N
            {
                public class Derived: BaseClass
                {
                    public override void M(int v) { }
                }
                public class DerivedDerived: Derived
                {
                    public void M2()
                    {
                        M(1);
                    }
                }
            }
            """;
        var fixedDocumentWithConflictAnnotation = """
            namespace N
            {
                public class Derived: BaseClass
                {
                    public override void M({|Conflict:int v|}) { }
                }
                public class DerivedDerived: Derived
                {
                    public void M2()
                    {
                        M(1);
                    }
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fixedDocumentWithoutConflictAnnotation, index: 0);
        await TestInRegularAndScriptAsync(code, fixedDocumentWithConflictAnnotation, index: 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Cascading_ManyReferencesInManyProjects()
    {
        // error CS1501: No overload for method 'M' takes 1 arguments
        var code =
            """
            <Workspace>
                <Project Language="C#" CommonReferences="true" AssemblyName="A1">
                    <Document FilePath="ReferencedDocument">
            namespace N
            {
                public class BaseClass
                {
                    public virtual void M() { }
                }
            }
                    </Document>
                </Project>
                <Project Language="C#" CommonReferences="true"  AssemblyName="A2">
                    <ProjectReference>A1</ProjectReference>
                    <Document>
            namespace N
            {
                public class Derived1: BaseClass
                {
                    public override void M() { }
                }
            }
                    </Document>
                </Project>
                <Project Language="C#" CommonReferences="true"  AssemblyName="A3">
                    <ProjectReference>A1</ProjectReference>
                    <Document>
            namespace N
            {
                public class Derived2: BaseClass
                {
                    public override void M() { }
                }
            }
                    </Document>
                </Project>
                <Project Language="C#" CommonReferences="true"  AssemblyName="A4">
                    <ProjectReference>A3</ProjectReference>
                    <Document>
            namespace N
            {
                public class T
                {
                    public void Test() { 
                        new Derived2().[|M|](1);
                    }
                }
            }
                    </Document>
                </Project>
            </Workspace>
            """;
        var fix_All =
            """
            <Workspace>
                <Project Language="C#" CommonReferences="true" AssemblyName="A1">
                    <Document FilePath="ReferencedDocument">
            namespace N
            {
                public class BaseClass
                {
                    public virtual void M(int v) { }
                }
            }
                    </Document>
                </Project>
                <Project Language="C#" CommonReferences="true"  AssemblyName="A2">
                    <ProjectReference>A1</ProjectReference>
                    <Document>
            namespace N
            {
                public class Derived1: BaseClass
                {
                    public override void M(int v) { }
                }
            }
                    </Document>
                </Project>
                <Project Language="C#" CommonReferences="true"  AssemblyName="A3">
                    <ProjectReference>A1</ProjectReference>
                    <Document>
            namespace N
            {
                public class Derived2: BaseClass
                {
                    public override void M(int v) { }
                }
            }
                    </Document>
                </Project>
                <Project Language="C#" CommonReferences="true"  AssemblyName="A4">
                    <ProjectReference>A3</ProjectReference>
                    <Document>
            namespace N
            {
                public class T
                {
                    public void Test() { 
                        new Derived2().M(1);
                    }
                }
            }
                    </Document>
                </Project>
            </Workspace>
            """;
        await TestInRegularAndScriptAsync(code, fix_All, index: 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Cascading_OfferFixCascadingForImplicitInterface()
    {
        // error CS1501: No overload for method 'M1' takes 1 arguments
        var code =
            """
            interface I1
            {
                void M1();
            }
            class C: I1
            {
                public void M1() { }
                void MTest() 
                {
                    [|M1|](1);
                }
            }
            """;
        var fix_DeclarationOnly =
            """
            interface I1
            {
                void M1();
            }
            class C: I1
            {
                public void M1(int v) { }
                void MTest() 
                {
                    M1(1);
                }
            }
            """;
        var fix_All =
            """
            interface I1
            {
                void M1(int v);
            }
            class C: I1
            {
                public void M1(int v) { }
                void MTest() 
                {
                    M1(1);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix_DeclarationOnly, index: 0);
        await TestInRegularAndScriptAsync(code, fix_All, index: 1);
    }
 
#if !CODE_STYLE
 
    // CodeStyle layer does not support cross language application of fixes.
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Cascading_CrossLanguage()
    {
        var code =
            """
            <Workspace>
                <Project Language="Visual Basic" CommonReferences="true" AssemblyName="VB1">
                    <Document FilePath="ReferencedDocument">
            Namespace N
                Public Class BaseClass
                    Public Overridable Sub M()
                    End Sub
                End Class
            End Namespace
                    </Document>
                </Project>
                <Project Language="C#" CommonReferences="true"  AssemblyName="A2">
                    <ProjectReference>VB1</ProjectReference>
                    <Document>
            namespace N
            {
                public class Derived: BaseClass
                {
                    public override void M() { }
                }
                public class T
                {
                    public void Test() { 
                        new Derived().[|M|](1);
                    }
                }
            }
                    </Document>
                </Project>
            </Workspace>
            """;
        var fix =
            """
            <Workspace>
                <Project Language="Visual Basic" CommonReferences="true" AssemblyName="VB1">
                    <Document FilePath="ReferencedDocument">
            Namespace N
                Public Class BaseClass
                    Public Overridable Sub M(v As Integer)
                    End Sub
                End Class
            End Namespace
                    </Document>
                </Project>
                <Project Language="C#" CommonReferences="true"  AssemblyName="A2">
                    <ProjectReference>VB1</ProjectReference>
                    <Document>
            namespace N
            {
                public class Derived: BaseClass
                {
                    public override void M(int v) { }
                }
                public class T
                {
                    public void Test() { 
                        new Derived().M(1);
                    }
                }
            }
                    </Document>
                </Project>
            </Workspace>
            """;
        await TestInRegularAndScriptAsync(code, fix, index: 1);
    }
 
#endif
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_Positional_MoreThanOneArgumentToMuch()
    {
        var code =
            """
            class C
            {
                void M() { }
                void Test()
                {
                    [|M|](1, 2, 3, 4);
                }
            }
            """;
        var fix0 =
            """
            class C
            {
                void M(int v) { }
                void Test()
                {
                    M(1, 2, 3, 4);
                }
            }
            """;
        await TestActionCountAsync(code, 1);
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_Positional_WithOptionalParam()
    {
        // error CS1501: No overload for method 'M' takes 2 arguments
        var code =
            """
            class C
            {
                void M(int i = 1) { }
                void Test()
                {
                    [|M|](1, 2);
                }
            }
            """;
        var fix0 =
            """
            class C
            {
                void M(int i = 1, int v = 0) { }
                void Test()
                {
                    M(1, 2);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_Named_WithOptionalParam()
    {
        // error CS1739: The best overload for 'M' does not have a parameter named 'i3'
        var code =
            """
            class C
            {
                void M(int i1, int i2 = 1) { }
                void Test()
                {
                    M(1, i2: 2, [|i3|]: 3);
                }
            }
            """;
        var fix0 =
            """
            class C
            {
                void M(int i1, int i2 = 1, int i3 = 0) { }
                void Test()
                {
                    M(1, i2: 2, i3: 3);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_Positional_WithParams()
    {
        // error CS1503: Argument 1: cannot convert from 'string' to 'int'
        var code =
            """
            class C
            {
                void M(params int[] ints) { }
                void Test()
                {
                    M([|"text"|]);
                }
            }
            """;
        var fix0 =
            """
            class C
            {
                void M(string v, params int[] ints) { }
                void Test()
                {
                    M("text");
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_Named_WithTypemissmatch()
    {
        // error CS1503: Argument 1: cannot convert from 'string' to 'int'
        var code =
            """
            class C
            {
                void M(int i) { }
                void Test()
                {
                    M(i: [|"text"|]);
                }
            }
            """;
        await TestMissingInRegularAndScriptAsync(code);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_NamedAndPositional1()
    {
        // error CS1739: The best overload for 'M' does not have a parameter named 'i2'
        var code =
            """
            class C
            {
                void M(int i1, string s) { }
                void Test()
                {
                    M(1, s: "text", [|i2|]: 0);
                }
            }
            """;
        var fix0 =
            """
            class C
            {
                void M(int i1, string s, int i2) { }
                void Test()
                {
                    M(1, s: "text", i2: 0);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_NamedAndPositional2()
    {
        // CS1744 is not yet a supported diagnostic (just declaring the diagnostic as supported does not work)
        // error CS1744: Named argument 's' specifies a parameter for which a positional argument has already been given
        var code =
            """
            class C
            {
                void M(string s) { }
                void Test()
                {
                    M(1, [|s|]: "text");
                }
            }
            """;
        await TestMissingInRegularAndScriptAsync(code);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_Incomplete_1()
    {
        // error CS1501: No overload for method 'M' takes 1 arguments
        var code =
            """
            class C
            {
                void M() { }
                void Test()
                {
                    [|M|](1
                }
            }
            """;
        var fix0 =
            """
            class C
            {
                void M(int v) { }
                void Test()
                {
                    M(1
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_Incomplete_2()
    {
        // error CS1503: Argument 1: cannot convert from 'string' to 'int'
        var code =
            """
            class C
            {
                void M(int v) { }
                void Test()
                {
                    [|M|]("text", 1
            """;
        var fix0 =
            """
            class C
            {
                void M(string v1, int v) { }
                void Test()
                {
                    M("text", 1
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_RefParameter()
    {
        // error CS1501: No overload for method 'M' takes 1 arguments            
        var code =
            """
            class C
            {
                void M() { }
                void Test()
                {
                    int i = 0;
                    [|M|](ref i);
                }
            }
            """;
        var fix0 =
            """
            class C
            {
                void M(ref int i) { }
                void Test()
                {
                    int i = 0;
                    M(ref i);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_OutParameter_WithTypeDeclarationOutsideArgument()
    {
        // error CS1501: No overload for method 'M' takes 1 arguments            
        var code =
            """
            class C
            {
                void M() { }
                void Test()
                {
                    int i = 0;
                    [|M|](out i);
                }
            }
            """;
        var fix0 =
            """
            class C
            {
                void M(out int i) { }
                void Test()
                {
                    int i = 0;
                    M(out i);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_OutParameter_WithTypeDeclarationInArgument()
    {
        // error CS1501: No overload for method 'M' takes 1 arguments            
        var code =
            """
            class C
            {
                void M() { }
                void Test()
                {
                    [|M|](out int i);
                }
            }
            """;
        var fix0 =
            """
            class C
            {
                void M(out int i) { }
                void Test()
                {
                    M(out int i);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_InvocationStyles_OutParameter_WithVarTypeDeclarationInArgument()
    {
        // error CS1501: No overload for method 'M' takes 1 arguments            
        var code =
            """
            class C
            {
                void M() { }
                void Test()
                {
                    [|M|](out var i);
                }
            }
            """;
        var fix0 =
            """
            class C
            {
                void M(out object i) { }
                void Test()
                {
                    M(out var i);
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21446")]
    public async Task TestInvocation_Indexer_NotSupported()
    {
        // Could be fixed by allowing ElementAccessExpression next to InvocationExpression
        // in AbstractAddParameterCodeFixProvider.RegisterCodeFixesAsync.
        // error CS1501: No overload for method 'this' takes 2 arguments
        var code =
            """
            public class C {
                public int this[int i] 
                { 
                    get => 1; 
                    set {} 
                }
 
                public void Test() {
                    var i = [|this[0,0]|];
                }
            }
            """;
        await TestMissingAsync(code);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/29061")]
    public async Task TestThis_DoNotOfferToFixTheConstructorWithTheDiagnosticOnIt()
    {
        // error CS1729: 'C' does not contain a constructor that takes 1 arguments
        var code =
            """
            public class C {
 
                public C(): [|this|](1)
                { }
            }
            """;
        await TestMissingAsync(code);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/29061")]
    public async Task TestThis_Fix_IfACandidateIsAvailable()
    {
        // error CS1729: 'C' does not contain a constructor that takes 2 arguments
        var code =
            """
            class C 
            {
                public C(int i) { }
 
                public C(): [|this|](1, 1)
                { }
            }
            """;
        var fix0 =
            """
            class C 
            {
                public C(int i, int v) { }
 
                public C(): this(1, 1)
                { }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
        await TestActionCountAsync(code, 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/29061")]
    public async Task TestBase_Fix_IfACandidateIsAvailable()
    {
        // error CS1729: 'B' does not contain a constructor that takes 1 arguments
        var code =
            """
            public class B
            {
                B() { }
            }
            public class C : B
            {
                public C(int i) : [|base|](i) { }
            }
            """;
        var fix0 =
            """
            public class B
            {
                B(int i) { }
            }
            public class C : B
            {
                public C(int i) : base(i) { }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
        await TestActionCountAsync(code, 1);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/29753")]
    public async Task LocalFunction_AddParameterToLocalFunctionWithOneParameter()
    {
        // CS1501 No overload for method takes 2 arguments
        var code =
            """
            class Rsrp
            {
              public void M()
              {
                [|Local|]("ignore this", true);
                void Local(string whatever)
                {
 
                }
              }
            }
            """;
        var fix0 =
            """
            class Rsrp
            {
              public void M()
              {
                Local("ignore this", true);
                void Local(string whatever, bool v)
                {
 
                }
              }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/29752")]
    public async Task LocalFunction_AddNamedParameterToLocalFunctionWithOneParameter()
    {
        // CS1739: The best overload for 'Local' does not have a parameter named 'mynewparameter'
        var code =
            """
            class Rsrp
            {
                public void M()
                {
                    Local("ignore this", [|mynewparameter|]: true);
                    void Local(string whatever)
                    {
 
                    }
                }
            }
            """;
        var fix0 =
            """
            class Rsrp
            {
                public void M()
                {
                    Local("ignore this", mynewparameter: true);
                    void Local(string whatever, bool mynewparameter)
                    {
 
                    }
                }
            }
            """;
        await TestInRegularAndScriptAsync(code, fix0, index: 0);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/39270")]
    public async Task TestWithArgThatHasImplicitConversionToParamType1()
    {
        await TestInRegularAndScriptAsync(
            """
            class BaseClass { }
 
            class MyClass : BaseClass
            {
                void TestFunc()
                {
                    MyClass param1 = new MyClass();
                    int newparam = 1;
 
                    [|MyFunc|](param1, newparam);
                }
 
                void MyFunc(BaseClass param1) { }
            }
            """,
            """
            class BaseClass { }
 
            class MyClass : BaseClass
            {
                void TestFunc()
                {
                    MyClass param1 = new MyClass();
                    int newparam = 1;
 
                    MyFunc(param1, newparam);
                }
 
                void MyFunc(BaseClass param1, int newparam) { }
            }
            """);
    }
 
    [Fact]
    public async Task TestOnExtensionGetEnumerator()
    {
        var code =
            """
            using System.Collections.Generic;
            namespace N {
            static class Extensions
            {
                public static IEnumerator<int> GetEnumerator(this object o)
                {
                }
            }
            class C1
            {
                void M1()
                {
                    new object().[|GetEnumerator|](1);
                    foreach (var a in new object());
                }
            }}
            """;
        var fix =
            """
            using System.Collections.Generic;
            namespace N {
            static class Extensions
            {
                public static IEnumerator<int> GetEnumerator(this object o, int v)
                {
                }
            }
            class C1
            {
                void M1()
                {
                    new object().GetEnumerator(1);
                    foreach (var a in new object());
                }
            }}
            """;
        await TestInRegularAndScriptAsync(code, fix);
    }
 
    [Fact]
    public async Task TestOnExtensionGetAsyncEnumerator()
    {
        var code =
            """
            using System.Collections.Generic;
            using System.Threading.Tasks;
            namespace N {
            static class Extensions
            {
                public static IAsyncEnumerator<int> GetAsyncEnumerator(this object o)
                {
                }
            }
            class C1
            {
                async Task M1()
                {
                    new object().[|GetAsyncEnumerator|](1);
                    await foreach (var a in new object());
                }
            }}
            """ + IAsyncEnumerable;
        var fix =
            """
            using System.Collections.Generic;
            using System.Threading.Tasks;
            namespace N {
            static class Extensions
            {
                public static IAsyncEnumerator<int> GetAsyncEnumerator(this object o, int v)
                {
                }
            }
            class C1
            {
                async Task M1()
                {
                    new object().GetAsyncEnumerator(1);
                    await foreach (var a in new object());
                }
            }}
            """ + IAsyncEnumerable;
        await TestInRegularAndScriptAsync(code, fix);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/44271")]
    public async Task TopLevelStatement()
    {
        await TestInRegularAndScriptAsync("""
            [|local|](1, 2, 3);
 
            void local(int x, int y)
            {
            }
            """,
            """
            [|local|](1, 2, 3);
 
            void local(int x, int y, int v)
            {
            }
            """, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp9));
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/44271")]
    public async Task TopLevelStatement_Nested()
    {
        await TestInRegularAndScriptAsync("""
            void outer()
            {
                [|local|](1, 2, 3);
 
                void local(int x, int y)
                {
                }
            }
            """,
            """
            void outer()
            {
                local(1, 2, 3);
 
                void local(int x, int y, int v)
                {
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/42559")]
    public async Task TestAddParameter_ImplicitObjectCreation()
    {
        await TestInRegularAndScriptAsync("""
            class C
            {
                C(int i) { }
 
                void M()
                {
                   C c = [||]new(1, 2);
                }
            }
            """,
            """
            class C
            {
                C(int i, int v) { }
 
                void M()
                {
                   C c = new(1, 2);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/48042")]
    public async Task TestNamedArgOnExtensionMethod()
    {
        await TestInRegularAndScriptAsync(
            """
            namespace r
            {
                static class AbcExtensions
                {
                    public static Abc Act(this Abc state, bool p = true) => state;
                }
                class Abc {
                    void Test()
                        => new Abc().Act([|param3|]: 123);
                }
            }
            """,
            """
            namespace r
            {
                static class AbcExtensions
                {
                    public static Abc Act(this Abc state, bool p = true, int param3 = 0) => state;
                }
                class Abc {
                    void Test()
                        => new Abc().Act(param3: 123);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/54408")]
    public async Task TestPositionalRecord()
    {
        await TestInRegularAndScriptAsync("""
            var b = "B";
            var r = [|new R(1, b)|];
 
            record R(int A);
 
            namespace System.Runtime.CompilerServices
            {
                public static class IsExternalInit { }
            }
            """, """
            var b = "B";
            var r = new R(1, b);
 
            record R(int A, string b);
 
            namespace System.Runtime.CompilerServices
            {
                public static class IsExternalInit { }
            }
            """, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp9));
    }
 
    [Fact]
    public async Task Test_PrimaryConstructor_Class()
    {
        await TestInRegularAndScriptAsync("""
            var b = "B";
            var r = [|new R(1, b)|];
 
            class R(int A);
            """, """
            var b = "B";
            var r = new R(1, b);
 
            class R(int A, string b);
            """, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp12));
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/54408")]
    public async Task TestPositionalRecordStruct()
    {
        await TestInRegularAndScriptAsync("""
            var b = "B";
            var r = [|new R(1, b)|];
 
            record struct R(int A);
 
            namespace System.Runtime.CompilerServices
            {
                public static class IsExternalInit { }
            }
            """, """
            var b = "B";
            var r = new R(1, b);
 
            record struct R(int A, string b);
 
            namespace System.Runtime.CompilerServices
            {
                public static class IsExternalInit { }
            }
            """, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp9));
    }
 
    [Fact]
    public async Task Test_PrimaryConstructor_Struct()
    {
        await TestInRegularAndScriptAsync("""
            var b = "B";
            var r = [|new R(1, b)|];
 
            struct R(int A);
            """, """
            var b = "B";
            var r = new R(1, b);
 
            struct R(int A, string b);
            """, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp12));
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/56952")]
    public async Task TestRecordsNamingConventions()
    {
        await TestInRegularAndScript1Async("""
            [|new Test("repro")|];
 
            record Test();
 
            """, """
            new Test("repro");
 
            record Test(string V);
 
            """);
    }
 
    [Fact]
    public async Task TestNamingConventions_PrimaryConstructor_Class()
    {
        await TestInRegularAndScript1Async("""
            [|new Test("repro")|];
 
            class Test();
            """, """
            new Test("repro");
 
            class Test(string v);
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/56952")]
    public async Task TestRecordsNamingConventions_RecordStruct()
    {
        await TestInRegularAndScript1Async("""
            [|new Test("repro")|];
 
            record struct Test();
 
            """, """
            new Test("repro");
 
            record struct Test(string V);
 
            """);
    }
 
    [Fact]
    public async Task TestNamingConventions_PrimaryConstructor_Struct()
    {
        await TestInRegularAndScript1Async("""
            [|new Test("repro")|];
 
            struct Test();
            """, """
            new Test("repro");
 
            struct Test(string v);
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/61715")]
    public async Task TestMethodGroup1()
    {
        await TestInRegularAndScript1Async("""
            public class Example
            {
                public void Add(int x)
                {
                }
 
                public void DoSomething()
                {
                }
 
                public void Main()
                {
                    [|DoSomething|](Add);
                }
            }
            """, """
            public class Example
            {
                public void Add(int x)
                {
                }
 
                public void DoSomething(System.Action<int> add)
                {
                }
 
                public void Main()
                {
                    DoSomething(Add);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/61715")]
    public async Task TestMethodGroup2()
    {
        await TestInRegularAndScript1Async("""
            public class Example
            {
                public void Add(int x, string y)
                {
                }
 
                public void DoSomething()
                {
                }
 
                public void Main()
                {
                    [|DoSomething|](Add);
                }
            }
            """, """
            public class Example
            {
                public void Add(int x, string y)
                {
                }
 
                public void DoSomething(System.Action<int, string> add)
                {
                }
 
                public void Main()
                {
                    DoSomething(Add);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/61715")]
    public async Task TestMethodGroup3()
    {
        await TestInRegularAndScript1Async("""
            public class Example
            {
                public int Add(int x, string y)
                {
                    return 0;
                }
 
                public void DoSomething()
                {
                }
 
                public void Main()
                {
                    [|DoSomething|](Add);
                }
            }
            """, """
            public class Example
            {
                public int Add(int x, string y)
                {
                    return 0;
                }
 
                public void DoSomething(System.Func<int, string, int> add)
                {
                }
 
                public void Main()
                {
                    DoSomething(Add);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/71428")]
    public async Task TestAddConstructorParameterWithExistingField_BlockInitialize()
    {
        await TestInRegularAndScript1Async(
            """
            class C
            {
                private readonly string s;
                private readonly string t;
                private readonly int i;
 
                public C(string s, int i)
                {
                    this.s = s;
                    this.i = i;
                }
            }
 
            class D
            {
                void M(string t)
                {
                    new [|C|]("", t, 0);
                }
            }
            """,
            """
            class C
            {
                private readonly string s;
                private readonly string t;
                private readonly int i;
 
                public C(string s, string t, int i)
                {
                    this.s = s;
                    this.t = t;
                    this.i = i;
                }
            }
            
            class D
            {
                void M(string t)
                {
                    new C("", t, 0);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/71428")]
    public async Task TestAddConstructorParameterWithExistingField_ExpressionBodyInitialize()
    {
        await TestInRegularAndScript1Async(
            """
            class C
            {
                private readonly string s;
                private readonly string t;
 
                public C(string s)
                    => this.s = s;
            }
 
            class D
            {
                void M(string t)
                {
                    new [|C|]("", t);
                }
            }
            """,
            """
            class C
            {
                private readonly string s;
                private readonly string t;
 
                public C(string s, string t)
                {
                    this.s = s;
                    this.t = t;
                }
            }
            
            class D
            {
                void M(string t)
                {
                    new C("", t);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/71428")]
    public async Task TestAddConstructorParameterWithExistingField_TupleInitialize()
    {
        await TestInRegularAndScript1Async(
            """
            class C
            {
                private readonly string s;
                private readonly string t;
                private readonly string i;
 
                public C(string s, string t)
                {
                    (this.s, this.t) = (s, t);
                }
            }
 
            class D
            {
                void M(string i)
                {
                    new [|C|]("", "", i);
                }
            }
            """,
            """
            class C
            {
                private readonly string s;
                private readonly string t;
                private readonly string i;
 
                public C(string s, string t, string i)
                {
                    (this.s, this.t, this.i) = (s, t, i);
                }
            }
            
            class D
            {
                void M(string i)
                {
                    new C("", "", i);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/71428")]
    public async Task TestAddConstructorParameterWithExistingField_UnderscoreName()
    {
        await TestInRegularAndScript1Async(
            """
            class C
            {
                private readonly string _s;
 
                public C()
                {
                }
            }
 
            class D
            {
                void M(string s)
                {
                    new [|C|](s);
                }
            }
            """,
            """
            class C
            {
                private readonly string _s;
 
                public C(string s)
                {
                    _s = s;
                }
            }
            
            class D
            {
                void M(string s)
                {
                    new C(s);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/71428")]
    public async Task TestAddConstructorParameterWithExistingField_PrimaryConstructor()
    {
        await TestInRegularAndScript1Async(
            """
            class C()
            {
                private readonly string _name;
            }
 
            class D
            {
                void M(string name)
                {
                    new [|C|](name);
                }
            }
            """,
            """
            class C(string name)
            {
                private readonly string _name = name;
            }
            
            class D
            {
                void M(string name)
                {
                    new C(name);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/71428")]
    public async Task TestAddConstructorParameterWithExistingProperty_PrimaryConstructor()
    {
        await TestInRegularAndScript1Async(
            """
            class C()
            {
                private string Name { get; }
            }
 
            class D
            {
                void M(string name)
                {
                    new [|C|](name);
                }
            }
            """,
            """
            class C(string name)
            {
                private string Name { get; } = name;
            }
            
            class D
            {
                void M(string name)
                {
                    new C(name);
                }
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/71428")]
    public async Task TestAddConstructorParameterWithExistingThrowingProperty_PrimaryConstructor()
    {
        await TestInRegularAndScript1Async(
            """
            using System;
 
            class C()
            {
                private string Name => throw new NotImplementedException();
            }
 
            class D
            {
                void M(string name)
                {
                    new [|C|](name);
                }
            }
            """,
            """
            using System;
 
            class C(string name)
            {
                private string Name { get; } = name;
            }
            
            class D
            {
                void M(string name)
                {
                    new C(name);
                }
            }
            """);
    }
}