File: RemoveUnusedVariable\CSharpRemoveUnusedVariableCodeFixProvider.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.
 
#nullable disable
 
using System.Collections.Immutable;
using System.Composition;
using System.Diagnostics.CodeAnalysis;
using Microsoft.CodeAnalysis.CodeFixes;
using Microsoft.CodeAnalysis.CSharp.Extensions;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using Microsoft.CodeAnalysis.Editing;
using Microsoft.CodeAnalysis.LanguageService;
using Microsoft.CodeAnalysis.RemoveUnusedVariable;
 
namespace Microsoft.CodeAnalysis.CSharp.RemoveUnusedVariable;
 
[ExportCodeFixProvider(LanguageNames.CSharp, Name = PredefinedCodeFixProviderNames.RemoveUnusedVariable), Shared]
[ExtensionOrder(After = PredefinedCodeFixProviderNames.AddImport)]
internal sealed partial class CSharpRemoveUnusedVariableCodeFixProvider : AbstractRemoveUnusedVariableCodeFixProvider<LocalDeclarationStatementSyntax, VariableDeclaratorSyntax, VariableDeclarationSyntax>
{
    public const string CS0168 = nameof(CS0168);
    public const string CS0219 = nameof(CS0219);
 
    [ImportingConstructor]
    [SuppressMessage("RoslynDiagnosticsReliability", "RS0033:Importing constructor should be [Obsolete]", Justification = "Used in test code: https://github.com/dotnet/roslyn/issues/42814")]
    public CSharpRemoveUnusedVariableCodeFixProvider()
    {
    }
 
    public sealed override ImmutableArray<string> FixableDiagnosticIds
        => [CS0168, CS0219];
 
    protected override bool IsCatchDeclarationIdentifier(SyntaxToken token)
        => token.Parent is CatchDeclarationSyntax catchDeclaration && catchDeclaration.Identifier == token;
 
    protected override SyntaxNode GetNodeToRemoveOrReplace(SyntaxNode node)
    {
        node = node.Parent;
        if (node.Kind() == SyntaxKind.SimpleAssignmentExpression)
        {
            var parent = node.Parent;
            if (parent.Kind() == SyntaxKind.ExpressionStatement)
            {
                return parent;
            }
            else
            {
                return node;
            }
        }
 
        return null;
    }
 
    protected override void RemoveOrReplaceNode(SyntaxEditor editor, SyntaxNode node, IBlockFactsService blockFacts)
    {
        switch (node.Kind())
        {
            case SyntaxKind.SimpleAssignmentExpression:
                editor.ReplaceNode(node, ((AssignmentExpressionSyntax)node).Right);
                return;
            default:
                RemoveNode(editor, node.IsParentKind(SyntaxKind.GlobalStatement) ? node.Parent : node, blockFacts);
                return;
        }
    }
 
    protected override SeparatedSyntaxList<SyntaxNode> GetVariables(LocalDeclarationStatementSyntax localDeclarationStatement)
        => localDeclarationStatement.Declaration.Variables;
 
    protected override bool ShouldOfferFixForLocalDeclaration(IBlockFactsService blockFacts, SyntaxNode node)
    {
        // If the fix location is not for a local declaration then we can allow it (eg, when inside a for
        // or catch).
        if (node.Parent?.Parent is not LocalDeclarationStatementSyntax localDeclaration)
            return true;
 
        // Local declarations must be parented by an executable block, or global statement, otherwise
        // removing them would be invalid (and more than likely crash the fixer)
        return blockFacts.GetImmediateParentExecutableBlockForStatement(localDeclaration) != null;
    }
}