File: Completion\CompletionProviders\CrefCompletionProvider.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.Diagnostics.CodeAnalysis;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.Completion;
using Microsoft.CodeAnalysis.Completion.Providers;
using Microsoft.CodeAnalysis.CSharp.Extensions;
using Microsoft.CodeAnalysis.CSharp.Symbols;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using Microsoft.CodeAnalysis.ErrorReporting;
using Microsoft.CodeAnalysis.Host.Mef;
using Microsoft.CodeAnalysis.PooledObjects;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Microsoft.CodeAnalysis.Text;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis.CSharp.Completion.Providers;
 
[ExportCompletionProvider(nameof(CrefCompletionProvider), LanguageNames.CSharp), Shared]
[ExtensionOrder(After = nameof(EnumAndCompletionListTagCompletionProvider))]
[method: ImportingConstructor]
[method: Obsolete(MefConstruction.ImportingConstructorMessage, error: true)]
internal sealed class CrefCompletionProvider() : AbstractCrefCompletionProvider
{
    private static readonly SymbolDisplayFormat QualifiedCrefFormat =
        new(globalNamespaceStyle: SymbolDisplayGlobalNamespaceStyle.Omitted,
            typeQualificationStyle: SymbolDisplayTypeQualificationStyle.NameOnly,
            propertyStyle: SymbolDisplayPropertyStyle.NameOnly,
            genericsOptions: SymbolDisplayGenericsOptions.IncludeTypeParameters,
            parameterOptions: SymbolDisplayParameterOptions.None,
            miscellaneousOptions:
                SymbolDisplayMiscellaneousOptions.EscapeKeywordIdentifiers | SymbolDisplayMiscellaneousOptions.ExpandValueTuple);
 
    private static readonly SymbolDisplayFormat CrefFormat =
        QualifiedCrefFormat.AddMiscellaneousOptions(SymbolDisplayMiscellaneousOptions.UseSpecialTypes);
 
    private static readonly SymbolDisplayFormat MinimalParameterTypeFormat =
        SymbolDisplayFormat.MinimallyQualifiedFormat.AddMiscellaneousOptions(SymbolDisplayMiscellaneousOptions.ExpandValueTuple);
 
    private Action<SyntaxNode?>? _testSpeculativeNodeCallback;
 
    internal override string Language => LanguageNames.CSharp;
 
    public override bool IsInsertionTrigger(SourceText text, int characterPosition, CompletionOptions options)
        => CompletionUtilities.IsTriggerCharacter(text, characterPosition, options);
 
    public override ImmutableHashSet<char> TriggerCharacters { get; } = CompletionUtilities.CommonTriggerCharacters;
 
    public override async Task ProvideCompletionsAsync(CompletionContext context)
    {
        try
        {
            var document = context.Document;
            var position = context.Position;
            var options = context.CompletionOptions;
            var cancellationToken = context.CancellationToken;
 
            var (token, semanticModel, symbols) = await GetSymbolsAsync(document, position, options, cancellationToken).ConfigureAwait(false);
 
            if (symbols.IsDefaultOrEmpty)
                return;
 
            Contract.ThrowIfNull(semanticModel);
 
            context.IsExclusive = true;
 
            var text = await document.GetValueTextAsync(cancellationToken).ConfigureAwait(false);
            var span = GetCompletionItemSpan(text, position);
            var serializedOptions = ImmutableArray.Create(KeyValuePairUtil.Create(HideAdvancedMembers, options.MemberDisplayOptions.HideAdvancedMembers.ToString()));
 
            var items = CreateCompletionItems(semanticModel, symbols, token, position, serializedOptions);
 
            context.AddItems(items);
        }
        catch (Exception e) when (FatalError.ReportAndCatchUnlessCanceled(e, ErrorSeverity.General))
        {
        }
    }
 
    protected override async Task<(SyntaxToken, SemanticModel?, ImmutableArray<ISymbol>)> GetSymbolsAsync(
        Document document, int position, CompletionOptions options, CancellationToken cancellationToken)
    {
        var tree = await document.GetRequiredSyntaxTreeAsync(cancellationToken).ConfigureAwait(false);
        if (!tree.IsEntirelyWithinCrefSyntax(position, cancellationToken))
            return default;
 
        var token = tree.FindTokenOnLeftOfPosition(position, cancellationToken, includeDocumentationComments: true)
                        .GetPreviousTokenIfTouchingWord(position);
 
        // To get a Speculative SemanticModel (which is much faster), we need to 
        // walk up to the node the DocumentationTrivia is attached to.
        var parentNode = token.Parent?.FirstAncestorOrSelf<DocumentationCommentTriviaSyntax>()?.ParentTrivia.Token.Parent;
        _testSpeculativeNodeCallback?.Invoke(parentNode);
        if (parentNode == null)
            return default;
 
        var semanticModel = await document.ReuseExistingSpeculativeModelAsync(
            parentNode, cancellationToken).ConfigureAwait(false);
 
        var symbols = GetSymbols(token, semanticModel, cancellationToken)
            .FilterToVisibleAndBrowsableSymbols(options.MemberDisplayOptions.HideAdvancedMembers, semanticModel.Compilation);
 
        return (token, semanticModel, symbols);
    }
 
    private static bool IsCrefStartContext(SyntaxToken token)
    {
        // cases:
        //   <see cref="|
        //   <see cref='|
 
        return token.Kind() is SyntaxKind.DoubleQuoteToken or SyntaxKind.SingleQuoteToken &&
               token.Parent.IsKind(SyntaxKind.XmlCrefAttribute);
    }
 
    private static bool IsCrefParameterListContext(SyntaxToken token)
    {
        // cases:
        //   <see cref="M(|
        //   <see cref="M(x, |
        //   <see cref="M(x, ref |
        //   <see cref="M(x, out |
        //   <see cref="M[|
        //   <see cref="M[x, |
        //   <see cref="M[x, ref |
        //   <see cref="M[x, out |
 
        if (token.Parent?.Kind() is not (SyntaxKind.CrefParameterList or SyntaxKind.CrefBracketedParameterList))
            return false;
 
        if (token.IsKind(SyntaxKind.OpenParenToken) &&
            token.Parent.IsKind(SyntaxKind.CrefParameterList))
        {
            return true;
        }
 
        if (token.IsKind(SyntaxKind.OpenBracketToken) &&
            token.Parent.IsKind(SyntaxKind.CrefBracketedParameterList))
        {
            return true;
        }
 
        return token is (kind: SyntaxKind.CommaToken or SyntaxKind.RefKeyword or SyntaxKind.OutKeyword);
    }
 
    private static bool IsCrefQualifiedNameContext(SyntaxToken token)
    {
        // cases:
        //   <see cref="x.|
 
        return token.IsKind(SyntaxKind.DotToken)
            && token.Parent.IsKind(SyntaxKind.QualifiedCref);
    }
 
    private static ImmutableArray<ISymbol> GetSymbols(
        SyntaxToken token, SemanticModel semanticModel, CancellationToken cancellationToken)
    {
        if (IsCrefStartContext(token))
            return GetUnqualifiedSymbols(token, semanticModel, cancellationToken);
 
        if (IsCrefParameterListContext(token))
            return semanticModel.LookupNamespacesAndTypes(token.SpanStart);
 
        if (IsCrefQualifiedNameContext(token))
            return GetQualifiedSymbols((QualifiedCrefSyntax)token.Parent!, token, semanticModel, cancellationToken);
 
        return [];
    }
 
    private static ImmutableArray<ISymbol> GetUnqualifiedSymbols(
        SyntaxToken token, SemanticModel semanticModel, CancellationToken cancellationToken)
    {
        using var _ = ArrayBuilder<ISymbol>.GetInstance(out var result);
        result.AddRange(semanticModel.LookupSymbols(token.SpanStart));
 
        // LookupSymbols doesn't return indexers or operators because they can't be referred to by name.
        // So, try to find the innermost type declaration and return its operators and indexers
        var typeDeclaration = token.Parent?.FirstAncestorOrSelf<TypeDeclarationSyntax>();
        if (typeDeclaration != null)
        {
            var type = semanticModel.GetDeclaredSymbol(typeDeclaration, cancellationToken);
            if (type != null)
            {
                foreach (var baseType in type.GetBaseTypesAndThis())
                {
                    foreach (var member in baseType.GetMembers())
                    {
                        if ((member.IsIndexer() || member.IsUserDefinedOperator()) &&
                            member.IsAccessibleWithin(type))
                        {
                            result.Add(member);
                        }
                    }
                }
            }
        }
 
        return result.ToImmutableAndClear();
    }
 
    private static ImmutableArray<ISymbol> GetQualifiedSymbols(
        QualifiedCrefSyntax parent, SyntaxToken token, SemanticModel semanticModel, CancellationToken cancellationToken)
    {
        var leftType = semanticModel.GetTypeInfo(parent.Container, cancellationToken).Type;
        var leftSymbol = semanticModel.GetSymbolInfo(parent.Container, cancellationToken).Symbol;
 
        var container = (leftSymbol ?? leftType) as INamespaceOrTypeSymbol;
 
        using var _ = ArrayBuilder<ISymbol>.GetInstance(out var result);
        result.AddRange(semanticModel.LookupSymbols(token.SpanStart, container));
 
        if (container is INamedTypeSymbol namedTypeContainer)
            result.AddRange(namedTypeContainer.InstanceConstructors);
 
        return result.ToImmutableAndClear();
    }
 
    private static TextSpan GetCompletionItemSpan(SourceText text, int position)
    {
        return CommonCompletionUtilities.GetWordSpan(
            text,
            position,
            ch => CompletionUtilities.IsCompletionItemStartCharacter(ch) || ch == '{',
            ch => CompletionUtilities.IsWordCharacter(ch) || ch is '{' or '}');
    }
 
    private static IEnumerable<CompletionItem> CreateCompletionItems(
        SemanticModel semanticModel, ImmutableArray<ISymbol> symbols, SyntaxToken token, int position, ImmutableArray<KeyValuePair<string, string>> options)
    {
        using var _ = SharedPools.Default<StringBuilder>().GetPooledObject(out var builder);
 
        foreach (var group in symbols.GroupBy(s => s.Name))
        {
            var groupCount = group.Count();
            foreach (var symbol in group.OrderBy(s => s.GetArity()))
            {
                // Include the arity in the sort text so that we show types/methods from least arity to most arity.
                var sortText = $"{symbol.Name}`{symbol.GetArity():000}";
 
                // For every symbol, we create an item that uses the regular CrefFormat,
                // which uses intrinsic type keywords
                yield return CreateItem(semanticModel, symbol, groupCount, token, position, builder, sortText, options, CrefFormat);
                if (TryCreateSpecialTypeItem(semanticModel, symbol, token, position, builder, options, out var item))
                    yield return item;
            }
        }
    }
 
    private static bool TryCreateSpecialTypeItem(
        SemanticModel semanticModel, ISymbol symbol, SyntaxToken token, int position, StringBuilder builder,
        ImmutableArray<KeyValuePair<string, string>> options, [NotNullWhen(true)] out CompletionItem? item)
    {
        // If the type is a SpecialType, create an additional item using 
        // its actual name (as opposed to intrinsic type keyword)
        var typeSymbol = symbol as ITypeSymbol;
        if (typeSymbol.IsSpecialType())
        {
            item = CreateItem(semanticModel, symbol, groupCount: 1, token, position, builder, builder.ToString(), options, QualifiedCrefFormat);
            return true;
        }
 
        item = null;
        return false;
    }
 
    private static CompletionItem CreateItem(
        SemanticModel semanticModel,
        ISymbol symbol,
        int groupCount,
        SyntaxToken token,
        int position,
        StringBuilder builder,
        string sortText,
        ImmutableArray<KeyValuePair<string, string>> options,
        SymbolDisplayFormat unqualifiedCrefFormat)
    {
        builder.Clear();
        if (symbol is INamespaceOrTypeSymbol && token.IsKind(SyntaxKind.DotToken))
        {
            // Handle qualified namespace and type names.
            builder.Append(symbol.ToDisplayString(QualifiedCrefFormat));
        }
        else
        {
            // Handle unqualified namespace and type names, or member names.
 
            builder.Append(symbol.ToMinimalDisplayString(semanticModel, token.SpanStart, unqualifiedCrefFormat));
 
            var parameters = symbol.GetParameters();
 
            // if this has parameters, then add them here.  Otherwise, if this is a method without parameters, but
            // there are overloads of it, then also add the parameters to disambiguate.
            if (parameters.Length > 0 ||
                (symbol is IMethodSymbol && groupCount >= 2))
            {
                // Note: we intentionally don't add the "params" modifier for any parameters.
 
                builder.Append(symbol.IsIndexer() ? '[' : '(');
                builder.AppendJoinedValues(", ", parameters,
                    (p, builder) =>
                    {
                        builder.Append(p.RefKind switch
                        {
                            RefKind.Ref => "ref ",
                            RefKind.Out => "out ",
                            RefKind.In => "in ",
                            RefKind.RefReadOnlyParameter => "ref readonly ",
                            _ => "",
                        });
                        builder.Append(p.Type.ToMinimalDisplayString(semanticModel, position, MinimalParameterTypeFormat));
                    });
                builder.Append(symbol.IsIndexer() ? ']' : ')');
            }
        }
 
        return CreateItemFromBuilder(symbol, position, builder, sortText, options);
    }
 
    private static CompletionItem CreateItemFromBuilder(
        ISymbol symbol, int position, StringBuilder builder, string sortText, ImmutableArray<KeyValuePair<string, string>> options)
    {
        var insertionText = builder
            .Replace('<', '{')
            .Replace('>', '}')
            .ToString();
 
        return SymbolCompletionItem.CreateWithNameAndKind(
            displayText: insertionText,
            displayTextSuffix: "",
            insertionText: insertionText,
            symbols: [symbol],
            contextPosition: position,
            sortText: sortText,
            filterText: insertionText,
            properties: options,
            rules: GetRules(insertionText));
    }
 
    private static readonly CharacterSetModificationRule s_WithoutOpenBrace = CharacterSetModificationRule.Create(CharacterSetModificationKind.Remove, '{');
    private static readonly CharacterSetModificationRule s_WithoutOpenParen = CharacterSetModificationRule.Create(CharacterSetModificationKind.Remove, '(');
 
    private static CompletionItemRules GetRules(string displayText)
    {
        var commitRules = ImmutableArray<CharacterSetModificationRule>.Empty;
 
        if (displayText.Contains("{"))
        {
            commitRules = commitRules.Add(s_WithoutOpenBrace);
        }
 
        if (displayText.Contains("("))
        {
            commitRules = commitRules.Add(s_WithoutOpenParen);
        }
 
        if (commitRules.IsEmpty)
        {
            return CompletionItemRules.Default;
        }
        else
        {
            return CompletionItemRules.Default.WithCommitCharacterRules(commitRules);
        }
    }
 
    protected override Task<TextChange?> GetTextChangeAsync(CompletionItem selectedItem, char? ch, CancellationToken cancellationToken)
    {
        if (!SymbolCompletionItem.TryGetInsertionText(selectedItem, out var insertionText))
        {
            insertionText = selectedItem.DisplayText;
        }
 
        return Task.FromResult<TextChange?>(new TextChange(selectedItem.Span, insertionText));
    }
 
    internal TestAccessor GetTestAccessor()
        => new(this);
 
    internal readonly struct TestAccessor(CrefCompletionProvider crefCompletionProvider)
    {
        private readonly CrefCompletionProvider _crefCompletionProvider = crefCompletionProvider;
 
        public void SetSpeculativeNodeCallback(Action<SyntaxNode?> value)
            => _crefCompletionProvider._testSpeculativeNodeCallback = value;
    }
}