File: CodeGeneration\CodeGenerationTests.CSharp.cs
Web Access
Project: src\src\EditorFeatures\Test\Microsoft.CodeAnalysis.EditorFeatures.UnitTests.csproj (Microsoft.CodeAnalysis.EditorFeatures.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.
 
#nullable disable
 
using System;
using System.Collections.Immutable;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.CodeGeneration;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.CSharp.CodeStyle;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using Microsoft.CodeAnalysis.Editor.UnitTests.CodeActions;
using Microsoft.CodeAnalysis.Test.Utilities;
using Roslyn.Test.Utilities;
using Roslyn.Utilities;
using Xunit;
using CS = Microsoft.CodeAnalysis.CSharp;
 
namespace Microsoft.CodeAnalysis.Editor.UnitTests.CodeGeneration;
 
using static CSharpSyntaxTokens;
 
[Trait(Traits.Feature, Traits.Features.CodeGeneration)]
public partial class CodeGenerationTests
{
    [UseExportProvider]
    public class CSharp
    {
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddNamespace()
        {
            var input = "namespace [|N1|] { }";
            var expected = @"namespace N1 {
    namespace N2
    {
    }
}";
            await TestAddNamespaceAsync(input, expected,
                name: "N2");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddField()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public int F;
}";
            await TestAddFieldAsync(input, expected,
                type: GetTypeSymbol(typeof(int)));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddStaticField()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    private static string F;
}";
            await TestAddFieldAsync(input, expected,
                type: GetTypeSymbol(typeof(string)),
                accessibility: Accessibility.Private,
                modifiers: new Editing.DeclarationModifiers(isStatic: true));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddArrayField()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public int[] F;
}";
            await TestAddFieldAsync(input, expected,
                type: CreateArrayType(typeof(int)));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddUnsafeField()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public unsafe int F;
}";
            await TestAddFieldAsync(input, expected,
                modifiers: new Editing.DeclarationModifiers(isUnsafe: true),
                type: GetTypeSymbol(typeof(int)));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddFieldToCompilationUnit()
        {
            var input = "";
            var expected = "public int F;\n";
            await TestAddFieldAsync(input, expected,
                type: GetTypeSymbol(typeof(int)), addToCompilationUnit: true);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddConstructor()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public C()
    {
    }
}";
            await TestAddConstructorAsync(input, expected);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddConstructorWithoutBody()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public C();
}";
            await TestAddConstructorAsync(input, expected,
                context: new CodeGenerationContext(generateMethodBodies: false));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddConstructorResolveNamespaceImport()
        {
            var input = "class [|C|] { }";
            var expected = @"using System;
 
class C
{
    public C(DateTime dt, int i)
    {
    }
}";
            await TestAddConstructorAsync(input, expected,
                parameters: Parameters(Parameter(typeof(DateTime), "dt"), Parameter(typeof(int), "i")));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddChainedConstructor()
        {
            var input = "class [|C|] { public C(int i) { } }";
            var expected = "class C { public C() : this(42) { } public C(int i) { } }";
            await TestAddConstructorAsync(input, expected,
                thisArguments: [CS.SyntaxFactory.ParseExpression("42")]);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddStaticConstructor()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    static C()
    {
    }
}";
            await TestAddConstructorAsync(input, expected,
                modifiers: new Editing.DeclarationModifiers(isStatic: true));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544082")]
        public async Task AddClass()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    public class C
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddClassEscapeName()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    public class @class
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected,
                name: "class");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddClassUnicodeName()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    public class classæøå
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected,
                name: "classæøå");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544405")]
        public async Task AddStaticClass()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    public static class C
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected,
                modifiers: new Editing.DeclarationModifiers(isStatic: true));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544405")]
        public async Task AddStaticAbstractClass()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    public static class C
    {
    }
}";
            // note that 'abstract' is dropped here
            await TestAddNamedTypeAsync(input, expected,
                modifiers: new Editing.DeclarationModifiers(isStatic: true, isAbstract: true));
        }
 
        [Theory, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544405")]
        [InlineData(Accessibility.NotApplicable)]
        [InlineData(Accessibility.Internal)]
        [InlineData(Accessibility.Public)]
        public async Task AddFileClass(Accessibility accessibility)
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    file class C
    {
    }
}";
            // note: when invalid combinations of modifiers+accessibility are present here,
            // we actually drop the accessibility. This is similar to what is done if someone declares a 'static abstract class C { }'.
            await TestAddNamedTypeAsync(input, expected,
                accessibility: accessibility,
                modifiers: new Editing.DeclarationModifiers(isFile: true));
        }
 
        [Theory, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544405")]
        [InlineData("struct", TypeKind.Struct)]
        [InlineData("interface", TypeKind.Interface)]
        [InlineData("enum", TypeKind.Enum)]
        public async Task AddFileType(string kindString, TypeKind typeKind)
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    file " + kindString + @" C
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected,
                typeKind: typeKind,
                modifiers: new Editing.DeclarationModifiers(isFile: true));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544405")]
        public async Task AddSealedClass()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    private sealed class C
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected,
                accessibility: Accessibility.Private,
                modifiers: new Editing.DeclarationModifiers(isSealed: true));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544405")]
        public async Task AddAbstractClass()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    protected internal abstract class C
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected,
                accessibility: Accessibility.ProtectedOrInternal,
                modifiers: new Editing.DeclarationModifiers(isAbstract: true));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddStruct()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    internal struct S
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected,
                name: "S",
                accessibility: Accessibility.Internal,
                typeKind: TypeKind.Struct);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/546224")]
        public async Task AddSealedStruct()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    public struct S
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected,
                name: "S",
                modifiers: new Editing.DeclarationModifiers(isSealed: true),
                accessibility: Accessibility.Public,
                typeKind: TypeKind.Struct);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddInterface()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    public interface I
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected,
                name: "I",
                typeKind: TypeKind.Interface);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544080")]
        public async Task AddEnum()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    public enum E
    {
    }
}";
            await TestAddNamedTypeAsync(input, expected, "E",
                typeKind: TypeKind.Enum);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544527")]
        public async Task AddEnumWithValues()
        {
            var input = "namespace [|N|] { }";
            var expected = @"namespace N
{
    public enum E
    {
        F1 = 1,
        F2 = 2
    }
}";
            await TestAddNamedTypeAsync(input, expected, "E",
                typeKind: TypeKind.Enum,
                members: Members(CreateEnumField("F1", 1), CreateEnumField("F2", 2)));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544080")]
        public async Task AddDelegateType()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public delegate int D(string s);
}";
            await TestAddDelegateTypeAsync(input, expected,
                returnType: typeof(int),
                parameters: Parameters(Parameter(typeof(string), "s")));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/546224")]
        public async Task AddSealedDelegateType()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public delegate int D(string s);
}";
            await TestAddDelegateTypeAsync(input, expected,
                returnType: typeof(int),
                parameters: Parameters(Parameter(typeof(string), "s")),
                modifiers: new Editing.DeclarationModifiers(isSealed: true));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddEvent()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public event System.Action E;
}";
            await TestAddEventAsync(input, expected,
                context: new CodeGenerationContext(addImports: false));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddCustomEventToClassFromSourceSymbol()
        {
            var sourceGenerated = @"class [|C2|]
{
    event EventHandler Click
    {
        add
        {
            Events.AddHandler(""ClickEvent"", value)
        }
        remove
        {
            Events.RemoveHandler(""ClickEvent"", value)
        }
    }
}";
            var input = "class [|C1|] { }";
            var expected = @"class C1
{
    event EventHandler Click
    {
        add
        {
            Events.AddHandler(""ClickEvent"", value)
        }
        remove
        {
            Events.RemoveHandler(""ClickEvent"", value)
        }
    }
}";
            var context = new CodeGenerationContext(reuseSyntax: true);
            await TestGenerateFromSourceSymbolAsync(sourceGenerated, input, expected, onlyGenerateMembers: true, context: context);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddUnsafeEvent()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public unsafe event System.Action E;
}";
            await TestAddEventAsync(input, expected,
                modifiers: new Editing.DeclarationModifiers(isUnsafe: true),
                context: new CodeGenerationContext(addImports: false));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddEventWithAccessors()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public event System.Action E
    {
        add
        {
        }
 
        remove
        {
        }
    }
}";
            await TestAddEventAsync(input, expected,
                addMethod: CodeGenerationSymbolFactory.CreateAccessorSymbol([], Accessibility.NotApplicable, []),
                removeMethod: CodeGenerationSymbolFactory.CreateAccessorSymbol([], Accessibility.NotApplicable, []),
                context: new CodeGenerationContext(addImports: false));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddMethodToClass()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public void M()
    {
    }
}";
            await TestAddMethodAsync(input, expected,
                returnType: typeof(void));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddMethodToClassFromSourceSymbol()
        {
            var sourceGenerated = @"class [|C2|]
{
    public int FInt()
    {
        return 0;
    }
}";
            var input = "class [|C1|] { }";
            var expected = @"class C1
{
    public int FInt()
    {
        return 0;
    }
}";
            var context = new CodeGenerationContext(reuseSyntax: true);
            await TestGenerateFromSourceSymbolAsync(sourceGenerated, input, expected, onlyGenerateMembers: true, context: context);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddMethodToClassEscapedName()
        {
            var input = "class [|C|] { }";
            var expected = @"using System;
 
class C
{
    public DateTime @static()
    {
    }
}";
            await TestAddMethodAsync(input, expected,
                name: "static",
                returnType: typeof(DateTime));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddStaticMethodToStruct()
        {
            var input = "struct [|S|] { }";
            var expected = @"struct S
{
    public static int M()
    {
        $$
    }
}";
            await TestAddMethodAsync(input, expected,
                modifiers: new Editing.DeclarationModifiers(isStatic: true),
                returnType: typeof(int),
                statements: "return 0;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddSealedOverrideMethod()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public sealed override int GetHashCode()
    {
        $$
    }
}";
            await TestAddMethodAsync(input, expected,
                name: "GetHashCode",
                modifiers: new Editing.DeclarationModifiers(isOverride: true, isSealed: true),
                returnType: typeof(int),
                statements: "return 0;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAbstractMethod()
        {
            var input = "abstract class [|C|] { }";
            var expected = @"abstract class C
{
    public abstract int M();
}";
            await TestAddMethodAsync(input, expected,
                modifiers: new Editing.DeclarationModifiers(isAbstract: true),
                returnType: typeof(int));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddMethodWithoutBody()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public int M();
}";
            await TestAddMethodAsync(input, expected,
                returnType: typeof(int),
                context: new CodeGenerationContext(generateMethodBodies: false));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddGenericMethod()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public int M<T>()
    {
        $$
    }
}";
            await TestAddMethodAsync(input, expected,
                returnType: typeof(int),
                typeParameters: [CodeGenerationSymbolFactory.CreateTypeParameterSymbol("T")],
                statements: "return new T().GetHashCode();");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddVirtualMethod()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    protected virtual int M()
    {
        $$
    }
}";
            await TestAddMethodAsync(input, expected,
                accessibility: Accessibility.Protected,
                modifiers: new Editing.DeclarationModifiers(isVirtual: true),
                returnType: typeof(int),
                statements: "return 0;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddUnsafeNewMethod()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public unsafe new string ToString()
    {
        $$
    }
}";
            await TestAddMethodAsync(input, expected,
                name: "ToString",
                modifiers: new Editing.DeclarationModifiers(isNew: true, isUnsafe: true),
                returnType: typeof(string),
                statements: "return String.Empty;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddExplicitImplementationOfUnsafeMethod()
        {
            var input = "interface I { unsafe void M(int i); } class [|C|] : I { }";
            var expected = @"interface I { unsafe void M(int i); }
class C : I
{
    unsafe void I.M(int i)
    {
    }
}";
            await TestAddMethodAsync(input, expected,
                name: "M",
                returnType: typeof(void),
                parameters: Parameters(Parameter(typeof(int), "i")),
                modifiers: new Editing.DeclarationModifiers(isUnsafe: true),
                getExplicitInterfaces: s => [.. s.LookupSymbols(input.IndexOf('M'), null, "M").OfType<IMethodSymbol>()]);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddExplicitImplementation()
        {
            var input = "interface I { void M(int i); } class [|C|] : I { }";
            var expected = @"interface I { void M(int i); }
class C : I
{
    void I.M(int i)
    {
    }
}";
            await TestAddMethodAsync(input, expected,
                name: "M",
                returnType: typeof(void),
                parameters: Parameters(Parameter(typeof(int), "i")),
                getExplicitInterfaces: s => [.. s.LookupSymbols(input.IndexOf('M'), null, "M").OfType<IMethodSymbol>()]);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddTrueFalseOperators()
        {
            var input = @"
class [|C|]
{
}";
            var expected = @"
class C
{
    public static bool operator true(C other)
    {
        $$
    }
 
    public static bool operator false(C other)
    {
        $$
    }
}";
            await TestAddOperatorsAsync(input, expected,
                [CodeGenerationOperatorKind.True, CodeGenerationOperatorKind.False],
                parameters: Parameters(Parameter("C", "other")),
                returnType: typeof(bool),
                statements: "return false;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddUnaryOperators()
        {
            var input = @"
class [|C|]
{
}";
            var expected = @"
class C
{
    public static object operator +(C other)
    {
        $$
    }
 
    public static object operator -(C other)
    {
        $$
    }
 
    public static object operator !(C other)
    {
        $$
    }
 
    public static object operator ~(C other)
    {
        $$
    }
 
    public static object operator ++(C other)
    {
        $$
    }
 
    public static object operator --(C other)
    {
        $$
    }
}";
            await TestAddOperatorsAsync(input, expected,
                [
                    CodeGenerationOperatorKind.UnaryPlus,
                    CodeGenerationOperatorKind.UnaryNegation,
                    CodeGenerationOperatorKind.LogicalNot,
                    CodeGenerationOperatorKind.OnesComplement,
                    CodeGenerationOperatorKind.Increment,
                    CodeGenerationOperatorKind.Decrement
                ],
                parameters: Parameters(Parameter("C", "other")),
                returnType: typeof(object),
                statements: "return null;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddBinaryOperators()
        {
            var input = @"
class [|C|]
{
}";
            var expected = @"
class C
{
    public static object operator +(C a, C b)
    {
        $$
    }
 
    public static object operator -(C a, C b)
    {
        $$
    }
 
    public static object operator *(C a, C b)
    {
        $$
    }
 
    public static object operator /(C a, C b)
    {
        $$
    }
 
    public static object operator %(C a, C b)
    {
        $$
    }
 
    public static object operator &(C a, C b)
    {
        $$
    }
 
    public static object operator |(C a, C b)
    {
        $$
    }
 
    public static object operator ^(C a, C b)
    {
        $$
    }
 
    public static object operator <<(C a, C b)
    {
        $$
    }
 
    public static object operator >>(C a, C b)
    {
        $$
    }
}";
            await TestAddOperatorsAsync(input, expected,
                [
                    CodeGenerationOperatorKind.Addition,
                    CodeGenerationOperatorKind.Subtraction,
                    CodeGenerationOperatorKind.Multiplication,
                    CodeGenerationOperatorKind.Division,
                    CodeGenerationOperatorKind.Modulus,
                    CodeGenerationOperatorKind.BitwiseAnd,
                    CodeGenerationOperatorKind.BitwiseOr,
                    CodeGenerationOperatorKind.ExclusiveOr,
                    CodeGenerationOperatorKind.LeftShift,
                    CodeGenerationOperatorKind.RightShift
                ],
                parameters: Parameters(Parameter("C", "a"), Parameter("C", "b")),
                returnType: typeof(object),
                statements: "return null;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddComparisonOperators()
        {
            var input = @"
class [|C|]
{
}";
            var expected = @"
class C
{
    public static bool operator ==(C a, C b)
    {
        $$
    }
 
    public static bool operator !=(C a, C b)
    {
        $$
    }
 
    public static bool operator <(C a, C b)
    {
        $$
    }
 
    public static bool operator >(C a, C b)
    {
        $$
    }
 
    public static bool operator <=(C a, C b)
    {
        $$
    }
 
    public static bool operator >=(C a, C b)
    {
        $$
    }
}";
            await TestAddOperatorsAsync(input, expected,
                [
                    CodeGenerationOperatorKind.Equality,
                    CodeGenerationOperatorKind.Inequality,
                    CodeGenerationOperatorKind.GreaterThan,
                    CodeGenerationOperatorKind.LessThan,
                    CodeGenerationOperatorKind.LessThanOrEqual,
                    CodeGenerationOperatorKind.GreaterThanOrEqual
                ],
                parameters: Parameters(Parameter("C", "a"), Parameter("C", "b")),
                returnType: typeof(bool),
                statements: "return true;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddUnsupportedOperator()
        {
            var input = "class [|C|] { }";
            await TestAddUnsupportedOperatorAsync(input,
                operatorKind: CodeGenerationOperatorKind.Like,
                parameters: Parameters(Parameter("C", "a"), Parameter("C", "b")),
                returnType: typeof(bool),
                statements: "return true;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddExplicitConversion()
        {
            var input = @"class [|C|] { }";
            var expected = @"class C
{
    public static explicit operator int(C other)
    {
        $$
    }
}";
            await TestAddConversionAsync(input, expected,
                toType: typeof(int),
                fromType: Parameter("C", "other"),
                statements: "return 0;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddImplicitConversion()
        {
            var input = @"class [|C|] { }";
            var expected = @"class C
{
    public static implicit operator int(C other)
    {
        $$
    }
}";
            await TestAddConversionAsync(input, expected,
                toType: typeof(int),
                fromType: Parameter("C", "other"),
                isImplicit: true,
                statements: "return 0;");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddStatements()
        {
            var input = "class C { public void [|M|]() { Console.WriteLine(1); } }";
            var expected = "class C { public void M() { Console.WriteLine(1); $$} }";
            await TestAddStatementsAsync(input, expected, "Console.WriteLine(2);");
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        [WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/840265")]
        public async Task AddDefaultParameterWithNonDefaultValueToMethod()
        {
            var input = "class C { public void [|M|]() { } }";
            var expected = "class C { public void M(string text =\"Hello\") { } }";
            await TestAddParametersAsync(input, expected,
                Parameters(Parameter(typeof(string), "text", true, "Hello")));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddDefaultParameterWithDefaultValueToMethod()
        {
            var input = "class C { public void [|M|]() { } }";
            var expected = "class C { public void M(double number =0) { } }";
            await TestAddParametersAsync(input, expected,
                Parameters(Parameter(typeof(double), "number", true)));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddParametersToMethod()
        {
            var input = "class C { public void [|M|]() { } }";
            var expected = "class C { public void M(int num, string text =\"Hello!\", float floating =0.5F) { } }";
            await TestAddParametersAsync(input, expected,
                Parameters(Parameter(typeof(int), "num"), Parameter(typeof(string), "text", true, "Hello!"), Parameter(typeof(float), "floating", true, .5f)));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        [WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/841365")]
        public async Task AddParamsParameterToMethod()
        {
            var input = "class C { public void [|M|]() { } }";
            var expected = "class C { public void M(params char[]characters) { } }";
            await TestAddParametersAsync(input, expected,
                Parameters(Parameter(typeof(char[]), "characters", isParams: true)));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration), WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/544015")]
        public async Task AddAutoProperty()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public int P { get; internal set; }
}";
            await TestAddPropertyAsync(input, expected,
                type: typeof(int),
                setterAccessibility: Accessibility.Internal);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddUnsafeAutoProperty()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public unsafe int P { get; internal set; }
}";
            await TestAddPropertyAsync(input, expected,
                type: typeof(int),
                modifiers: new Editing.DeclarationModifiers(isUnsafe: true),
                setterAccessibility: Accessibility.Internal);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddPropertyToClassFromSourceSymbol()
        {
            var sourceGenerated = @"class [|C2|]
{
    public int P
    {
        get
        {
            return 0;
        }
    }
}";
            var input = "class [|C1|] { }";
            var expected = @"class C1
{
    public int P
    {
        get
        {
            return 0;
        }
    }
}";
            var context = new CodeGenerationContext(reuseSyntax: true);
            await TestGenerateFromSourceSymbolAsync(sourceGenerated, input, expected, onlyGenerateMembers: true, context: context);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddIndexer1()
        {
            var input = "class [|C|] { }";
            var expected = "class C { public string this[int i] => String.Empty; }";
            await TestAddPropertyAsync(input, expected,
                type: typeof(string),
                parameters: Parameters(Parameter(typeof(int), "i")),
                getStatements: "return String.Empty;",
                isIndexer: true);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddIndexer2()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public string this[int i]
    {
        get
        {
            $$
        }
    }
}";
            await TestAddPropertyAsync(input, expected,
                type: typeof(string),
                parameters: Parameters(Parameter(typeof(int), "i")),
                getStatements: "return String.Empty;",
                isIndexer: true,
                options: new OptionsCollection(LanguageNames.CSharp)
                {
                    { CSharpCodeStyleOptions.PreferExpressionBodiedAccessors, CSharpCodeStyleOptions.NeverWithSilentEnforcement },
                    { CSharpCodeStyleOptions.PreferExpressionBodiedIndexers, CSharpCodeStyleOptions.NeverWithSilentEnforcement },
                });
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddParameterfulProperty()
        {
            var input = "class [|C|] { }";
            var expected = @"class C
{
    public string get_P(int i, int j)
    {
        $$
    }
 
    public void set_P(int i, int j, string value)
    {
    }
}";
            await TestAddPropertyAsync(input, expected,
                type: typeof(string),
                getStatements: "return String.Empty;",
                setStatements: "",
                parameters: Parameters(Parameter(typeof(int), "i"), Parameter(typeof(int), "j")));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToTypes()
        {
            var input = "class [|C|] { }";
            var expected = @"[System.Serializable]
class C { }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromTypes()
        {
            var input = @"[System.Serializable] class [|C|] { }";
            var expected = "class C { }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToMethods()
        {
            var input = "class C { public void [|M()|] { } }";
            var expected = "class C { [System.Serializable] public void M() { } }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromMethods()
        {
            var input = "class C { [System.Serializable] public void [|M()|] { } }";
            var expected = "class C { public void M() { } }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToFields()
        {
            var input = "class C { [|public int F|]; }";
            var expected = "class C { [System.Serializable] public int F; }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromFields()
        {
            var input = "class C { [System.Serializable] public int [|F|]; }";
            var expected = "class C { public int F; }";
            await TestRemoveAttributeAsync<FieldDeclarationSyntax>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToProperties()
        {
            var input = "class C { public int [|P|] { get; set; }}";
            var expected = "class C { [System.Serializable] public int P { get; set; } }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromProperties()
        {
            var input = "class C { [System.Serializable] public int [|P|] { get; set; }}";
            var expected = "class C { public int P { get; set; } }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToPropertyAccessor()
        {
            var input = "class C { public int P { [|get|]; set; }}";
            var expected = "class C { public int P { [System.Serializable] get; set; }}";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromPropertyAccessor()
        {
            var input = "class C { public int P { [System.Serializable] [|get|]; set; } }";
            var expected = "class C { public int P { get; set; } }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToEnums()
        {
            var input = "enum [|C|] { One, Two }";
            var expected = @"[System.Serializable]
enum C { One, Two }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromEnums()
        {
            var input = "[System.Serializable] enum [|C|] { One, Two }";
            var expected = "enum C { One, Two }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToEnumMembers()
        {
            var input = "enum C { [|One|], Two }";
            var expected = "enum C { [System.Serializable] One, Two }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromEnumMembers()
        {
            var input = "enum C { [System.Serializable] [|One|], Two }";
            var expected = "enum C { One, Two }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToIndexer()
        {
            var input = "class C { public int [|this[int y]|] { get; set; }}";
            var expected = "class C { [System.Serializable] public int this[int y] { get; set; } }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromIndexer()
        {
            var input = "class C { [System.Serializable] public int [|this[int y]|] { get; set; }}";
            var expected = "class C { public int this[int y] { get; set; } }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToOperator()
        {
            var input = "class C { public static C operator [|+|] (C c1, C c2) { return new C(); }}";
            var expected = "class C { [System.Serializable] public static C operator +(C c1, C c2) { return new C(); } }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromOperator()
        {
            var input = "class C { [System.Serializable] public static C operator [|+|](C c1, C c2) { return new C(); }}";
            var expected = "class C { public static C operator +(C c1, C c2) { return new C(); } }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToDelegate()
        {
            var input = "delegate int [|D()|];";
            var expected = @"[System.Serializable]
delegate int D();";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromDelegate()
        {
            var input = "[System.Serializable] delegate int [|D()|];";
            var expected = "delegate int D();";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToParam()
        {
            var input = "class C { public void M([|int x|]) { } }";
            var expected = "class C { public void M([System.Serializable] int x) { } }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromParam()
        {
            var input = "class C { public void M([System.Serializable] [|int x|]) { } }";
            var expected = "class C { public void M(int x) { } }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToTypeParam()
        {
            var input = "class C<[|T|]> { }";
            var expected = "class C<[System.Serializable] T> { }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeFromTypeParam()
        {
            var input = "class C<[System.Serializable] [|T|]> { }";
            var expected = "class C<T> { }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeToCompilationUnit()
        {
            var input = "[|class C { } class D {} |]";
            var expected = @"[assembly: System.Serializable]
 
class C { }
class D { }";
            await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute), AssemblyKeyword);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task AddAttributeWithWrongTarget()
        {
            var input = "[|class C { } class D {} |]";
            var expected = "";
            await Assert.ThrowsAsync<AggregateException>(async () =>
                await TestAddAttributeAsync(input, expected, typeof(SerializableAttribute), RefKeyword));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeWithTrivia()
        {
            // With trivia.
            var input = @"// Comment 1
[System.Serializable] // Comment 2
/* Comment 3*/ class [|C|] { }";
            var expected = @"// Comment 1
/* Comment 3*/ class C { }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeWithTrivia_NewLine()
        {
            // With trivia, redundant newline at end of attribute removed.
            var input = @"// Comment 1
[System.Serializable]
/* Comment 3*/ class [|C|] { }";
            var expected = @"// Comment 1
/* Comment 3*/ class C { }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeWithMultipleAttributes()
        {
            // Multiple attributes.
            var input = @"// Comment 1
/*Comment2*/[ /*Comment3*/ System.Serializable /*Comment4*/, /*Comment5*/System.Flags /*Comment6*/] /*Comment7*/
/* Comment 8*/
class [|C|] { }";
            var expected = @"// Comment 1
/*Comment2*/[ /*Comment3*/  /*Comment5*/System.Flags /*Comment6*/] /*Comment7*/
/* Comment 8*/
class C { }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task RemoveAttributeWithMultipleAttributeLists()
        {
            // Multiple attributes.
            var input = @"// Comment 1
/*Comment2*/[ /*Comment3*/ System.Serializable /*Comment4*/, /*Comment5*/System.Flags /*Comment6*/] /*Comment7*/
[ /*Comment9*/ System.Obsolete /*Comment10*/] /*Comment11*/
/* Comment12*/
class [|C|] { }";
            var expected = @"// Comment 1
/*Comment2*/[ /*Comment3*/  /*Comment5*/System.Flags /*Comment6*/] /*Comment7*/
[ /*Comment9*/ System.Obsolete /*Comment10*/] /*Comment11*/
/* Comment12*/
class C { }";
            await TestRemoveAttributeAsync<SyntaxNode>(input, expected, typeof(SerializableAttribute));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task TestUpdateModifiers()
        {
            var input = @"public static class [|C|] // Comment 1
{
    // Comment 2
}";
            var expected = @"internal partial sealed class C // Comment 1
{
    // Comment 2
}";
            var eol = SyntaxFactory.EndOfLine(@"");
            var newModifiers = new[] { InternalKeyword.WithLeadingTrivia(eol) }.Concat(
                CreateModifierTokens(new Editing.DeclarationModifiers(isSealed: true, isPartial: true), LanguageNames.CSharp));
 
            await TestUpdateDeclarationAsync<ClassDeclarationSyntax>(input, expected, modifiers: newModifiers);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task TestUpdateAccessibility()
        {
            var input = @"// Comment 0
public static class [|C|] // Comment 1
{
    // Comment 2
}";
            var expected = @"// Comment 0
internal static class C // Comment 1
{
    // Comment 2
}";
            await TestUpdateDeclarationAsync<ClassDeclarationSyntax>(input, expected, accessibility: Accessibility.Internal);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task TestUpdateDeclarationType()
        {
            var input = @"
public static class C
{
    // Comment 1
    public static char [|F|]() { return 0; }
}";
            var expected = @"
public static class C
{
    // Comment 1
    public static int F() { return 0; }
}";
            await TestUpdateDeclarationAsync<MethodDeclarationSyntax>(input, expected, getType: GetTypeSymbol(typeof(int)));
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task TestUpdateDeclarationMembers()
        {
            var input = @"
public static class [|C|]
{
    // Comment 0
    public int {|RetainedMember:f|};
 
    // Comment 1
    public static char F() { return 0; }
}";
            var expected = @"
public static class C
{
    // Comment 0
    public int f;
    public int f2;
}";
            var getField = CreateField(Accessibility.Public, new Editing.DeclarationModifiers(), typeof(int), "f2");
            var getMembers = ImmutableArray.Create(getField);
            await TestUpdateDeclarationAsync<ClassDeclarationSyntax>(
                input, expected, getNewMembers: getMembers);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGeneration)]
        public async Task TestUpdateDeclarationMembers_DifferentOrder()
        {
            var input = @"
public static class [|C|]
{
    // Comment 0
    public int f;
 
    // Comment 1
    public static char {|RetainedMember:F|}() { return 0; }
}";
            var expected = @"
public static class C
{
    public int f2;
 
    // Comment 1
    public static char F() { return 0; }
}";
            var getField = CreateField(Accessibility.Public, new Editing.DeclarationModifiers(), typeof(int), "f2");
            var getMembers = ImmutableArray.Create(getField);
            await TestUpdateDeclarationAsync<ClassDeclarationSyntax>(input, expected, getNewMembers: getMembers, declareNewMembersAtTop: true);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGenerationSortDeclarations)]
        public async Task SortAroundDestructor()
        {
            var generationSource = "public class [|C|] { public C(){} public int this[int index]{get{return 0;}set{value = 0;}} }";
            var initial = "public class [|C|] { ~C(){} }";
            var expected = @"public class C {
    public C()
    {
    }
 
    ~C(){}
 
    public int this[int index]
    {
        get
        {
        }
 
        set
        {
        }
    }
}";
            await TestGenerateFromSourceSymbolAsync(generationSource, initial, expected, onlyGenerateMembers: true);
        }
 
        [Fact, Trait(Traits.Feature, Traits.Features.CodeGenerationSortDeclarations)]
        public async Task SortOperators()
        {
            var generationSource = @"
namespace N
{
    public class [|C|]
    {
        // Unary operators
        public static bool operator false (C other) { return false; }
        public static bool operator true (C other) { return true; }
        public static C operator ++ (C other) { return null; }
        public static C operator -- (C other) { return null; }
        public static C operator ~ (C other) { return null; }
        public static C operator ! (C other) { return null; }
        public static C operator - (C other) { return null; }
        public static C operator + (C other) { return null; }
    
        // Binary operators
        public static C operator >> (C a, int shift) { return null; }
        public static C operator << (C a, int shift) { return null; }
        public static C operator ^ (C a, C b) { return null; }
        public static C operator | (C a, C b) { return null; }
        public static C operator & (C a, C b) { return null; }
        public static C operator % (C a, C b) { return null; }
        public static C operator / (C a, C b) { return null; }
        public static C operator * (C a, C b) { return null; }
        public static C operator - (C a, C b) { return null; }
        public static C operator + (C a, C b) { return null; }
 
        // Comparison operators
        public static bool operator >= (C a, C b) { return true; }
        public static bool operator <= (C a, C b) { return true; }
        public static bool operator > (C a, C b) { return true; }
        public static bool operator < (C a, C b) { return true; }
        public static bool operator != (C a, C b) { return true; }
        public static bool operator == (C a, C b) { return true; }
    }
}";
            var initial = "namespace [|N|] { }";
            var expected = @"namespace N
{
    public class C
    {
        public static C operator +(C other);
        public static C operator +(C a, C b);
        public static C operator -(C other);
        public static C operator -(C a, C b);
        public static C operator !(C other);
        public static C operator ~(C other);
        public static C operator ++(C other);
        public static C operator --(C other);
        public static C operator *(C a, C b);
        public static C operator /(C a, C b);
        public static C operator %(C a, C b);
        public static C operator &(C a, C b);
        public static C operator |(C a, C b);
        public static C operator ^(C a, C b);
        public static C operator <<(C a, int shift);
        public static C operator >>(C a, int shift);
        public static bool operator ==(C a, C b);
        public static bool operator !=(C a, C b);
        public static bool operator <(C a, C b);
        public static bool operator >(C a, C b);
        public static bool operator <=(C a, C b);
        public static bool operator >=(C a, C b);
        public static bool operator true(C other);
        public static bool operator false(C other);
    }
}";
            await TestGenerateFromSourceSymbolAsync(generationSource, initial, expected,
                forceLanguage: LanguageNames.CSharp,
                context: new CodeGenerationContext(generateMethodBodies: false));
        }
    }
 
    [Fact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/665008")]
    public async Task TestExtensionMethods()
    {
        var generationSource = @"
public static class [|C|]
{
    public static void ExtMethod1(this string s, int y, string z) {}
}";
        var initial = "public static class [|C|] {}";
        var expected = @"public static class C
{
    public static void ExtMethod1(this string s, int y, string z);
}";
        await TestGenerateFromSourceSymbolAsync(generationSource, initial, expected,
            context: new CodeGenerationContext(generateMethodBodies: false),
            onlyGenerateMembers: true);
    }
 
    [Fact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/530829")]
    public async Task TestVBPropertiesWithParams()
    {
        var generationSource = @"
Namespace N
    Public Class [|C|]
        Public Overridable Property IndexProp(ByVal p1 As Integer) As String
            Get
                Return Nothing
            End Get
            Set(ByVal value As String)
            End Set
        End Property
    End Class
End Namespace
";
 
        var initial = "namespace [|N|] {}";
        var expected = @"namespace N
{
    public class C
    {
        public virtual string get_IndexProp(int p1);
        public virtual void set_IndexProp(int p1, string value);
    }
}";
        await TestGenerateFromSourceSymbolAsync(generationSource, initial, expected,
            context: new CodeGenerationContext(generateMethodBodies: false));
    }
 
    [Fact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/812738")]
    public async Task TestRefParamsWithDefaultValue()
    {
        var generationSource = @"
Public Class [|C|]
    Public Sub Goo(x As Integer, Optional ByRef y As Integer = 10, Optional ByRef z As Object = Nothing)
    End Sub
End Class";
        var initial = "public class [|C|] {}";
        var expected = @"public class C
{
    public void Goo(int x, ref int y, ref object z);
}";
        await TestGenerateFromSourceSymbolAsync(generationSource, initial, expected,
            context: new CodeGenerationContext(generateMethodBodies: false),
            onlyGenerateMembers: true);
    }
 
    [Fact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/848357")]
    public async Task TestConstraints()
    {
        var generationSource = @"
namespace N
{
    public class [|C|]<T, U> where T : struct where U : class
    {
        public void Goo<Q, R>() where Q : new() where R : IComparable { }
        public delegate void D<T, U>(T t, U u) where T : struct where U : class;
    }
}
";
        var initial = "namespace [|N|] {}";
        var expected = @"namespace N
{
    public class C<T, U>
        where T : struct
        where U : class
    {
        public void Goo<Q, R>()
            where Q : new()
            where R : IComparable;
 
        public delegate void D<T, U>(T t, U u)
            where T : struct
            where U : class;
    }
}";
        await TestGenerateFromSourceSymbolAsync(generationSource, initial, expected,
            context: new CodeGenerationContext(generateMethodBodies: false),
            onlyGenerateMembers: true);
    }
}