|
// 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;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.CSharp.Formatting;
using Microsoft.CodeAnalysis.CSharp.Test.Utilities;
using Microsoft.CodeAnalysis.Editing;
using Microsoft.CodeAnalysis.Editor.UnitTests.CodeActions;
using Microsoft.CodeAnalysis.Editor.UnitTests.Utilities;
using Microsoft.CodeAnalysis.Formatting;
using Microsoft.CodeAnalysis.Indentation;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Microsoft.CodeAnalysis.Test.Utilities;
using Microsoft.VisualStudio.Text.Editor.Commanding.Commands;
using Roslyn.Test.Utilities;
using Xunit;
using Xunit.Abstractions;
namespace Microsoft.CodeAnalysis.Editor.CSharp.UnitTests.Formatting;
[Trait(Traits.Feature, Traits.Features.Formatting)]
public sealed class FormattingEngineTests(ITestOutputHelper output) : CSharpFormattingEngineTestBase(output)
{
private static OptionsCollection SmartIndentButDoNotFormatWhileTyping()
=> new(LanguageNames.CSharp)
{
{ IndentationOptionsStorage.SmartIndent, FormattingOptions2.IndentStyle.Smart },
{ AutoFormattingOptionsStorage.FormatOnTyping, false },
{ AutoFormattingOptionsStorage.FormatOnCloseBrace, false },
};
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/539682")]
public void FormatDocumentCommandHandler()
{
var code = """
class Program
{
static void Main(string[] args)
{
int x;$$
int y;
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
int x;$$
int y;
}
}
""";
AssertFormatWithView(expected, code);
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/539682")]
public void FormatDocumentPasteCommandHandler()
{
var code = """
class Program
{
static void Main(string[] args)
{
int x;$$
int y;
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
int x;$$
int y;
}
}
""";
AssertFormatWithPasteOrReturn(expected, code, allowDocumentChanges: true);
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/547261")]
public void FormatDocumentReadOnlyWorkspacePasteCommandHandler()
{
var code = """
class Program
{
static void Main(string[] args)
{
int x;$$
int y;
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
int x;$$
int y;
}
}
""";
AssertFormatWithPasteOrReturn(expected, code, allowDocumentChanges: false);
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/912965")]
public void DoNotFormatUsingStatementOnReturn()
{
var code = """
class Program
{
static void Main(string[] args)
{
using (null)
using (null)$$
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
using (null)
using (null)$$
}
}
""";
AssertFormatWithPasteOrReturn(expected, code, allowDocumentChanges: true, isPaste: false);
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/912965")]
public void FormatUsingStatementWhenTypingCloseParen()
{
var code = """
class Program
{
static void Main(string[] args)
{
using (null)
using (null)$$
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
using (null)
using (null)
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/912965")]
public void FormatNotUsingStatementOnReturn()
{
var code = """
class Program
{
static void Main(string[] args)
{
using (null)
for (;;)$$
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
using (null)
for (;;)$$
}
}
""";
AssertFormatWithPasteOrReturn(expected, code, allowDocumentChanges: true, isPaste: false);
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/977133")]
public void DoNotFormatRangeOrFormatTokenOnOpenBraceOnSameLine()
{
var code = """
class C
{
public void M()
{
if (true) {$$
}
}
""";
var expected = """
class C
{
public void M()
{
if (true) {
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/pull/14491")]
public void DoNotFormatRangeButFormatTokenOnOpenBraceOnNextLine()
{
var code = """
class C
{
public void M()
{
if (true)
{$$
}
}
""";
var expected = """
class C
{
public void M()
{
if (true)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1007071")]
public void FormatPragmaWarningInbetweenDelegateDeclarationStatement()
{
var code = """
using System;
class Program
{
static void Main(string[] args)
{
Func <bool> a = delegate ()
#pragma warning disable CA0001
{
return true;
};$$
}
}
""";
var expected = """
using System;
class Program
{
static void Main(string[] args)
{
Func<bool> a = delegate ()
#pragma warning disable CA0001
{
return true;
};
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/771761")]
public void FormatHashRegion()
{
var code = """
using System;
class Program
{
static void Main(string[] args)
{
#region$$
}
}
""";
var expected = """
using System;
class Program
{
static void Main(string[] args)
{
#region
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/771761")]
public void FormatHashEndRegion()
{
var code = """
using System;
class Program
{
static void Main(string[] args)
{
#region
#endregion$$
}
}
""";
var expected = """
using System;
class Program
{
static void Main(string[] args)
{
#region
#endregion
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/987373")]
public async Task FormatSpansIndividuallyWithoutCollapsing()
{
var code = """
class C
{
public void M()
{
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
[|if(true){}|]
if(true){}
[|if(true){}|]
}
}
""";
var expected = """
class C
{
public void M()
{
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if (true) { }
if(true){}
if (true) { }
}
}
""";
using var workspace = EditorTestWorkspace.CreateCSharp(code);
var subjectDocument = workspace.Documents.Single();
var spans = subjectDocument.SelectedSpans;
var document = workspace.CurrentSolution.Projects.Single().Documents.Single();
var syntaxRoot = await document.GetRequiredSyntaxRootAsync(CancellationToken.None);
var options = CSharpSyntaxFormattingOptions.Default;
var node = Formatter.Format(syntaxRoot, spans, workspace.Services.SolutionServices, options, rules: default, CancellationToken.None);
Assert.Equal(expected, node.ToFullString());
}
[WpfFact, WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1044118")]
public void SemicolonInCommentOnLastLineDoesNotFormat()
{
var code = """
using System;
class Program
{
static void Main(string[] args)
{
}
}
// ;$$
""";
var expected = """
using System;
class Program
{
static void Main(string[] args)
{
}
}
// ;
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideSingleLineRegularComment_1()
{
var code = """
class Program
{
// {$$
static void Main(int a, int b)
{
}
}
""";
var expected = """
class Program
{
// {
static void Main(int a, int b)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideSingleLineRegularComment_2()
{
var code = """
class Program
{
// {$$
static void Main(int a, int b)
{
}
}
""";
var expected = """
class Program
{
// {
static void Main(int a, int b)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideMultiLineRegularComment_1()
{
var code = """
class Program
{
static void Main(int a/* {$$ */, int b)
{
}
}
""";
var expected = """
class Program
{
static void Main(int a/* { */, int b)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideMultiLineRegularComment_2()
{
var code = """
class Program
{
static void Main(int a/* {$$
*/, int b)
{
}
}
""";
var expected = """
class Program
{
static void Main(int a/* {
*/, int b)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideMultiLineRegularComment_3()
{
var code = """
class Program
{
static void Main(int a/* {$$
*/, int b)
{
}
}
""";
var expected = """
class Program
{
static void Main(int a/* {
*/, int b)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideSingleLineDocComment_1()
{
var code = """
class Program
{
/// {$$
static void Main(int a, int b)
{
}
}
""";
var expected = """
class Program
{
/// {
static void Main(int a, int b)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideSingleLineDocComment_2()
{
var code = """
class Program
{
/// {$$
static void Main(int a, int b)
{
}
}
""";
var expected = """
class Program
{
/// {
static void Main(int a, int b)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideMultiLineDocComment_1()
{
var code = """
class Program
{
/** {$$ **/
static void Main(int a, int b)
{
}
}
""";
var expected = """
class Program
{
/** { **/
static void Main(int a, int b)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideMultiLineDocComment_2()
{
var code = """
class Program
{
/** {$$
**/
static void Main(int a, int b)
{
}
}
""";
var expected = """
class Program
{
/** {
**/
static void Main(int a, int b)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideMultiLineDocComment_3()
{
var code = """
class Program
{
/** {$$
**/
static void Main(int a, int b)
{
}
}
""";
var expected = """
class Program
{
/** {
**/
static void Main(int a, int b)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideInactiveCode()
{
var code = """
class Program
{
#if false
{$$
#endif
static void Main(string[] args)
{
}
}
""";
var expected = """
class Program
{
#if false
{
#endif
static void Main(string[] args)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideStringLiteral()
{
var code = """
class Program
{
static void Main(string[] args)
{
var asdas = "{$$" ;
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
var asdas = "{" ;
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideCharLiteral()
{
var code = """
class Program
{
static void Main(string[] args)
{
var asdas = '{$$' ;
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
var asdas = '{' ;
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/449")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/1077103")]
public void NoFormattingInsideCommentsOfPreprocessorDirectives()
{
var code = """
class Program
{
#region
#endregion // a/*{$$*/
static void Main(string[] args)
{
}
}
""";
var expected = """
class Program
{
#region
#endregion // a/*{*/
static void Main(string[] args)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/464")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/908729")]
public void ColonInSwitchCase()
{
var code = """
class Program
{
static void Main(string[] args)
{
int f = 0;
switch(f)
{
case 1 :$$ break;
}
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
int f = 0;
switch(f)
{
case 1: break;
}
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/464")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/908729")]
public void ColonInDefaultSwitchCase()
{
var code = """
class Program
{
static void Main(string[] args)
{
int f = 0;
switch(f)
{
case 1: break;
default :$$ break;
}
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
int f = 0;
switch(f)
{
case 1: break;
default: break;
}
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/9097")]
public void ColonInPatternSwitchCase01()
{
var code = """
class Program
{
static void Main()
{
switch(f)
{
case int i :$$ break;
}
}
}
""";
var expected = """
class Program
{
static void Main()
{
switch(f)
{
case int i: break;
}
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/464")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/908729")]
public void ColonInLabeledStatement()
{
var code = """
class Program
{
static void Main(string[] args)
{
label1 :$$ int s = 0;
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
label1: int s = 0;
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/464")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/908729")]
public void DoNotFormatColonInTargetAttribute()
{
var code = """
using System;
[method :$$ C]
class C : Attribute
{
}
""";
var expected = """
using System;
[method : C]
class C : Attribute
{
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/464")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/908729")]
public void DoNotFormatColonInBaseList()
{
var code = """
class C :$$ Attribute
{
}
""";
var expected = """
class C : Attribute
{
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/464")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/908729")]
public void DoNotFormatColonInThisConstructor()
{
var code = """
class Goo
{
Goo(int s) :$$ this()
{
}
Goo()
{
}
}
""";
var expected = """
class Goo
{
Goo(int s) : this()
{
}
Goo()
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/464")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/908729")]
public void DoNotFormatColonInConditionalOperator()
{
var code = """
class Program
{
static void Main(string[] args)
{
var vari = goo() ? true :$$ false;
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
var vari = goo() ? true : false;
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/464")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/908729")]
public void DoNotFormatColonInArgument()
{
var code = """
class Program
{
static void Main(string[] args)
{
Main(args :$$ args);
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
Main(args : args);
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/464")]
[WorkItem("http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/908729")]
public void DoNotFormatColonInTypeParameter()
{
var code = """
class Program<T>
{
class C1<U>
where T :$$ U
{
}
}
""";
var expected = """
class Program<T>
{
class C1<U>
where T : U
{
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/2224")]
public void DoNotSmartFormatBracesOnSmartIndentNone()
{
var code = """
class Program<T>
{
class C1<U>
{$$
}
""";
var expected = """
class Program<T>
{
class C1<U>
{
}
""";
var globalOptions = new OptionsCollection(LanguageNames.CSharp)
{
{ IndentationOptionsStorage.SmartIndent, FormattingOptions2.IndentStyle.None }
};
AssertFormatAfterTypeChar(code, expected, globalOptions);
}
[WpfFact]
[Trait(Traits.Feature, Traits.Features.SmartTokenFormatting)]
public void StillAutoIndentCloseBraceWhenFormatOnCloseBraceIsOff()
{
var code = """
namespace N
{
class C
{
// improperly indented code
int x = 10;
}$$
}
""";
var expected = """
namespace N
{
class C
{
// improperly indented code
int x = 10;
}
}
""";
var globalOptions = new OptionsCollection(LanguageNames.CSharp)
{
{ AutoFormattingOptionsStorage.FormatOnCloseBrace, false }
};
AssertFormatAfterTypeChar(code, expected, globalOptions);
}
[WpfFact]
[Trait(Traits.Feature, Traits.Features.SmartTokenFormatting)]
public void AutoIndentCloseBraceWhenFormatOnTypingIsOff()
{
var code = """
namespace N
{
class C
{
// improperly indented code
int x = 10;
}$$
}
""";
var expected = """
namespace N
{
class C
{
// improperly indented code
int x = 10;
}
}
""";
var globalOptions = new OptionsCollection(LanguageNames.CSharp)
{
{ AutoFormattingOptionsStorage.FormatOnTyping, false }
};
AssertFormatAfterTypeChar(code, expected, globalOptions);
}
[WpfFact, Trait(Traits.Feature, Traits.Features.SmartTokenFormatting)]
[WorkItem("https://github.com/dotnet/roslyn/issues/5873")]
public void KeepTabsInCommentsWhenFormattingIsOff()
{
// There are tabs in this test case. Tools that touch the Roslyn repo should
// not remove these as we are explicitly testing tab behavior.
var code =
"""
class Program
{
static void Main()
{
return; /* Comment preceded by tabs */ // This one too
}$$
}
""";
var expected =
"""
class Program
{
static void Main()
{
return; /* Comment preceded by tabs */ // This one too
}
}
""";
var globalOptions = new OptionsCollection(LanguageNames.CSharp)
{
{ AutoFormattingOptionsStorage.FormatOnTyping, false }
};
AssertFormatAfterTypeChar(code, expected, globalOptions);
}
[WpfFact, Trait(Traits.Feature, Traits.Features.SmartTokenFormatting)]
[WorkItem("https://github.com/dotnet/roslyn/issues/5873")]
public void DoNotKeepTabsInCommentsWhenFormattingIsOn()
{
// There are tabs in this test case. Tools that touch the Roslyn repo should
// not remove these as we are explicitly testing tab behavior.
var code = """
class Program
{
static void Main()
{
return; /* Comment preceded by tabs */ // This one too
}$$
}
""";
var expected =
"""
class Program
{
static void Main()
{
return; /* Comment preceded by tabs */ // This one too
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
[Trait(Traits.Feature, Traits.Features.SmartTokenFormatting)]
public void DoNotFormatStatementIfSemicolonOptionIsOff()
{
var code =
"""
namespace N
{
class C
{
int x = 10 ;$$
}
}
""";
var expected =
"""
namespace N
{
class C
{
int x = 10 ;
}
}
""";
var globalOptions = new OptionsCollection(LanguageNames.CSharp)
{
{ AutoFormattingOptionsStorage.FormatOnSemicolon, false }
};
AssertFormatAfterTypeChar(code, expected, globalOptions);
}
[WpfFact]
[Trait(Traits.Feature, Traits.Features.SmartTokenFormatting)]
public void DoNotFormatStatementIfTypingOptionIsOff()
{
var code =
"""
namespace N
{
class C
{
int x = 10 ;$$
}
}
""";
var expected =
"""
namespace N
{
class C
{
int x = 10 ;
}
}
""";
var globalOptions = new OptionsCollection(LanguageNames.CSharp)
{
{ AutoFormattingOptionsStorage.FormatOnTyping, false }
};
AssertFormatAfterTypeChar(code, expected, globalOptions);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/4435")]
[Trait(Traits.Feature, Traits.Features.SmartTokenFormatting)]
public void OpenCurlyNotFormattedIfNotAtStartOfLine()
{
var code =
"""
class C
{
public int P {$$
}
""";
var expected =
"""
class C
{
public int P {
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/4435")]
[Trait(Traits.Feature, Traits.Features.SmartTokenFormatting)]
public void OpenCurlyFormattedIfAtStartOfLine()
{
var code =
"""
class C
{
public int P
{$$
}
""";
var expected =
"""
class C
{
public int P
{
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void DoNotFormatIncompleteBlockOnSingleLineIfNotTypingCloseCurly1()
{
var code = """
namespace ConsoleApplication1
{
class Program
{
static bool Property
{
get { return true;$$
}
}
""";
var expected = """
namespace ConsoleApplication1
{
class Program
{
static bool Property
{
get { return true;
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void DoNotFormatIncompleteBlockOnSingleLineIfNotTypingCloseCurly2()
{
var code = """
namespace ConsoleApplication1
{
class Program
{
static bool Property { get { return true;$$
}
}
""";
var expected = """
namespace ConsoleApplication1
{
class Program
{
static bool Property { get { return true;
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void DoNotFormatIncompleteBlockOnSingleLineIfNotTypingCloseCurly3()
{
var code = """
namespace ConsoleApplication1
{
class Program
{
static bool Property { get;$$
}
}
""";
var expected = """
namespace ConsoleApplication1
{
class Program
{
static bool Property { get;
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void DoNotFormatCompleteBlockOnSingleLineIfTypingCloseCurly1()
{
var code = """
namespace ConsoleApplication1
{
class Program
{
static bool Property
{
get { return true; }$$
}
""";
var expected = """
namespace ConsoleApplication1
{
class Program
{
static bool Property
{
get { return true; }
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void DoNotFormatCompleteBlockOnSingleLineIfTypingCloseCurly2()
{
var code = """
namespace ConsoleApplication1
{
class Program
{
static bool Property { get { return true; }$$
}
""";
var expected = """
namespace ConsoleApplication1
{
class Program
{
static bool Property { get { return true; }
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void FormatIncompleteBlockOnMultipleLinesIfTypingCloseCurly1()
{
var code = """
namespace ConsoleApplication1
{
class Program
{
static bool Property
{
get { return true;
}$$
}
""";
var expected = """
namespace ConsoleApplication1
{
class Program
{
static bool Property
{
get
{
return true;
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void FormatIncompleteBlockOnMultipleLinesIfTypingCloseCurly2()
{
var code = """
namespace ConsoleApplication1
{
class Program
{
static bool Property
{
get { return true;
}
}$$
""";
var expected = """
namespace ConsoleApplication1
{
class Program
{
static bool Property
{
get
{
return true;
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void DoNotFormatCompleteBlockOnSingleLineIfTypingSemicolon()
{
var code =
"""
public class Class1
{
void M()
{
try { }
catch { return;$$
x.ToString();
}
}
""";
var expected =
"""
public class Class1
{
void M()
{
try { }
catch { return;
x.ToString();
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void FormatCompleteBlockOnSingleLineIfTypingCloseCurlyOnLaterLine()
{
var code =
"""
public class Class1
{
void M()
{
try { }
catch { return;
x.ToString();
}$$
}
}
""";
var expected =
"""
public class Class1
{
void M()
{
try { }
catch
{
return;
x.ToString();
}
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/7900")]
public void FormatLockStatementWithEmbeddedStatementOnSemicolonDifferentLine()
{
var code = """
class C
{
private object _l = new object();
public void M()
{
lock (_l)
Console.WriteLine("d");$$
}
}
""";
var expected = """
class C
{
private object _l = new object();
public void M()
{
lock (_l)
Console.WriteLine("d");
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/7900")]
public void FormatLockStatementWithEmbeddedStatementOnSemicolonSameLine()
{
var code = """
class C
{
private object _l = new object();
public void M()
{
lock (_l) Console.WriteLine("d");$$
}
}
""";
var expected = """
class C
{
private object _l = new object();
public void M()
{
lock (_l) Console.WriteLine("d");
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/11642")]
public void FormatArbitraryNodeParenthesizedLambdaExpression()
{
// code equivalent to an expression synthesized like so:
// ParenthesizedExpression(ParenthesizedLambdaExpression(ParameterList(), Block()))
var code = @"(()=>{})";
var node = SyntaxFactory.ParseExpression(code);
var expected = @"(() => { })";
AssertFormatOnArbitraryNode(node, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/57465")]
public async Task FormatLambdaWithDirective()
{
var code = """
namespace N
{
public class C
{
protected void Render()
{
if (outer)
{
M(() =>
{
#nullable enable
if (inner)
{
}
}
);
}
}
}
}
""";
var expected = """
namespace N
{
public class C
{
protected void Render()
{
if (outer)
{
M(() =>
{
#nullable enable
if (inner)
{
}
}
);
}
}
}
}
""";
await AssertFormatAsync(expected, code, spans: null);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/57465")]
public async Task FormatLambdaWithComment()
{
var code = """
namespace N
{
public class C
{
protected void Render()
{
if (true)
{
M(() =>
{
if (true)
{
/* marker */
}
});
}
}
}
}
""";
var expected = """
namespace N
{
public class C
{
protected void Render()
{
if (true)
{
M(() =>
{
if (true)
{
/* marker */
}
});
}
}
}
}
""";
await AssertFormatAsync(expected, code, spans: null);
}
[WpfTheory]
[CombinatorialData]
[WorkItem("https://github.com/dotnet/roslyn/issues/59637")]
public async Task FormatAttributeAtEndOfFile(bool trailingNewLine)
{
var endOfFile = trailingNewLine ? Environment.NewLine : "";
var code = $@"using System.Diagnostics.CodeAnalysis;
[assembly:SuppressMessage(""Globalization"", ""CA1308: Normalize strings to uppercase"", Justification = ""My reason"", Scope = ""member"", Target = ""~M:Method"") ] {endOfFile}";
var expected = $@"using System.Diagnostics.CodeAnalysis;
[assembly: SuppressMessage(""Globalization"", ""CA1308: Normalize strings to uppercase"", Justification = ""My reason"", Scope = ""member"", Target = ""~M:Method"")]{endOfFile}";
await AssertFormatAsync(expected, code, spans: null);
await AssertFormatAsync(expected, expected, spans: null);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentOpenBraceEvenWithFormatWhileTypingOff1()
{
var code =
"""
class Program
{
void M()
{
if (true)
{$$
}
}
""";
var expected =
"""
class Program
{
void M()
{
if (true)
{
}
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentOpenBraceEvenWithFormatWhileTypingOff2()
{
var code =
"""
class Program
{
void M()
{
if (true)
{}$$
}
}
""";
var expected =
"""
class Program
{
void M()
{
if (true)
{ }
}
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentOpenBraceEvenWithFormatWhileTypingOff3()
{
// We only smart indent the { if it's on it's own line.
var code =
"""
class Program
{
void M()
{
if (true){$$
}
}
""";
var expected =
"""
class Program
{
void M()
{
if (true){
}
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentOpenBraceEvenWithFormatWhileTypingOff4()
{
// We only smart indent the { if it's on it's own line.
var code =
"""
class Program
{
void M()
{
if (true){}$$
}
}
""";
var expected =
"""
class Program
{
void M()
{
if (true){ }
}
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentOpenBraceEvenWithFormatWhileTypingOff5()
{
// Typing the { should not affect the formating of the preceding tokens.
var code =
"""
class Program
{
void M()
{
if ( true )
{$$
}
}
""";
var expected =
"""
class Program
{
void M()
{
if ( true )
{
}
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentOpenBraceEvenWithFormatWhileTypingOff6()
{
// Typing the { should not affect the formating of the preceding tokens.
var code =
"""
class Program
{
void M()
{
if ( true ){$$
}
}
""";
var expected =
"""
class Program
{
void M()
{
if ( true ){
}
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentOpenBraceEvenWithFormatWhileTypingOff7()
{
var code =
"""
class Program
{
void M()
{$$
}
""";
var expected =
"""
class Program
{
void M()
{
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentCloseBraceEvenWithFormatWhileTypingOff1()
{
var code =
"""
class Program
{
void M()
{
if (true)
{
}$$
}
}
""";
var expected =
"""
class Program
{
void M()
{
if (true)
{
}
}
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentCloseBraceEvenWithFormatWhileTypingOff2()
{
// Note that the { is not updated since we are not formatting.
var code =
"""
class Program
{
void M()
{
if (true) {
}$$
}
}
""";
var expected =
"""
class Program
{
void M()
{
if (true) {
}
}
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentCloseBraceEvenWithFormatWhileTypingOff3()
{
var code =
"""
class Program
{
void M()
{
}$$
}
""";
var expected =
"""
class Program
{
void M()
{
}
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30787")]
public void DoSmartIndentCloseBraceEvenWithFormatWhileTypingOff4()
{
// Should not affect formatting of open brace
var code =
"""
class Program
{
void M() {
}$$
}
""";
var expected =
"""
class Program
{
void M() {
}
}
""";
AssertFormatAfterTypeChar(code, expected, SmartIndentButDoNotFormatWhileTyping());
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/31907")]
public async Task NullableReferenceTypes()
{
var code = """
[|
class MyClass
{
void MyMethod()
{
var returnType = (_useMethodSignatureReturnType ? _methodSignatureOpt !: method).ReturnType;
}
}
|]
""";
var expected = """
class MyClass
{
void MyMethod()
{
var returnType = (_useMethodSignatureReturnType ? _methodSignatureOpt! : method).ReturnType;
}
}
""";
await AssertFormatWithBaseIndentAsync(expected, code, baseIndentation: 4);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/30518")]
public void FormatGeneratedNodeInInitializer()
{
var code = """
new bool[] {
true,
true
}
""";
var expected = """
new bool[] {
true,
true == false, true
}
""";
var tree = SyntaxFactory.ParseSyntaxTree(code, options: TestOptions.Script);
var root = tree.GetRoot();
var entry = SyntaxFactory.BinaryExpression(SyntaxKind.EqualsExpression, SyntaxFactory.LiteralExpression(SyntaxKind.TrueLiteralExpression), SyntaxFactory.LiteralExpression(SyntaxKind.FalseLiteralExpression));
var newRoot = root.InsertNodesBefore(root.DescendantNodes().Last(), [entry]);
AssertFormatOnArbitraryNode(newRoot, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/27268")]
public async Task PositionalPattern()
{
var code = """
[|
class MyClass
{
void MyMethod()
{
var point = new Point (3, 4);
if (point is Point (3, 4) _
&& point is Point{x: 3, y: 4} _)
{
}
}
}
|]
""";
var expected = """
class MyClass
{
void MyMethod()
{
var point = new Point(3, 4);
if (point is Point(3, 4) _
&& point is Point { x: 3, y: 4 } _)
{
}
}
}
""";
await AssertFormatWithBaseIndentAsync(expected, code, baseIndentation: 4);
}
[WpfFact]
public async Task WithExpression()
{
var code = """
[|
record C(int Property)
{
void M()
{
_ = this with { Property = 1 } ;
}
}
|]
""";
var expected = """
record C(int Property)
{
void M()
{
_ = this with { Property = 1 };
}
}
""";
await AssertFormatWithBaseIndentAsync(expected, code, baseIndentation: 4);
}
[WpfFact]
public async Task WithExpression_MultiLine()
{
var code = """
[|
record C(int Property, int Property2)
{
void M()
{
_ = this with
{
Property = 1,
Property2 = 2
} ;
}
}
|]
""";
var expected = """
record C(int Property, int Property2)
{
void M()
{
_ = this with
{
Property = 1,
Property2 = 2
};
}
}
""";
await AssertFormatWithBaseIndentAsync(expected, code, baseIndentation: 4);
}
[WpfFact]
public async Task WithExpression_MultiLine_UserPositionedBraces()
{
var code = """
[|
record C(int Property, int Property2)
{
void M()
{
_ = this with
{
Property = 1,
Property2 = 2
} ;
}
}
|]
""";
var expected = """
record C(int Property, int Property2)
{
void M()
{
_ = this with
{
Property = 1,
Property2 = 2
};
}
}
""";
await AssertFormatWithBaseIndentAsync(expected, code, baseIndentation: 4);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/25003")]
public void SeparateGroups_KeepMultipleLinesBetweenGroups()
{
var code = """
$$
using System.A;
using System.B;
using MS.A;
using MS.B;
""";
var expected = """
$$
using System.A;
using System.B;
using MS.A;
using MS.B;
""";
AssertFormatWithView(expected, code, new OptionsCollection(LanguageNames.CSharp) { { GenerationOptions.SeparateImportDirectiveGroups, true } });
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/25003")]
public void SeparateGroups_KeepMultipleLinesBetweenGroups_FileScopedNamespace()
{
var code = """
$$
namespace N;
using System.A;
using System.B;
using MS.A;
using MS.B;
""";
var expected = """
$$
namespace N;
using System.A;
using System.B;
using MS.A;
using MS.B;
""";
AssertFormatWithView(expected, code, new OptionsCollection(LanguageNames.CSharp) { { GenerationOptions.SeparateImportDirectiveGroups, true } });
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/25003")]
public void SeparateGroups_DoNotGroupIfNotSorted()
{
var code = """
$$
using System.B;
using System.A;
using MS.B;
using MS.A;
""";
var expected = """
$$
using System.B;
using System.A;
using MS.B;
using MS.A;
""";
AssertFormatWithView(expected, code, new OptionsCollection(LanguageNames.CSharp) { { GenerationOptions.SeparateImportDirectiveGroups, true } });
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/25003")]
public void SeparateGroups_GroupIfSorted()
{
var code = """
$$
using System.A;
using System.B;
using MS.A;
using MS.B;
""";
var expected = """
$$
using System.A;
using System.B;
using MS.A;
using MS.B;
""";
AssertFormatWithView(expected, code, new OptionsCollection(LanguageNames.CSharp) { { GenerationOptions.SeparateImportDirectiveGroups, true } });
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/25003")]
public void SeparateGroups_GroupIfSorted_RecognizeSystemNotFirst()
{
var code = """
$$
using MS.A;
using MS.B;
using System.A;
using System.B;
""";
var expected = """
$$
using MS.A;
using MS.B;
using System.A;
using System.B;
""";
AssertFormatWithView(expected, code, new OptionsCollection(LanguageNames.CSharp) { { GenerationOptions.SeparateImportDirectiveGroups, true } });
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/58157")]
public void FormatImplicitObjectCollection()
{
var code = """
class Program
{
static void Main(string[] args)
{
$$List<int> list = new()
{
1, 2, 3, 4,
};
}
}
""";
var expected = """
class Program
{
static void Main(string[] args)
{
$$List<int> list = new()
{
1, 2, 3, 4,
};
}
}
""";
AssertFormatWithView(expected, code);
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/49492")]
public void PreserveAnnotationsOnMultiLineTrivia()
{
var text = """
namespace TestApp
{
class Test
{
/* __marker__ */
}
}
""";
var position = text.IndexOf("/* __marker__ */");
var syntaxTree = CSharpSyntaxTree.ParseText(text);
var root = syntaxTree.GetRoot();
var annotation = new SyntaxAnnotation("marker");
var markerTrivia = root.FindTrivia(position, findInsideTrivia: true);
var annotatedMarkerTrivia = markerTrivia.WithAdditionalAnnotations(annotation);
root = root.ReplaceTrivia(markerTrivia, annotatedMarkerTrivia);
using var workspace = new AdhocWorkspace();
var options = CSharpSyntaxFormattingOptions.Default;
var formattedRoot = Formatter.Format(root, workspace.Services.SolutionServices, options, CancellationToken.None);
var annotatedTrivia = formattedRoot.GetAnnotatedTrivia("marker");
Assert.Single(annotatedTrivia);
}
[WpfFact]
public void FormatUserDefinedOperator()
{
var code = """
$$
class C
{
public static C operator + ( C x, C y){
}
}
""";
var expected = """
$$
class C
{
public static C operator +(C x, C y)
{
}
}
""";
AssertFormatWithView(expected, code);
}
[WpfFact]
public void FormatUserDefinedUnaryOperator()
{
var code = """
$$
class C
{
public static C operator ++ ( C x){
}
}
""";
var expected = """
$$
class C
{
public static C operator ++(C x)
{
}
}
""";
AssertFormatWithView(expected, code);
}
[WpfFact]
public void FormatUserDefinedExplicitCastOperator()
{
var code = """
$$
class C
{
public static explicit operator C ( int x){
}
}
""";
var expected = """
$$
class C
{
public static explicit operator C(int x)
{
}
}
""";
AssertFormatWithView(expected, code);
}
[WpfFact]
public void FormatUserDefinedOperatorOnType()
{
var code = """
interface I1
{
abstract static I1 operator + ( I1 x, I1 y);$$
}
""";
var expected = """
interface I1
{
abstract static I1 operator +(I1 x, I1 y);
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void FormatUserDefinedUnaryOperatorOnType()
{
var code = """
interface I1
{
abstract static I1 operator ++ ( I1 x);$$
}
""";
var expected = """
interface I1
{
abstract static I1 operator ++(I1 x);
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void FormatUserDefinedExplicitCastOperatorOnType()
{
var code = """
interface I1<T> where T : I1<T>
{
abstract static explicit operator string ( T x);$$
}
""";
var expected = """
interface I1<T> where T : I1<T>
{
abstract static explicit operator string(T x);
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact]
public void FormatUserDefinedCheckedOperator()
{
var code = """
$$
class C
{
public static C operator checked + ( C x, C y){
}
}
""";
var expected = """
$$
class C
{
public static C operator checked +(C x, C y)
{
}
}
""";
AssertFormatWithView(expected, code, parseOptions: CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp12));
}
[WpfFact]
public void FormatUserDefinedCheckedUnaryOperator()
{
var code = """
$$
class C
{
public static C operator checked ++ ( C x){
}
}
""";
var expected = """
$$
class C
{
public static C operator checked ++(C x)
{
}
}
""";
AssertFormatWithView(expected, code, parseOptions: CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp12));
}
[WpfFact]
public void FormatUserDefinedExplicitCheckedCastOperator()
{
var code = """
$$
class C
{
public static explicit operator checked C ( int x){
}
}
""";
var expected = """
$$
class C
{
public static explicit operator checked C(int x)
{
}
}
""";
AssertFormatWithView(expected, code, parseOptions: CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp12));
}
[WpfFact]
public void FormatUserDefinedCheckedOperatorOnType()
{
var code = """
interface I1
{
abstract static I1 operator checked + ( I1 x, I1 y);$$
}
""";
var expected = """
interface I1
{
abstract static I1 operator checked +(I1 x, I1 y);
}
""";
AssertFormatAfterTypeChar(code, expected, parseOptions: CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp12));
}
[WpfFact]
public void FormatUserDefinedCheckedUnaryOperatorOnType()
{
var code = """
interface I1
{
abstract static I1 operator checked ++ ( I1 x);$$
}
""";
var expected = """
interface I1
{
abstract static I1 operator checked ++(I1 x);
}
""";
AssertFormatAfterTypeChar(code, expected, parseOptions: CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp12));
}
[WpfFact]
public void FormatUserDefinedExplicitCheckedCastOperatorOnType()
{
var code = """
interface I1<T> where T : I1<T>
{
abstract static explicit operator checked string ( T x);$$
}
""";
var expected = """
interface I1<T> where T : I1<T>
{
abstract static explicit operator checked string(T x);
}
""";
AssertFormatAfterTypeChar(code, expected, parseOptions: CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp12));
}
[WpfFact]
public void FormatUnsignedRightShift()
{
var code = """
$$
class C
{
public static C operator>>> ( C x, C y){
}
}
""";
var expected = """
$$
class C
{
public static C operator >>>(C x, C y)
{
}
}
""";
AssertFormatWithView(expected, code);
}
[WpfFact]
public void FormatCollectionExpressionAfterEquals()
{
var code = """
$$
var v = [ 1 , 2 , 3 ] ;
""";
var expected = """
$$
var v = [1, 2, 3];
""";
AssertFormatWithView(expected, code);
}
[WpfFact]
public void FormatCollectionExpressionAfterEquals2()
{
var code = """
class C
{
void M()
{
List<int> list = [ ] ;$$
}
}
""";
var expected = """
class C
{
void M()
{
List<int> list = [];$$
}
}
""";
AssertFormatWithView(expected, code);
}
[WpfFact]
public void FormatUnsignedRightShiftOnType()
{
var code = """
interface I1
{
abstract static I1 operator >>> ( I1 x, I1 y);$$
}
""";
var expected = """
interface I1
{
abstract static I1 operator >>>(I1 x, I1 y);
}
""";
AssertFormatAfterTypeChar(code, expected);
}
[WpfFact, WorkItem("https://github.com/dotnet/roslyn/issues/13981")]
public void FormatLabeledStatementAfterColon()
{
var code = """
class C
{
void M()
{
foo:$$
}
}
""";
var expected = """
class C
{
void M()
{
foo:
}
}
""";
AssertFormatAfterTypeChar(code, expected);
}
private static void AssertFormatAfterTypeChar(
[StringSyntax(PredefinedEmbeddedLanguageNames.CSharpTest)] string code,
[StringSyntax(PredefinedEmbeddedLanguageNames.CSharpTest)] string expected,
OptionsCollection? globalOptions = null,
ParseOptions? parseOptions = null)
{
using var workspace = EditorTestWorkspace.CreateCSharp(code, parseOptions: parseOptions);
var subjectDocument = workspace.Documents.Single();
var commandHandler = workspace.GetService<FormatCommandHandler>();
var typedChar = subjectDocument.GetTextBuffer().CurrentSnapshot.GetText(subjectDocument.CursorPosition!.Value - 1, 1);
var textView = subjectDocument.GetTextView();
globalOptions?.SetGlobalOptions(workspace.GlobalOptions);
workspace.GlobalOptions.SetEditorOptions(textView.Options.GlobalOptions, subjectDocument.Project.Language);
commandHandler.ExecuteCommand(new TypeCharCommandArgs(textView, subjectDocument.GetTextBuffer(), typedChar[0]), () => { }, TestCommandExecutionContext.Create());
var newSnapshot = subjectDocument.GetTextBuffer().CurrentSnapshot;
AssertEx.EqualOrDiff(expected, newSnapshot.GetText());
}
}
|