File: BracePairs\IBracePairsService.cs
Web Access
Project: ..\..\..\src\Features\Core\Portable\Microsoft.CodeAnalysis.Features.csproj (Microsoft.CodeAnalysis.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.Collections.Generic;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.Host;
using Microsoft.CodeAnalysis.LanguageService;
using Microsoft.CodeAnalysis.PooledObjects;
using Microsoft.CodeAnalysis.Text;
 
namespace Microsoft.CodeAnalysis.BracePairs
{
    internal readonly record struct BracePairData(
        TextSpan Start,
        TextSpan End);
 
    internal interface IBracePairsService : ILanguageService
    {
        Task AddBracePairsAsync(Document document, TextSpan textSpan, ArrayBuilder<BracePairData> bracePairs, CancellationToken cancellationToken);
    }
 
    internal abstract class AbstractBracePairsService : IBracePairsService
    {
        private readonly Dictionary<int, int> _bracePairKinds = new();
 
        protected AbstractBracePairsService(
            ISyntaxKinds syntaxKinds)
        {
            Add(syntaxKinds.OpenBraceToken, syntaxKinds.CloseBraceToken);
            Add(syntaxKinds.OpenBracketToken, syntaxKinds.CloseBracketToken);
            Add(syntaxKinds.OpenParenToken, syntaxKinds.CloseParenToken);
            Add(syntaxKinds.LessThanToken, syntaxKinds.GreaterThanToken);
 
            return;
 
            void Add(int? open, int? close)
            {
                if (open != null && close != null)
                    _bracePairKinds[open.Value] = close.Value;
            }
        }
 
        public async Task AddBracePairsAsync(
            Document document, TextSpan span, ArrayBuilder<BracePairData> bracePairs, CancellationToken cancellationToken)
        {
            var root = await document.GetSyntaxRootAsync(cancellationToken).ConfigureAwait(false);
            using var _ = ArrayBuilder<SyntaxNodeOrToken>.GetInstance(out var stack);
 
            stack.Add(root);
 
            while (stack.Count > 0)
            {
                var current = stack.Pop();
                if (current.IsNode)
                {
                    // Ignore nodes that have no intersection at all with the span we're being asked with. Note: if
                    // there is any node intersection, then we want to process it.  We specifically don't check token
                    // intersection as the start token might not be in the span we're asked about, but the close token
                    // may be.
                    if (!current.Span.IntersectsWith(span))
                        continue;
 
                    foreach (var child in current.ChildNodesAndTokens().Reverse())
                        stack.Push(child);
                }
                else if (current.IsToken)
                {
                    if (_bracePairKinds.TryGetValue(current.AsToken().RawKind, out var closeKind))
                    {
                        // hit an open token.  Try to find the corresponding close token in the parent.
                        if (current.Parent != null)
                        {
                            foreach (var sibling in current.Parent.ChildNodesAndTokens())
                            {
                                if (sibling.IsToken && sibling.RawKind == closeKind)
                                    bracePairs.Add(new BracePairData(current.Span, sibling.Span));
                            }
                        }
                    }
                }
            }
        }
    }
}