|  | 
// 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);
    }
}
 |