File: Recommendations\ReadOnlyKeywordRecommenderTests.cs
Web Access
Project: src\src\EditorFeatures\CSharpTest2\Microsoft.CodeAnalysis.CSharp.EditorFeatures2.UnitTests.csproj (Microsoft.CodeAnalysis.CSharp.EditorFeatures2.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.Threading.Tasks;
using Microsoft.CodeAnalysis.Test.Utilities;
using Roslyn.Test.Utilities;
using Xunit;
 
namespace Microsoft.CodeAnalysis.Editor.CSharp.UnitTests.Recommendations;
 
[Trait(Traits.Feature, Traits.Features.KeywordRecommending)]
public sealed class ReadOnlyKeywordRecommenderTests : KeywordRecommenderTests
{
    [Fact]
    public async Task TestAtRoot()
    {
        await VerifyKeywordAsync(
@"$$");
    }
 
    [Fact]
    public async Task TestAfterClass()
    {
        await VerifyKeywordAsync(
            """
            class C { }
            $$
            """);
    }
 
    [Fact]
    public async Task TestAfterGlobalStatement()
    {
        await VerifyKeywordAsync(
            """
            System.Console.WriteLine();
            $$
            """);
    }
 
    [Fact]
    public async Task TestAfterGlobalVariableDeclaration()
    {
        await VerifyKeywordAsync(
            """
            int i = 0;
            $$
            """);
    }
 
    [Fact]
    public async Task TestNotInUsingAlias()
    {
        await VerifyAbsenceAsync(
@"using Goo = $$");
    }
 
    [Fact]
    public async Task TestNotInGlobalUsingAlias()
    {
        await VerifyAbsenceAsync(
@"global using Goo = $$");
    }
 
    [Fact]
    public async Task TestNotInEmptyStatement()
    {
        await VerifyAbsenceAsync(AddInsideMethod(
@"$$"));
    }
 
    [Fact]
    public async Task TestAfterExtern()
    {
        await VerifyKeywordAsync("""
            extern alias Goo;
            $$
            """);
    }
 
    [Fact]
    public async Task TestAfterUsing()
    {
        await VerifyKeywordAsync("""
            using Goo;
            $$
            """);
    }
 
    [Fact]
    public async Task TestAfterGlobalUsing()
    {
        await VerifyKeywordAsync("""
            global using Goo;
            $$
            """);
    }
 
    [Fact]
    public async Task TestAfterNamespace()
    {
        await VerifyKeywordAsync("""
            namespace N {}
            $$
            """);
    }
 
    [Fact]
    public async Task TestAfterFileScopedNamespace()
    {
        await VerifyKeywordAsync(
            """
            namespace N;
            $$
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/66319")]
    public async Task TestFileKeywordInsideNamespace()
    {
        await VerifyKeywordAsync(
            """
            namespace N {
            file $$
            }
            """);
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/66319")]
    public async Task TestFileKeywordInsideNamespaceBeforeClass()
    {
        await VerifyKeywordAsync(
            """
            namespace N {
            file $$
            class C {}
            }
            """);
    }
 
    [Fact]
    public async Task TestAfterTypeDeclaration()
    {
        await VerifyKeywordAsync("""
            class C {}
            $$
            """);
    }
 
    [Fact]
    public async Task TestAfterDelegateDeclaration()
    {
        await VerifyKeywordAsync("""
            delegate void Goo();
            $$
            """);
    }
 
    [Fact]
    public async Task TestAfterMethod()
    {
        await VerifyKeywordAsync(
            """
            class C {
              void Goo() {}
              $$
            """);
    }
 
    [Fact]
    public async Task TestAfterField()
    {
        await VerifyKeywordAsync(
            """
            class C {
              int i;
              $$
            """);
    }
 
    [Fact]
    public async Task TestAfterProperty()
    {
        await VerifyKeywordAsync(
            """
            class C {
              int i { get; }
              $$
            """);
    }
 
    [Fact]
    public async Task TestNotBeforeUsing()
    {
        await VerifyAbsenceAsync(SourceCodeKind.Regular,
            """
            $$
            using Goo;
            """);
    }
 
    [Fact(Skip = "https://github.com/dotnet/roslyn/issues/9880"), Trait(Traits.Feature, Traits.Features.KeywordRecommending)]
    public async Task TestNotBeforeUsing_Interactive()
    {
        await VerifyAbsenceAsync(SourceCodeKind.Script,
            """
            $$
            using Goo;
            """);
    }
 
    [Fact]
    public async Task TestNotBeforeGlobalUsing()
    {
        await VerifyAbsenceAsync(SourceCodeKind.Regular,
            """
            $$
            global using Goo;
            """);
    }
 
    [Fact(Skip = "https://github.com/dotnet/roslyn/issues/9880"), Trait(Traits.Feature, Traits.Features.KeywordRecommending)]
    public async Task TestNotBeforeGlobalUsing_Interactive()
    {
        await VerifyAbsenceAsync(SourceCodeKind.Script,
            """
            $$
            global using Goo;
            """);
    }
 
    [Fact]
    public async Task TestAfterAssemblyAttribute()
    {
        await VerifyKeywordAsync("""
            [assembly: goo]
            $$
            """);
    }
 
    [Fact]
    public async Task TestAfterRootAttribute()
    {
        await VerifyKeywordAsync("""
            [goo]
            $$
            """);
    }
 
    [Fact]
    public async Task TestAfterNestedAttribute()
    {
        await VerifyKeywordAsync(
            """
            class C {
              [goo]
              $$
            """);
    }
 
    [Fact]
    public async Task TestInsideStruct()
    {
        await VerifyKeywordAsync(
            """
            struct S {
               $$
            """);
    }
 
    [Fact]
    public async Task TestInsideInterface()
    {
        await VerifyKeywordAsync("""
            interface I {
               $$
            """);
    }
 
    [Fact]
    public async Task TestNotInsideEnum()
    {
        await VerifyAbsenceAsync("""
            enum E {
               $$
            """);
    }
 
    [Fact]
    public async Task TestInsideClass()
    {
        await VerifyKeywordAsync(
            """
            class C {
               $$
            """);
    }
 
    [Fact]
    public async Task TestNotAfterPartial()
        => await VerifyAbsenceAsync(@"partial $$");
 
    [Fact]
    public async Task TestAfterAbstract()
        => await VerifyKeywordAsync(@"abstract $$");
 
    [Fact]
    public async Task TestAfterInternal()
        => await VerifyKeywordAsync(@"internal $$");
 
    [Fact]
    public async Task TestAfterNestedInternal()
    {
        await VerifyKeywordAsync(
            """
            class C {
                internal $$
            """);
    }
 
    [Fact]
    public async Task TestAfterPublic()
        => await VerifyKeywordAsync(@"public $$");
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/66319")]
    public async Task TestAfterFile()
        => await VerifyKeywordAsync(SourceCodeKind.Regular, @"file $$");
 
    [Fact]
    public async Task TestAfterNestedPublic()
    {
        await VerifyKeywordAsync(
            """
            class C {
                public $$
            """);
    }
 
    [Fact]
    public async Task TestAfterPrivate()
    {
        await VerifyKeywordAsync(@"private $$");
    }
 
    [Fact]
    public async Task TestAfterNestedPrivate()
    {
        await VerifyKeywordAsync(
            """
            class C {
                private $$
            """);
    }
 
    [Fact]
    public async Task TestAfterProtected()
    {
        await VerifyKeywordAsync(
@"protected $$");
    }
 
    [Fact]
    public async Task TestAfterNestedProtected()
    {
        await VerifyKeywordAsync(
            """
            class C {
                protected $$
            """);
    }
 
    [Fact]
    public async Task TestAfterSealed()
        => await VerifyKeywordAsync(@"sealed $$");
 
    [Fact]
    public async Task TestAfterNestedSealed()
    {
        await VerifyKeywordAsync(
            """
            class C {
                sealed $$
            """);
    }
 
    [Fact]
    public async Task TestAfterStatic()
        => await VerifyKeywordAsync(@"static $$");
 
    [Fact]
    public async Task TestAfterNestedStatic()
    {
        await VerifyKeywordAsync(
            """
            class C {
                static $$
            """);
    }
 
    [Fact]
    public async Task TestAfterStaticPublic()
        => await VerifyKeywordAsync(@"static public $$");
 
    [Fact]
    public async Task TestAfterNestedStaticPublic()
    {
        await VerifyKeywordAsync(
            """
            class C {
                static public $$
            """);
    }
 
    [Fact]
    public async Task TestNotAfterDelegate()
        => await VerifyAbsenceAsync(@"delegate $$");
 
    [Fact]
    public async Task TestNotAfterEvent()
    {
        await VerifyAbsenceAsync(
            """
            class C {
                event $$
            """);
    }
 
    [Fact]
    public async Task TestNotAfterConst()
    {
        await VerifyAbsenceAsync(
            """
            class C {
                const $$
            """);
    }
 
    [Fact]
    public async Task TestNotAfterPublicReadOnly()
    {
        await VerifyAbsenceAsync(
            """
            class C {
                public readonly $$
            """);
    }
 
    [Fact]
    public async Task TestNotAfterReadOnlyPartial()
    {
        await VerifyAbsenceAsync(
            """
            class C {
                readonly partial $$
            """);
    }
 
    [Fact]
    public async Task TestNotAfterReadOnly()
    {
        await VerifyAbsenceAsync(
            """
            class C {
                readonly $$
            """);
    }
 
    [Fact]
    public async Task TestNotAfterVolatile()
    {
        await VerifyAbsenceAsync(
            """
            class C {
                volatile $$
            """);
    }
 
    [Fact]
    public async Task TestAfterRef()
        => await VerifyKeywordAsync(@"ref $$");
 
    [Fact]
    public async Task TestInRefStruct()
        => await VerifyKeywordAsync(@"ref $$ struct { }");
 
    [Fact]
    public async Task TestInRefStructBeforeRef()
        => await VerifyKeywordAsync(@"$$ ref struct { }");
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/44423")]
    public async Task TestAfterNew()
        => await VerifyAbsenceAsync(@"new $$");
 
    [Fact]
    public async Task TestAfterNewInClass()
        => await VerifyKeywordAsync(@"class C { new $$ }");
 
    [Fact]
    public async Task TestAfterNestedNew()
    {
        await VerifyKeywordAsync(
            """
            class C {
               new $$
            """);
    }
 
    [Fact]
    public async Task TestNotInMethod()
    {
        await VerifyAbsenceAsync(
            """
            class C {
               void Goo() {
                 $$
            """);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Fact, Trait(Traits.Feature, Traits.Features.Completion)]
    public async Task TestRefReadonlyAsParameterModifierInMethods()
    {
        await VerifyKeywordAsync("""
            class Program
            {
                public static void Test(ref $$ p) { }
            }
            """);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Fact, Trait(Traits.Feature, Traits.Features.Completion)]
    public async Task TestRefReadonlyAsParameterModifierInSecondParameter()
    {
        await VerifyKeywordAsync("""
            class Program
            {
                public static void Test(int p1, ref $$ p2) { }
            }
            """);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Fact, Trait(Traits.Feature, Traits.Features.Completion)]
    public async Task TestRefReadonlyAsParameterModifierInDelegates()
    {
        await VerifyKeywordAsync("""
            public delegate int Delegate(ref $$ int p);
            """);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Theory, Trait(Traits.Feature, Traits.Features.Completion)]
    [CombinatorialData]
    public async Task TestRefReadonlyAsParameterModifierInLocalFunctions(bool topLevelStatement)
    {
        await VerifyKeywordAsync(AddInsideMethod(
@"void localFunc(ref $$ int p) { }", topLevelStatement: topLevelStatement), options: CSharp9ParseOptions);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Fact, Trait(Traits.Feature, Traits.Features.Completion)]
    public async Task TestRefReadonlyAsParameterModifierInLambdaExpressions()
    {
        await VerifyKeywordAsync("""
            public delegate int Delegate(ref int p);
 
            class Program
            {
                public static void Test()
                {
                    Delegate lambda = (ref $$ int p) => p;
                }
            }
            """);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Fact, Trait(Traits.Feature, Traits.Features.Completion)]
    public async Task TestRefReadonlyAsParameterModifierInAnonymousMethods()
    {
        await VerifyKeywordAsync("""
            public delegate int Delegate(ref int p);
 
            class Program
            {
                public static void Test()
                {
                    Delegate anonymousDelegate = delegate (ref $$ int p) { return p; };
                }
            }
            """);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Fact, Trait(Traits.Feature, Traits.Features.Completion)]
    public async Task TestRefReadonlyAsModifierInMethodReturnTypes()
    {
        await VerifyKeywordAsync("""
            class Program
            {
                public ref $$ int Test()
                {
                    return ref x;
                }
            }
            """);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Fact, Trait(Traits.Feature, Traits.Features.Completion)]
    public async Task TestRefReadonlyAsModifierInGlobalMemberDeclaration()
    {
        await VerifyKeywordAsync("""
            public ref $$
            """);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Fact, Trait(Traits.Feature, Traits.Features.Completion)]
    public async Task TestRefReadonlyAsModifierInDelegateReturnType()
    {
        await VerifyKeywordAsync("""
            public delegate ref $$ int Delegate();
 
            class Program
            {
            }
            """);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Fact, Trait(Traits.Feature, Traits.Features.Completion)]
    public async Task TestRefReadonlyAsModifierInMemberDeclaration()
    {
        await VerifyKeywordAsync("""
            class Program
            {
                public ref $$ int Test { get; set; }
            }
            """);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Theory, Trait(Traits.Feature, Traits.Features.Completion)]
    [WorkItem("https://github.com/dotnet/roslyn/issues/25569")]
    [CombinatorialData]
    public async Task TestRefReadonlyInStatementContext(bool topLevelStatement)
    {
        await VerifyKeywordAsync(AddInsideMethod(
@"ref $$", topLevelStatement: topLevelStatement), options: CSharp9ParseOptions);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Theory, Trait(Traits.Feature, Traits.Features.Completion)]
    [CombinatorialData]
    public async Task TestRefReadonlyInLocalDeclaration(bool topLevelStatement)
    {
        await VerifyKeywordAsync(AddInsideMethod(
@"ref $$ int local;", topLevelStatement: topLevelStatement), options: CSharp9ParseOptions);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Theory, Trait(Traits.Feature, Traits.Features.Completion)]
    [CombinatorialData]
    public async Task TestRefReadonlyInLocalFunction(bool topLevelStatement)
    {
        await VerifyKeywordAsync(AddInsideMethod(
@"ref $$ int Function();", topLevelStatement: topLevelStatement), options: CSharp9ParseOptions);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Theory, Trait(Traits.Feature, Traits.Features.Completion)]
    [CombinatorialData]
    public async Task TestRefReadonlyNotInRefExpression(bool topLevelStatement)
    {
        await VerifyAbsenceAsync(AddInsideMethod(
@"ref int x = ref $$", topLevelStatement: topLevelStatement), options: CSharp9ParseOptions,
            // Top level statement with script tested below, so skip it here.
            scriptOptions: topLevelStatement ? CSharp9ParseOptions : null);
    }
 
    [CompilerTrait(CompilerFeature.ReadOnlyReferences)]
    [Fact, Trait(Traits.Feature, Traits.Features.Completion)]
    public async Task TestRefReadonlyInRefExpression_TopLevelStatementScript()
    {
        // Recognized as parameter context, so readonly keyword is suggested.
        await VerifyKeywordAsync(AddInsideMethod(
@"ref int x = ref $$", topLevelStatement: true), options: CSharp9ParseOptions.WithKind(SourceCodeKind.Script));
    }
 
    [Fact]
    public async Task TestInFunctionPointerTypeAfterRef()
    {
        await VerifyKeywordAsync("""
            class C
            {
                delegate*<ref $$
            """);
    }
 
    [Fact]
    public async Task TestNotInFunctionPointerTypeWithoutRef()
    {
        await VerifyAbsenceAsync("""
            class C
            {
                delegate*<$$
            """);
    }
 
    [Theory, Trait(Traits.Feature, Traits.Features.KeywordRecommending)]
    [InlineData("in")]
    [InlineData("out")]
    [InlineData("ref readonly")]
    public async Task TestNotInFunctionPointerTypeAfterOtherRefModifier(string modifier)
    {
        await VerifyAbsenceAsync($$"""
            class C
            {
                delegate*<{{modifier}} $$
            """);
    }
 
    [Fact]
    public async Task TestWithinExtension()
    {
        await VerifyAbsenceAsync(
            """
            static class C
            {
                extension(string s)
                {
                    $$
                }
            }
            """, CSharpNextParseOptions);
    }
}