File: Formatting\FormatCommandHandler.cs
Web Access
Project: src\src\EditorFeatures\Core\Microsoft.CodeAnalysis.EditorFeatures.csproj (Microsoft.CodeAnalysis.EditorFeatures)
// 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.ComponentModel.Composition;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.Editor;
using Microsoft.CodeAnalysis.Editor.Shared.Extensions;
using Microsoft.CodeAnalysis.Editor.Shared.Utilities;
using Microsoft.CodeAnalysis.Formatting;
using Microsoft.CodeAnalysis.Formatting.Rules;
using Microsoft.CodeAnalysis.Host.Mef;
using Microsoft.CodeAnalysis.Internal.Log;
using Microsoft.CodeAnalysis.Options;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Microsoft.CodeAnalysis.Text;
using Microsoft.VisualStudio.Commanding;
using Microsoft.VisualStudio.Language.Intellisense.AsyncCompletion;
using Microsoft.VisualStudio.Text;
using Microsoft.VisualStudio.Text.Editor;
using Microsoft.VisualStudio.Text.Editor.Commanding;
using Microsoft.VisualStudio.Text.Editor.Commanding.Commands;
using Microsoft.VisualStudio.Text.Operations;
using Microsoft.VisualStudio.Utilities;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis.Formatting;
 
[Export]
[Export(typeof(ICommandHandler))]
[ContentType(ContentTypeNames.RoslynContentType)]
[Name(PredefinedCommandHandlerNames.FormatDocument)]
[Order(After = PredefinedCommandHandlerNames.Rename)]
[Order(Before = PredefinedCommandHandlerNames.StringCopyPaste)]
[Order(Before = PredefinedCompletionNames.CompletionCommandHandler)]
[method: ImportingConstructor]
[method: Obsolete(MefConstruction.ImportingConstructorMessage, error: true)]
internal sealed partial class FormatCommandHandler(
    IThreadingContext threadingContext,
    ITextUndoHistoryRegistry undoHistoryRegistry,
    IEditorOperationsFactoryService editorOperationsFactoryService,
    IGlobalOptionService globalOptions) :
    ICommandHandler<FormatDocumentCommandArgs>,
    ICommandHandler<FormatSelectionCommandArgs>,
    IChainedCommandHandler<PasteCommandArgs>,
    IChainedCommandHandler<TypeCharCommandArgs>,
    IChainedCommandHandler<ReturnKeyCommandArgs>
{
    private readonly IThreadingContext _threadingContext = threadingContext;
    private readonly ITextUndoHistoryRegistry _undoHistoryRegistry = undoHistoryRegistry;
    private readonly IEditorOperationsFactoryService _editorOperationsFactoryService = editorOperationsFactoryService;
    private readonly IGlobalOptionService _globalOptions = globalOptions;
 
    public string DisplayName => EditorFeaturesResources.Automatic_Formatting;
 
    private void Format(ITextView textView, ITextBuffer textBuffer, Document document, TextSpan? selectionOpt, CancellationToken cancellationToken)
    {
        using (Logger.LogBlock(FunctionId.CommandHandler_FormatCommand, KeyValueLogMessage.Create(LogType.UserAction, m => m["Span"] = selectionOpt?.Length ?? -1), cancellationToken))
        using (var transaction = CreateEditTransaction(textView, EditorFeaturesResources.Formatting))
        {
            _threadingContext.JoinableTaskFactory.Run(() => FormatAsync(
                textBuffer, document, selectionOpt, cancellationToken));
 
            transaction.Complete();
        }
    }
 
    private static async Task FormatAsync(
        ITextBuffer textBuffer, Document document, TextSpan? selectionOpt, CancellationToken cancellationToken)
    {
        var formattingService = document.GetRequiredLanguageService<IFormattingInteractionService>();
 
        // Note: C# always completes synchronously, TypeScript is async
        var changes = await formattingService.GetFormattingChangesAsync(
            document, textBuffer, selectionOpt, cancellationToken).ConfigureAwait(true);
        if (changes.IsEmpty)
            return;
 
        if (selectionOpt.HasValue)
        {
            var ruleFactory = document.Project.Solution.Services.GetRequiredService<IHostDependentFormattingRuleFactoryService>();
            changes = [.. ruleFactory.FilterFormattedChanges(document.Id, selectionOpt.Value, changes)];
        }
 
        if (!changes.IsEmpty)
        {
            using (Logger.LogBlock(FunctionId.Formatting_ApplyResultToBuffer, cancellationToken))
            {
                textBuffer.ApplyChanges(changes);
            }
        }
    }
 
    private static bool CanExecuteCommand(ITextBuffer buffer)
        => buffer.CanApplyChangeDocumentToWorkspace();
 
    private static CommandState GetCommandState(ITextBuffer buffer)
        => CanExecuteCommand(buffer) ? CommandState.Available : CommandState.Unspecified;
 
    public void ExecuteReturnOrTypeCommand(EditorCommandArgs args, Action nextHandler, CancellationToken cancellationToken)
    {
        // run next handler first so that editor has chance to put the return into the buffer first.
        nextHandler();
        if (cancellationToken.IsCancellationRequested)
        {
            return;
        }
 
        try
        {
            _threadingContext.JoinableTaskFactory.Run(() => ExecuteReturnOrTypeCommandWorkerAsync(args, cancellationToken));
        }
        catch (OperationCanceledException)
        {
            // According to Editor command handler API guidelines, it's best if we return early if cancellation
            // is requested instead of throwing. Otherwise, we could end up in an invalid state due to already
            // calling nextHandler().
        }
    }
 
    private async Task ExecuteReturnOrTypeCommandWorkerAsync(EditorCommandArgs args, CancellationToken cancellationToken)
    {
        var textView = args.TextView;
        var subjectBuffer = args.SubjectBuffer;
        if (!CanExecuteCommand(subjectBuffer))
            return;
 
        var caretPosition = textView.GetCaretPoint(args.SubjectBuffer);
        if (!caretPosition.HasValue)
            return;
 
        var document = subjectBuffer.CurrentSnapshot.GetOpenDocumentInCurrentContextWithChanges();
        if (document == null)
            return;
 
        var service = document.GetLanguageService<IFormattingInteractionService>();
        if (service == null)
            return;
 
        IList<TextChange>? textChanges;
 
        // save current caret position
        if (args is ReturnKeyCommandArgs)
        {
            if (!service.SupportsFormatOnReturn)
                return;
 
            // Note: C# always completes synchronously, TypeScript is async
            textChanges = await service.GetFormattingChangesOnReturnAsync(
                document, caretPosition.Value, cancellationToken).ConfigureAwait(true);
        }
        else if (args is TypeCharCommandArgs typeCharArgs)
        {
            if (!service.SupportsFormattingOnTypedCharacter(document, typeCharArgs.TypedChar))
            {
                return;
            }
 
            // Note: C# always completes synchronously, TypeScript is async
            textChanges = await service.GetFormattingChangesAsync(
                document, typeCharArgs.SubjectBuffer, typeCharArgs.TypedChar, caretPosition.Value, cancellationToken).ConfigureAwait(true);
        }
        else
        {
            throw ExceptionUtilities.UnexpectedValue(args);
        }
 
        if (textChanges == null || textChanges.Count == 0)
            return;
 
        using (var transaction = CreateEditTransaction(textView, EditorFeaturesResources.Automatic_Formatting))
        {
            transaction.MergePolicy = AutomaticCodeChangeMergePolicy.Instance;
            subjectBuffer.ApplyChanges(textChanges);
            transaction.Complete();
        }
 
        // get new caret position after formatting
        var newCaretPositionMarker = args.TextView.GetCaretPoint(args.SubjectBuffer);
        if (!newCaretPositionMarker.HasValue)
            return;
 
        var snapshotAfterFormatting = subjectBuffer.CurrentSnapshot;
 
        var oldCaretPosition = caretPosition.Value.TranslateTo(snapshotAfterFormatting, PointTrackingMode.Negative);
        var newCaretPosition = newCaretPositionMarker.Value.TranslateTo(snapshotAfterFormatting, PointTrackingMode.Negative);
        if (oldCaretPosition.Position == newCaretPosition.Position)
            return;
 
        // caret has moved to wrong position, move it back to correct position
        args.TextView.TryMoveCaretToAndEnsureVisible(oldCaretPosition);
    }
 
    private CaretPreservingEditTransaction CreateEditTransaction(ITextView view, string description)
        => new(description, view, _undoHistoryRegistry, _editorOperationsFactoryService);
}