File: src\Analyzers\CSharp\CodeFixes\UseExpressionBody\UseExpressionBodyCodeFixProvider.cs
Web Access
Project: src\src\Features\CSharp\Portable\Microsoft.CodeAnalysis.CSharp.Features.csproj (Microsoft.CodeAnalysis.CSharp.Features)
// 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.Collections.Generic;
using System.Collections.Immutable;
using System.Composition;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.CodeActions;
using Microsoft.CodeAnalysis.CodeFixes;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using Microsoft.CodeAnalysis.Editing;
using Microsoft.CodeAnalysis.Formatting;
using Microsoft.CodeAnalysis.Host.Mef;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis.CSharp.UseExpressionBody;
 
[ExportCodeFixProvider(LanguageNames.CSharp, Name = PredefinedCodeFixProviderNames.UseExpressionBody), Shared]
internal sealed partial class UseExpressionBodyCodeFixProvider : SyntaxEditorBasedCodeFixProvider
{
    public sealed override ImmutableArray<string> FixableDiagnosticIds { get; }
 
    private static readonly ImmutableArray<UseExpressionBodyHelper> _helpers = UseExpressionBodyHelper.Helpers;
 
    [ImportingConstructor]
    [Obsolete(MefConstruction.ImportingConstructorMessage, error: true)]
    public UseExpressionBodyCodeFixProvider()
        => FixableDiagnosticIds = _helpers.SelectAsArray(h => h.DiagnosticId);
 
    protected override bool IncludeDiagnosticDuringFixAll(Diagnostic diagnostic)
        => !diagnostic.IsSuppressed ||
           diagnostic.Properties.ContainsKey(UseExpressionBodyDiagnosticAnalyzer.FixesError);
 
    public sealed override Task RegisterCodeFixesAsync(CodeFixContext context)
    {
        var diagnostic = context.Diagnostics.First();
 
        var priority = diagnostic.Severity == DiagnosticSeverity.Hidden
            ? CodeActionPriority.Low
            : CodeActionPriority.Default;
 
        var title = diagnostic.GetMessage();
 
        RegisterCodeFix(context, title, title, priority);
        return Task.CompletedTask;
    }
 
    protected override async Task FixAllAsync(
        Document document, ImmutableArray<Diagnostic> diagnostics,
        SyntaxEditor editor, CancellationToken cancellationToken)
    {
        var semanticModel = await document.GetRequiredSemanticModelAsync(cancellationToken).ConfigureAwait(false);
 
        var accessorLists = new HashSet<AccessorListSyntax>();
        foreach (var diagnostic in diagnostics)
        {
            cancellationToken.ThrowIfCancellationRequested();
            AddEdits(semanticModel, editor, diagnostic, accessorLists, cancellationToken);
        }
 
        // Ensure that if we changed any accessors that the accessor lists they're contained
        // in are formatted properly as well.  Do this as a last pass so that we see all
        // individual changes made to the child accessors if we're doing a fix-all.
        foreach (var accessorList in accessorLists)
        {
            editor.ReplaceNode(accessorList, (current, _) => current.WithAdditionalAnnotations(Formatter.Annotation));
        }
    }
 
    private static void AddEdits(
        SemanticModel semanticModel, SyntaxEditor editor, Diagnostic diagnostic,
        HashSet<AccessorListSyntax> accessorLists,
        CancellationToken cancellationToken)
    {
        var declarationLocation = diagnostic.AdditionalLocations[0];
        var helper = _helpers.Single(h => h.DiagnosticId == diagnostic.Id);
        var declaration = declarationLocation.FindNode(getInnermostNodeForTie: true, cancellationToken);
        var useExpressionBody = diagnostic.Properties.ContainsKey(nameof(UseExpressionBody));
 
        var updatedDeclaration = helper.Update(semanticModel, declaration, useExpressionBody, cancellationToken)
                                       .WithAdditionalAnnotations(Formatter.Annotation);
 
        editor.ReplaceNode(declaration, updatedDeclaration);
 
        if (declaration.Parent is AccessorListSyntax accessorList)
        {
            accessorLists.Add(accessorList);
        }
    }
}