|
// 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.ComponentModel.Composition;
using System.Diagnostics;
using Microsoft.CodeAnalysis.EditAndContinue;
using Microsoft.CodeAnalysis.Editor;
using Microsoft.CodeAnalysis.Editor.Shared.Options;
using Microsoft.CodeAnalysis.Editor.Shared.Utilities;
using Microsoft.CodeAnalysis.Host.Mef;
using Microsoft.CodeAnalysis.Options;
using Microsoft.CodeAnalysis.Shared.TestHooks;
using Microsoft.CodeAnalysis.Workspaces;
using Microsoft.VisualStudio.Text.Adornments;
using Microsoft.VisualStudio.Text.Tagging;
using Microsoft.VisualStudio.Utilities;
using Roslyn.Utilities;
namespace Microsoft.CodeAnalysis.Diagnostics
{
[Export(typeof(ITaggerProvider))]
[ContentType(ContentTypeNames.RoslynContentType)]
[ContentType(ContentTypeNames.XamlContentType)]
[TagType(typeof(IErrorTag))]
internal sealed partial class DiagnosticsSquiggleTaggerProvider : AbstractDiagnosticsAdornmentTaggerProvider<IErrorTag>
{
protected override ImmutableArray<IOption2> Options { get; } = ImmutableArray.Create<IOption2>(DiagnosticsOptionsStorage.Squiggles);
[ImportingConstructor]
[Obsolete(MefConstruction.ImportingConstructorMessage, error: true)]
public DiagnosticsSquiggleTaggerProvider(
IThreadingContext threadingContext,
IDiagnosticService diagnosticService,
IDiagnosticAnalyzerService analyzerService,
IGlobalOptionService globalOptions,
[Import(AllowDefault = true)] ITextBufferVisibilityTracker? visibilityTracker,
IAsynchronousOperationListenerProvider listenerProvider)
: base(threadingContext, diagnosticService, analyzerService, globalOptions, visibilityTracker, listenerProvider)
{
}
protected sealed override bool SupportsDiagnosticMode(DiagnosticMode mode)
{
// We only support solution crawler push diagnostics. When lsp pull diagnostics are on, squiggles
// are handled by the lsp client.
return mode == DiagnosticMode.SolutionCrawlerPush;
}
protected sealed override bool IncludeDiagnostic(DiagnosticData diagnostic)
{
var isUnnecessary = diagnostic.Severity == DiagnosticSeverity.Hidden && diagnostic.CustomTags.Contains(WellKnownDiagnosticTags.Unnecessary);
return
(diagnostic.Severity == DiagnosticSeverity.Warning || diagnostic.Severity == DiagnosticSeverity.Error || isUnnecessary) &&
!string.IsNullOrWhiteSpace(diagnostic.Message);
}
protected sealed override IErrorTag? CreateTag(Workspace workspace, DiagnosticData diagnostic)
{
Debug.Assert(!string.IsNullOrWhiteSpace(diagnostic.Message));
var errorType = GetErrorTypeFromDiagnostic(diagnostic);
if (errorType == null)
{
// unknown diagnostic kind.
// we don't provide tagging for unknown diagnostic kind.
//
// it should be provided by the one who introduced the new diagnostic kind.
return null;
}
return new RoslynErrorTag(errorType, workspace, diagnostic);
}
private static string? GetErrorTypeFromDiagnostic(DiagnosticData diagnostic)
{
if (diagnostic.IsSuppressed)
{
// Don't squiggle suppressed diagnostics.
return null;
}
return GetErrorTypeFromDiagnosticTags(diagnostic) ??
GetErrorTypeFromDiagnosticSeverity(diagnostic);
}
private static string? GetErrorTypeFromDiagnosticTags(DiagnosticData diagnostic)
{
if (diagnostic.Severity == DiagnosticSeverity.Error &&
diagnostic.CustomTags.Contains(WellKnownDiagnosticTags.EditAndContinue))
{
return EditAndContinueErrorTypeDefinition.Name;
}
return null;
}
private static string? GetErrorTypeFromDiagnosticSeverity(DiagnosticData diagnostic)
{
switch (diagnostic.Severity)
{
case DiagnosticSeverity.Error:
return PredefinedErrorTypeNames.SyntaxError;
case DiagnosticSeverity.Warning:
return PredefinedErrorTypeNames.Warning;
case DiagnosticSeverity.Info:
return null;
case DiagnosticSeverity.Hidden:
if (diagnostic.CustomTags.Contains(WellKnownDiagnosticTags.Unnecessary))
{
// This ensures that we have an 'invisible' squiggle (which will in turn
// display Quick Info on mouse hover) for the hidden diagnostics that we
// report for 'Remove Unnecessary Usings' and 'Simplify Type Name'. The
// presence of Quick Info pane for such squiggles allows platform
// to display Light Bulb for the corresponding fixes (per their current
// design platform can only display light bulb if Quick Info pane is present).
return PredefinedErrorTypeNames.Suggestion;
}
return null;
default:
return PredefinedErrorTypeNames.OtherError;
}
}
protected sealed override bool TagEquals(IErrorTag tag1, IErrorTag tag2)
{
Contract.ThrowIfFalse(tag1 is RoslynErrorTag);
Contract.ThrowIfFalse(tag2 is RoslynErrorTag);
return tag1.Equals(tag2);
}
}
}
|