|
// 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 Microsoft.CodeAnalysis.CSharp.Extensions;
using Microsoft.CodeAnalysis.CSharp.Extensions.ContextQuery;
using Microsoft.CodeAnalysis.CSharp.Utilities;
namespace Microsoft.CodeAnalysis.CSharp.Completion.KeywordRecommenders
{
internal class DelegateKeywordRecommender : AbstractSyntacticSingleKeywordRecommender
{
private static readonly ISet<SyntaxKind> s_validModifiers = new HashSet<SyntaxKind>(SyntaxFacts.EqualityComparer)
{
SyntaxKind.InternalKeyword,
SyntaxKind.PublicKeyword,
SyntaxKind.PrivateKeyword,
SyntaxKind.ProtectedKeyword,
SyntaxKind.UnsafeKeyword
};
public DelegateKeywordRecommender()
: base(SyntaxKind.DelegateKeyword)
{
}
protected override bool IsValidContext(int position, CSharpSyntaxContext context, CancellationToken cancellationToken)
{
return
context.IsGlobalStatementContext ||
context.IsUsingAliasTypeContext ||
ValidTypeContext(context) ||
IsAfterAsyncKeywordInExpressionContext(context, cancellationToken) ||
context.IsTypeDeclarationContext(
validModifiers: s_validModifiers,
validTypeDeclarations: SyntaxKindSet.ClassInterfaceStructRecordTypeDeclarations,
canBePartial: false,
cancellationToken: cancellationToken);
static bool ValidTypeContext(CSharpSyntaxContext context)
=> (context.IsNonAttributeExpressionContext || context.IsTypeContext)
&& !context.IsConstantExpressionContext
&& !context.LeftToken.IsTopLevelOfUsingAliasDirective();
}
private static bool IsAfterAsyncKeywordInExpressionContext(CSharpSyntaxContext context, CancellationToken cancellationToken)
{
return
context.TargetToken.IsKindOrHasMatchingText(SyntaxKind.AsyncKeyword) &&
context.SyntaxTree.IsExpressionContext(
context.TargetToken.SpanStart,
context.TargetToken,
attributes: false,
cancellationToken: cancellationToken,
semanticModel: context.SemanticModel);
}
}
}
|