File: Completion\CSharpCompletionService.cs
Web Access
Project: src\src\roslyn\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.Composition;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.Completion;
using Microsoft.CodeAnalysis.CSharp.Completion.Providers;
using Microsoft.CodeAnalysis.Host;
using Microsoft.CodeAnalysis.Host.Mef;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Microsoft.CodeAnalysis.Shared.TestHooks;
using Microsoft.CodeAnalysis.Text;

namespace Microsoft.CodeAnalysis.CSharp.Completion;

internal sealed class CSharpCompletionService : CommonCompletionService
{
    [ExportLanguageServiceFactory(typeof(CompletionService), LanguageNames.CSharp), Shared]
    [method: ImportingConstructor]
    [method: Obsolete(MefConstruction.ImportingConstructorMessage, error: true)]
    internal sealed class Factory(IAsynchronousOperationListenerProvider listenerProvider) : ILanguageServiceFactory
    {
        private readonly IAsynchronousOperationListenerProvider _listenerProvider = listenerProvider;

        [Obsolete(MefConstruction.FactoryMethodMessage, error: true)]
        public ILanguageService CreateLanguageService(HostLanguageServices languageServices)
            => new CSharpCompletionService(languageServices.LanguageServices.SolutionServices, _listenerProvider);
    }

    private CompletionRules _latestRules = CompletionRules.Default;

    private CSharpCompletionService(SolutionServices services, IAsynchronousOperationListenerProvider listenerProvider)
        : base(services, listenerProvider)
    {
    }

    public override string Language => LanguageNames.CSharp;

    public override TextSpan GetDefaultCompletionListSpan(SourceText text, int caretPosition)
        => CompletionUtilities.GetCompletionItemSpan(text, caretPosition);

    internal override CompletionRules GetRules(CompletionOptions options)
    {
        var enterRule = options.EnterKeyBehavior;
        var snippetRule = options.SnippetsBehavior;

        // Although EnterKeyBehavior is a per-language setting, the meaning of an unset setting (Default) differs between C# and VB
        // In C# the default means Never to maintain previous behavior
        if (enterRule == EnterKeyRule.Default)
        {
            enterRule = EnterKeyRule.Never;
        }

        if (snippetRule == SnippetsRule.Default)
        {
            snippetRule = SnippetsRule.AlwaysInclude;
        }

        // use interlocked + stored rules to reduce # of times this gets created when option is different than default
        var newRules = _latestRules.WithDefaultEnterKeyRule(enterRule)
                                   .WithSnippetsRule(snippetRule);

        Interlocked.Exchange(ref _latestRules, newRules);

        return newRules;
    }

    internal override async Task<bool> IsSpeculativeTypeParameterContextAsync(Document document, int position, CancellationToken cancellationToken)
    {
        var syntaxTree = await document.GetRequiredSyntaxTreeAsync(cancellationToken).ConfigureAwait(false);

        // Because it's less likely the user wants to type a (undeclared) type parameter when they are inside a method body, treating them so
        // might intefere with user intention. For example, while it's fine to provide a speculative `T` item in a statement context,
        // since typing 2 characters would filter it out, but for selection, we don't want to soft-select item `TypeBuilder`after `TB`
        // is typed in the example below (as if user want to add `TBuilder` to method declaration later):
        //
        // class C
        // {
        //     void M()
        //     {
        //         TB$$
        //     }
        return CompletionUtilities.IsSpeculativeTypeParameterContext(syntaxTree, position, semanticModel: null, includeStatementContexts: false, cancellationToken);
    }
}