File: src\Workspaces\SharedUtilitiesAndExtensions\Compiler\Core\Extensions\TextSpanExtensions.cs
Web Access
Project: src\src\Workspaces\Core\Portable\Microsoft.CodeAnalysis.Workspaces.csproj (Microsoft.CodeAnalysis.Workspaces)
// 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.Linq;
using Microsoft.CodeAnalysis.Text;
 
namespace Microsoft.CodeAnalysis.Shared.Extensions;
 
internal static class TextSpanExtensions
{
    /// <summary>
    /// merge provided spans to each distinct group of spans in ascending order
    /// </summary>
    public static IEnumerable<TextSpan> ToNormalizedSpans(this IEnumerable<TextSpan> spans)
        => new NormalizedTextSpanCollection(spans);
 
    public static ImmutableArray<TextSpan> ToNormalizedSpans(this ImmutableArray<TextSpan> spans)
        => [.. spans];
 
    public static TextSpan Collapse(this IEnumerable<TextSpan> spans)
    {
        var start = int.MaxValue;
        var end = 0;
 
        foreach (var span in spans)
        {
            if (span.Start < start)
            {
                start = span.Start;
            }
 
            if (span.End > end)
            {
                end = span.End;
            }
        }
 
        if (start > end)
        {
            // there were no changes.
            return default;
        }
 
        return TextSpan.FromBounds(start, end);
    }
 
    /// <summary>
    /// Returns true if the span encompasses the specified node or token and is contained within its trivia.
    /// </summary>
    public static bool IsAround(this TextSpan span, SyntaxNodeOrToken node) => IsAround(span, node, node);
 
    /// <summary>
    /// Returns true if the span encompasses a span between the specified nodes or tokens
    /// and is contained within trivia around them.
    /// </summary>
    public static bool IsAround(this TextSpan span, SyntaxNodeOrToken startNode, SyntaxNodeOrToken endNode)
    {
        var innerSpan = TextSpan.FromBounds(startNode.Span.Start, endNode.Span.End);
        var outerSpan = TextSpan.FromBounds(startNode.FullSpan.Start, endNode.FullSpan.End);
        return span.Contains(innerSpan) && outerSpan.Contains(span);
    }
 
    public static IEnumerable<TextSpan> Subtract(this TextSpan span, TextSpan except)
    {
        if (except.IsEmpty)
        {
            if (span != except)
            {
                yield return span;
            }
 
            yield break;
        }
 
        var startSegmentEnd = Math.Min(span.End, except.Start);
        if (span.Start < startSegmentEnd)
            yield return TextSpan.FromBounds(span.Start, startSegmentEnd);
 
        var endSegmentStart = Math.Max(span.Start, except.End);
        if (endSegmentStart < span.End)
            yield return TextSpan.FromBounds(endSegmentStart, span.End);
    }
 
    public static IEnumerable<TextSpan> Subtract(this IEnumerable<TextSpan> spans, TextSpan except)
        => spans.SelectMany(span => span.Subtract(except));
}