File: InlineDiagnostics\InlineDiagnosticsTaggerProvider.cs
Web Access
Project: src\src\EditorFeatures\Core.Wpf\Microsoft.CodeAnalysis.EditorFeatures.Wpf_mf1xp1fn_wpftmp.csproj (Microsoft.CodeAnalysis.EditorFeatures.Wpf)
// 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.Immutable;
using System.ComponentModel.Composition;
using System.Diagnostics;
using Microsoft.CodeAnalysis.Diagnostics;
using Microsoft.CodeAnalysis.EditAndContinue;
using Microsoft.CodeAnalysis.Editor.Tagging;
using Microsoft.CodeAnalysis.Host.Mef;
using Microsoft.CodeAnalysis.Options;
using Microsoft.CodeAnalysis.Shared.TestHooks;
using Microsoft.VisualStudio.Text.Adornments;
using Microsoft.VisualStudio.Text.Classification;
using Microsoft.VisualStudio.Text.Tagging;
using Microsoft.VisualStudio.Utilities;
 
namespace Microsoft.CodeAnalysis.Editor.InlineDiagnostics;
 
[Export(typeof(ITaggerProvider))]
[ContentType(ContentTypeNames.RoslynContentType)]
[TagType(typeof(InlineDiagnosticsTag))]
[method: ImportingConstructor]
[method: Obsolete(MefConstruction.ImportingConstructorMessage, error: true)]
internal sealed class InlineDiagnosticsTaggerProvider(
    IDiagnosticAnalyzerService analyzerService,
    TaggerHost taggerHost,
    IEditorFormatMapService editorFormatMapService,
    IClassificationFormatMapService classificationFormatMapService,
    IClassificationTypeRegistryService classificationTypeRegistryService)
    : AbstractDiagnosticsTaggerProvider<InlineDiagnosticsTag>(
        analyzerService,
        taggerHost,
        FeatureAttribute.ErrorSquiggles)
{
    private readonly IEditorFormatMap _editorFormatMap = editorFormatMapService.GetEditorFormatMap("text");
    private readonly IClassificationFormatMapService _classificationFormatMapService = classificationFormatMapService;
    private readonly IClassificationTypeRegistryService _classificationTypeRegistryService = classificationTypeRegistryService;
 
    protected sealed override ImmutableArray<IOption2> Options { get; } = [InlineDiagnosticsOptionsStorage.EnableInlineDiagnostics];
    protected sealed override ImmutableArray<IOption2> FeatureOptions { get; } = [InlineDiagnosticsOptionsStorage.Location];
 
    protected sealed override bool IncludeDiagnostic(DiagnosticData diagnostic)
    {
        return
            diagnostic.Severity is DiagnosticSeverity.Warning or DiagnosticSeverity.Error &&
            !string.IsNullOrWhiteSpace(diagnostic.Message) &&
            !diagnostic.IsSuppressed;
    }
 
    protected override InlineDiagnosticsTag? CreateTag(Workspace workspace, DiagnosticData diagnostic)
    {
        Debug.Assert(!string.IsNullOrWhiteSpace(diagnostic.Message));
        var errorType = GetErrorTypeFromDiagnostic(diagnostic);
        if (errorType is null)
        {
            return null;
        }
 
        if (diagnostic.DocumentId is null)
        {
            return null;
        }
 
        var project = workspace.CurrentSolution.GetProject(diagnostic.DocumentId.ProjectId);
        if (project is null)
        {
            return null;
        }
 
        var locationOption = GlobalOptions.GetOption(InlineDiagnosticsOptionsStorage.Location, project.Language);
        var navigateService = workspace.Services.GetRequiredService<INavigateToLinkService>();
        return new InlineDiagnosticsTag(errorType, diagnostic, _editorFormatMap, _classificationFormatMapService,
            _classificationTypeRegistryService, locationOption, navigateService);
    }
 
    private static string? GetErrorTypeFromDiagnostic(DiagnosticData diagnostic)
    {
        if (diagnostic.Severity == DiagnosticSeverity.Error)
        {
            return diagnostic.CustomTags.Contains(WellKnownDiagnosticTags.EditAndContinue)
                ? EditAndContinueErrorTypeDefinition.Name
                : PredefinedErrorTypeNames.SyntaxError;
        }
        else if (diagnostic.Severity == DiagnosticSeverity.Warning)
        {
            return PredefinedErrorTypeNames.Warning;
        }
        else
        {
            return null;
        }
    }
 
    /// <summary>
    /// TODO: is there anything we can do better here? Inline diagnostic tags are not really data, but more UI
    /// elements with specific controls, positions and events attached to them.  There doesn't seem to be a safe way
    /// to reuse any of these currently.  Ideally we could do something similar to inline-hints where there's a data
    /// tagger portion (which is async and has clean equality semantics), and then the UI portion which just
    /// translates those data-tags to the UI tags.
    /// <para>
    /// Doing direct equality means we'll always end up regenerating all tags.  But hopefully there won't be that
    /// many in a document to matter.
    /// </para>
    /// </summary>
    protected sealed override bool TagEquals(InlineDiagnosticsTag tag1, InlineDiagnosticsTag tag2)
        => tag1 == tag2;
}