File: DocumentHighlighting\CSharpDocumentHighlightsService.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.Linq;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.CSharp.EmbeddedLanguages.LanguageServices;
using Microsoft.CodeAnalysis.CSharp.LanguageService;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using Microsoft.CodeAnalysis.DocumentHighlighting;
using Microsoft.CodeAnalysis.EmbeddedLanguages;
using Microsoft.CodeAnalysis.Host.Mef;
using Microsoft.CodeAnalysis.PooledObjects;
using Microsoft.CodeAnalysis.Shared.Extensions;
 
namespace Microsoft.CodeAnalysis.CSharp.DocumentHighlighting;
 
[ExportLanguageService(typeof(IDocumentHighlightsService), LanguageNames.CSharp), Shared]
[method: ImportingConstructor]
[method: Obsolete(MefConstruction.ImportingConstructorMessage, error: true)]
internal class CSharpDocumentHighlightsService(
    [ImportMany] IEnumerable<Lazy<IEmbeddedLanguageDocumentHighlighter, EmbeddedLanguageMetadata>> services)
    : AbstractDocumentHighlightsService(
        LanguageNames.CSharp,
        CSharpEmbeddedLanguagesProvider.Info,
        CSharpSyntaxKinds.Instance,
        services)
{
    protected override async Task<ImmutableArray<Location>> GetAdditionalReferencesAsync(
        Document document, ISymbol symbol, CancellationToken cancellationToken)
    {
        // The FindRefs engine won't find references through 'var' for performance reasons.
        // Also, they are not needed for things like rename/sig change, and the normal find refs
        // feature.  However, we would like the results to be highlighted to get a good experience
        // while editing (especially since highlighting may have been invoked off of 'var' in
        // the first place).
        //
        // So we look for the references through 'var' directly in this file and add them to the
        // results found by the engine.
        using var _ = ArrayBuilder<Location>.GetInstance(out var results);
 
        if (symbol is INamedTypeSymbol && symbol.Name != "var")
        {
            var originalSymbol = symbol.OriginalDefinition;
            var root = await document.GetRequiredSyntaxRootAsync(cancellationToken).ConfigureAwait(false);
 
            var descendants = root.DescendantNodes();
            var semanticModel = (SemanticModel?)null;
 
            foreach (var type in descendants.OfType<IdentifierNameSyntax>())
            {
                cancellationToken.ThrowIfCancellationRequested();
 
                if (type.IsVar)
                {
                    // Document highlights are not impacted by nullable analysis.  Get a semantic model with nullability
                    // disabled to lower the amount of work we need to do here.
                    semanticModel ??= await document.GetRequiredNullableDisabledSemanticModelAsync(cancellationToken).ConfigureAwait(false);
 
                    var boundSymbol = semanticModel.GetSymbolInfo(type, cancellationToken).Symbol;
                    boundSymbol = boundSymbol?.OriginalDefinition;
 
                    if (originalSymbol.Equals(boundSymbol))
                        results.Add(type.GetLocation());
                }
            }
        }
 
        return results.ToImmutableAndClear();
    }
}