-
Notifications
You must be signed in to change notification settings - Fork 4.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Continued Extract method cleanup #76577
Changes from 21 commits
988a92e
9a212ad
d2b41d1
412e22a
67a613e
b297641
4c4c7dd
9578118
2893412
da25b2f
b8addf2
cbc7ca8
40f32b8
f1e3461
652a504
e00bb87
432ba07
c9f7a0c
d86f8a3
f16cf64
2b8801a
cc6848f
f11598a
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -16,16 +16,13 @@ namespace Microsoft.CodeAnalysis.CSharp.ExtractMethod; | |
[method: ImportingConstructor] | ||
[method: SuppressMessage("RoslynDiagnosticsReliability", "RS0033:Importing constructor should be [Obsolete]", Justification = "Used in test code: https://github.com/dotnet/roslyn/issues/42814")] | ||
internal sealed partial class CSharpExtractMethodService() : AbstractExtractMethodService< | ||
CSharpExtractMethodService.CSharpSelectionValidator, | ||
CSharpExtractMethodService.CSharpMethodExtractor, | ||
CSharpExtractMethodService.CSharpSelectionResult, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. got rid of theese generics. they are not needed, and just add type system complexity. |
||
StatementSyntax, | ||
StatementSyntax, | ||
ExpressionSyntax> | ||
{ | ||
protected override CSharpSelectionValidator CreateSelectionValidator(SemanticDocument document, TextSpan textSpan, bool localFunction) | ||
=> new(document, textSpan, localFunction); | ||
protected override SelectionValidator CreateSelectionValidator(SemanticDocument document, TextSpan textSpan, bool localFunction) | ||
=> new CSharpSelectionValidator(document, textSpan, localFunction); | ||
|
||
protected override CSharpMethodExtractor CreateMethodExtractor(CSharpSelectionResult selectionResult, ExtractMethodGenerationOptions options, bool localFunction) | ||
=> new(selectionResult, options, localFunction); | ||
protected override MethodExtractor CreateMethodExtractor(SelectionResult selectionResult, ExtractMethodGenerationOptions options, bool localFunction) | ||
=> new CSharpMethodExtractor(selectionResult, options, localFunction); | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -15,14 +15,9 @@ internal sealed partial class CSharpExtractMethodService | |
{ | ||
internal sealed partial class CSharpMethodExtractor | ||
{ | ||
private sealed class CSharpAnalyzer(CSharpSelectionResult selectionResult, bool localFunction, CancellationToken cancellationToken) : Analyzer(selectionResult, localFunction, cancellationToken) | ||
private sealed class CSharpAnalyzer(SelectionResult selectionResult, bool localFunction, CancellationToken cancellationToken) | ||
: Analyzer(selectionResult, localFunction, cancellationToken) | ||
{ | ||
public static AnalyzerResult Analyze(CSharpSelectionResult selectionResult, bool localFunction, CancellationToken cancellationToken) | ||
{ | ||
var analyzer = new CSharpAnalyzer(selectionResult, localFunction, cancellationToken); | ||
return analyzer.Analyze(); | ||
} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. inlined these static construction methods at callsite. |
||
|
||
protected override bool TreatOutAsRef | ||
=> false; | ||
|
||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -44,20 +44,8 @@ private abstract partial class CSharpCodeGenerator : CodeGenerator<StatementSynt | |
private const string NewMethodPascalCaseStr = "NewMethod"; | ||
private const string NewMethodCamelCaseStr = "newMethod"; | ||
|
||
public static Task<GeneratedCode> GenerateAsync( | ||
InsertionPoint insertionPoint, | ||
CSharpSelectionResult selectionResult, | ||
AnalyzerResult analyzerResult, | ||
ExtractMethodGenerationOptions options, | ||
bool localFunction, | ||
CancellationToken cancellationToken) | ||
{ | ||
var codeGenerator = Create(selectionResult, analyzerResult, options, localFunction); | ||
return codeGenerator.GenerateAsync(insertionPoint, cancellationToken); | ||
} | ||
|
||
public static CSharpCodeGenerator Create( | ||
CSharpSelectionResult selectionResult, | ||
SelectionResult selectionResult, | ||
AnalyzerResult analyzerResult, | ||
ExtractMethodGenerationOptions options, | ||
bool localFunction) | ||
|
@@ -72,7 +60,7 @@ public static CSharpCodeGenerator Create( | |
} | ||
|
||
protected CSharpCodeGenerator( | ||
CSharpSelectionResult selectionResult, | ||
SelectionResult selectionResult, | ||
AnalyzerResult analyzerResult, | ||
ExtractMethodGenerationOptions options, | ||
bool localFunction) | ||
|
@@ -202,9 +190,25 @@ protected override SyntaxNode GetCallSiteContainerFromOutermostMoveInVariable(Ca | |
return declStatement.Parent; | ||
} | ||
|
||
private bool ShouldPutUnsafeModifier() | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this used to be on the strongly typed CSharpXXX types, but were only ever called from a single location (here). so instead of needing all the generic type system wonkyness to pass in one of these subtypes, just to call special methods (not on the base type), i just moved the mtehods to the places that use them. simple, easy. |
||
{ | ||
var token = this.SelectionResult.GetFirstTokenInSelection(); | ||
var ancestors = token.GetAncestors<SyntaxNode>(); | ||
|
||
// if enclosing type contains unsafe keyword, we don't need to put it again | ||
if (ancestors.Where(a => CSharp.SyntaxFacts.IsTypeDeclaration(a.Kind())) | ||
.Cast<MemberDeclarationSyntax>() | ||
.Any(m => m.GetModifiers().Any(SyntaxKind.UnsafeKeyword))) | ||
{ | ||
return false; | ||
} | ||
|
||
return token.Parent.IsUnsafeContext(); | ||
} | ||
|
||
private DeclarationModifiers CreateMethodModifiers() | ||
{ | ||
var isUnsafe = this.SelectionResult.ShouldPutUnsafeModifier(); | ||
var isUnsafe = ShouldPutUnsafeModifier(); | ||
var isAsync = this.SelectionResult.CreateAsyncMethod(); | ||
var isStatic = !AnalyzerResult.UseInstanceMember; | ||
var isReadOnly = AnalyzerResult.ShouldBeReadOnly; | ||
|
@@ -269,9 +273,27 @@ private ImmutableArray<StatementSyntax> CreateMethodBody( | |
return statements; | ||
} | ||
|
||
protected SyntaxKind UnderCheckedExpressionContext() | ||
=> UnderCheckedContext<CheckedExpressionSyntax>(); | ||
|
||
protected SyntaxKind UnderCheckedStatementContext() | ||
=> UnderCheckedContext<CheckedStatementSyntax>(); | ||
|
||
protected SyntaxKind UnderCheckedContext<T>() where T : SyntaxNode | ||
{ | ||
var token = this.SelectionResult.GetFirstTokenInSelection(); | ||
var contextNode = token.Parent.GetAncestor<T>(); | ||
if (contextNode == null) | ||
{ | ||
return SyntaxKind.None; | ||
} | ||
|
||
return contextNode.Kind(); | ||
} | ||
|
||
private ImmutableArray<StatementSyntax> WrapInCheckStatementIfNeeded(ImmutableArray<StatementSyntax> statements) | ||
{ | ||
var kind = this.SelectionResult.UnderCheckedStatementContext(); | ||
var kind = UnderCheckedStatementContext(); | ||
if (kind == SyntaxKind.None) | ||
return statements; | ||
|
||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -19,23 +19,9 @@ internal sealed partial class CSharpExtractMethodService | |
{ | ||
internal sealed partial class CSharpMethodExtractor | ||
{ | ||
private sealed class CSharpTriviaResult : TriviaResult | ||
private sealed class CSharpTriviaResult(SemanticDocument document, ITriviaSavedResult result) | ||
: TriviaResult(document, result, (int)SyntaxKind.EndOfLineTrivia, (int)SyntaxKind.WhitespaceTrivia) | ||
{ | ||
public static async Task<CSharpTriviaResult> ProcessAsync(CSharpSelectionResult selectionResult, CancellationToken cancellationToken) | ||
{ | ||
var preservationService = selectionResult.SemanticDocument.Document.Project.Services.GetService<ISyntaxTriviaService>(); | ||
var root = selectionResult.SemanticDocument.Root; | ||
var result = preservationService.SaveTriviaAroundSelection(root, selectionResult.FinalSpan); | ||
return new CSharpTriviaResult( | ||
await selectionResult.SemanticDocument.WithSyntaxRootAsync(result.Root, cancellationToken).ConfigureAwait(false), | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. static factories inlined to callsite. |
||
result); | ||
} | ||
|
||
private CSharpTriviaResult(SemanticDocument document, ITriviaSavedResult result) | ||
: base(document, result, (int)SyntaxKind.EndOfLineTrivia, (int)SyntaxKind.WhitespaceTrivia) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. primary constructorified. |
||
{ | ||
} | ||
|
||
protected override AnnotationResolver GetAnnotationResolver(SyntaxNode callsite, SyntaxNode method) | ||
{ | ||
var isMethodOrLocalFunction = method is MethodDeclarationSyntax or LocalFunctionStatementSyntax; | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -19,19 +19,23 @@ namespace Microsoft.CodeAnalysis.CSharp.ExtractMethod; | |
|
||
internal sealed partial class CSharpExtractMethodService | ||
{ | ||
internal sealed partial class CSharpMethodExtractor(CSharpSelectionResult result, ExtractMethodGenerationOptions options, bool localFunction) | ||
internal sealed partial class CSharpMethodExtractor( | ||
SelectionResult result, ExtractMethodGenerationOptions options, bool localFunction) | ||
: MethodExtractor(result, options, localFunction) | ||
{ | ||
protected override CodeGenerator CreateCodeGenerator(AnalyzerResult analyzerResult) | ||
=> CSharpCodeGenerator.Create(this.OriginalSelectionResult, analyzerResult, this.Options, this.LocalFunction); | ||
|
||
protected override AnalyzerResult Analyze(CSharpSelectionResult selectionResult, bool localFunction, CancellationToken cancellationToken) | ||
=> CSharpAnalyzer.Analyze(selectionResult, localFunction, cancellationToken); | ||
protected override AnalyzerResult Analyze(CancellationToken cancellationToken) | ||
{ | ||
var analyzer = new CSharpAnalyzer(this.OriginalSelectionResult, this.LocalFunction, cancellationToken); | ||
return analyzer.Analyze(); | ||
} | ||
|
||
protected override SyntaxNode GetInsertionPointNode( | ||
AnalyzerResult analyzerResult, CancellationToken cancellationToken) | ||
{ | ||
var originalSpanStart = OriginalSelectionResult.OriginalSpan.Start; | ||
var originalSpanStart = OriginalSelectionResult.FinalSpan.Start; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. unified a lot of the behavior around which pieces of data we use. the user's original selection, or what we updated the selection to. Note: the followup PR on this cleans this up much further, but separating out this data into two types so that the feature itself is very clear about if it's using the pre/post selection and why. |
||
Contract.ThrowIfFalse(originalSpanStart >= 0); | ||
|
||
var document = this.OriginalSelectionResult.SemanticDocument; | ||
|
@@ -61,10 +65,10 @@ protected override SyntaxNode GetInsertionPointNode( | |
{ | ||
if (currentNode is AnonymousFunctionExpressionSyntax anonymousFunction) | ||
{ | ||
if (OriginalSelectionWithin(anonymousFunction.Body) || OriginalSelectionWithin(anonymousFunction.ExpressionBody)) | ||
if (SelectionWithin(anonymousFunction.Body) || SelectionWithin(anonymousFunction.ExpressionBody)) | ||
return currentNode; | ||
|
||
if (!OriginalSelectionResult.OriginalSpan.Contains(anonymousFunction.Span)) | ||
if (!OriginalSelectionResult.FinalSpan.Contains(anonymousFunction.Span)) | ||
{ | ||
// If we encountered a function but the selection isn't within the body, it's likely the user | ||
// is attempting to move the function (which is behavior that is supported). Stop looking up the | ||
|
@@ -76,10 +80,10 @@ protected override SyntaxNode GetInsertionPointNode( | |
|
||
if (currentNode is LocalFunctionStatementSyntax localFunction) | ||
{ | ||
if (OriginalSelectionWithin(localFunction.ExpressionBody) || OriginalSelectionWithin(localFunction.Body)) | ||
if (SelectionWithin(localFunction.ExpressionBody) || SelectionWithin(localFunction.Body)) | ||
return currentNode; | ||
|
||
if (!OriginalSelectionResult.OriginalSpan.Contains(localFunction.Span)) | ||
if (!OriginalSelectionResult.FinalSpan.Contains(localFunction.Span)) | ||
{ | ||
// If we encountered a function but the selection isn't within the body, it's likely the user | ||
// is attempting to move the function (which is behavior that is supported). Stop looking up the | ||
|
@@ -146,21 +150,31 @@ SyntaxNode GetInsertionPointForGlobalStatement(GlobalStatementSyntax globalState | |
} | ||
} | ||
|
||
private bool OriginalSelectionWithin(SyntaxNode node) | ||
private bool SelectionWithin(SyntaxNode node) | ||
{ | ||
if (node is null) | ||
{ | ||
return false; | ||
} | ||
|
||
return node.Span.Contains(OriginalSelectionResult.OriginalSpan); | ||
return node.Span.Contains(OriginalSelectionResult.FinalSpan); | ||
} | ||
|
||
protected override async Task<TriviaResult> PreserveTriviaAsync(CSharpSelectionResult selectionResult, CancellationToken cancellationToken) | ||
=> await CSharpTriviaResult.ProcessAsync(selectionResult, cancellationToken).ConfigureAwait(false); | ||
protected override async Task<TriviaResult> PreserveTriviaAsync(SyntaxNode root, CancellationToken cancellationToken) | ||
{ | ||
var semanticDocument = this.OriginalSelectionResult.SemanticDocument; | ||
var preservationService = semanticDocument.Document.Project.Services.GetService<ISyntaxTriviaService>(); | ||
var result = preservationService.SaveTriviaAroundSelection(root, this.OriginalSelectionResult.FinalSpan); | ||
return new CSharpTriviaResult( | ||
await semanticDocument.WithSyntaxRootAsync(result.Root, cancellationToken).ConfigureAwait(false), | ||
result); | ||
} | ||
|
||
protected override Task<GeneratedCode> GenerateCodeAsync(InsertionPoint insertionPoint, CSharpSelectionResult selectionResult, AnalyzerResult analyzeResult, ExtractMethodGenerationOptions options, CancellationToken cancellationToken) | ||
=> CSharpCodeGenerator.GenerateAsync(insertionPoint, selectionResult, analyzeResult, options, LocalFunction, cancellationToken); | ||
protected override Task<GeneratedCode> GenerateCodeAsync(InsertionPoint insertionPoint, SelectionResult selectionResult, AnalyzerResult analyzeResult, ExtractMethodGenerationOptions options, CancellationToken cancellationToken) | ||
{ | ||
var codeGenerator = CSharpCodeGenerator.Create(selectionResult, analyzeResult, options, this.LocalFunction); | ||
return codeGenerator.GenerateAsync(insertionPoint, cancellationToken); | ||
} | ||
|
||
protected override AbstractFormattingRule GetCustomFormattingRule(Document document) | ||
=> FormattingRule.Instance; | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -24,10 +24,9 @@ internal abstract partial class CSharpSelectionResult | |
private sealed class ExpressionResult( | ||
SemanticDocument document, | ||
SelectionType selectionType, | ||
TextSpan originalSpan, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. remove this. this should only be the starting info from the user. we don't want to pass this along as it's actively confusing as to when you'd use this over 'final span'. |
||
TextSpan finalSpan, | ||
bool selectionChanged) : CSharpSelectionResult( | ||
document, selectionType, originalSpan, finalSpan, selectionChanged) | ||
document, selectionType, finalSpan, selectionChanged) | ||
{ | ||
public override bool ContainingScopeHasAsyncKeyword() | ||
=> false; | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
note: the test previously were non-functional. they were not actually validating that the spans unless a special bit was set on a reseult avlue (which wasn't computed properly). I also remove this bit in a follow up pr because the whole "feature adds special out of band data, for tests to validate, but which doesn't impactthe feature itself" is an awful paradigm.