From 61153ca46ba62e0062d7f37276ca7ea5627f7b00 Mon Sep 17 00:00:00 2001 From: Vasily Kirichenko Date: Fri, 16 Dec 2016 21:08:41 +0300 Subject: [PATCH] Implement DocumentHighlightsService (#1943) * implement DocumentHighlightsService * use the right ImportingConstructor attribute * tolerate being on the right of the identifier * add a DocumentHighlightsService test * fix erroneous qualified symbol ranges (wip) * fix `fixInvalidSymbolSpans` * fix after merge * update Microsoft.CodeAnalysis.xxx packages to 2.0.0-rc * remove double try to find symbol * fix after merge fix tryClassifyAtPosition --- CommonHelpers.fs | 64 ++++++---- .../DocumentHighlightsService.fs | 115 ++++++++++++++++++ FSharp.Editor.fsproj | 18 +-- Navigation/GoToDefinitionService.fs | 2 +- QuickInfo/QuickInfoProvider.fs | 5 +- 5 files changed, 167 insertions(+), 37 deletions(-) create mode 100644 DocumentHighlights/DocumentHighlightsService.fs diff --git a/CommonHelpers.fs b/CommonHelpers.fs index 32fcbf76dc1..65978303188 100644 --- a/CommonHelpers.fs +++ b/CommonHelpers.fs @@ -15,7 +15,12 @@ open Microsoft.VisualStudio.FSharp.LanguageService open Microsoft.FSharp.Compiler.SourceCodeServices open Microsoft.FSharp.Compiler.SourceCodeServices.ItemDescriptionIcons -module CommonHelpers = +[] +type internal SymbolSearchKind = + | IncludeRightColumn + | DoesNotIncludeRightColumn + +module internal CommonHelpers = type private SourceLineData(lineStart: int, lexStateAtStartOfLine: FSharpTokenizerLexState, lexStateAtEndOfLine: FSharpTokenizerLexState, hashCode: int, classifiedSpans: IReadOnlyList) = member val LineStart = lineStart member val LexStateAtStartOfLine = lexStateAtStartOfLine @@ -96,7 +101,8 @@ module CommonHelpers = SourceLineData(textLine.Start, lexState, previousLexState.Value, lineContents.GetHashCode(), classifiedSpans) - let getColorizationData(documentKey: DocumentId, sourceText: SourceText, textSpan: TextSpan, fileName: string option, defines: string list, cancellationToken: CancellationToken) : List = + let getColorizationData(documentKey: DocumentId, sourceText: SourceText, textSpan: TextSpan, fileName: string option, defines: string list, + cancellationToken: CancellationToken) : List = try let sourceTokenizer = FSharpSourceTokenizer(defines, fileName) let lines = sourceText.Lines @@ -159,36 +165,40 @@ module CommonHelpers = Assert.Exception(ex) List() - let tryClassifyAtPosition (documentKey, sourceText: SourceText, filePath, defines, position: int, cancellationToken) = + let tryClassifyAtPosition (documentKey, sourceText: SourceText, filePath, defines, position: int, symbolSearchKind: SymbolSearchKind, cancellationToken) = let textLine = sourceText.Lines.GetLineFromPosition(position) let textLinePos = sourceText.Lines.GetLinePosition(position) let textLineColumn = textLinePos.Character + let spans = getColorizationData(documentKey, sourceText, textLine.Span, Some filePath, defines, cancellationToken) - let classifiedSpanOption = - getColorizationData(documentKey, sourceText, textLine.Span, Some(filePath), defines, cancellationToken) - |> Seq.tryFind(fun classifiedSpan -> classifiedSpan.TextSpan.Contains(position)) - - match classifiedSpanOption with - | Some(classifiedSpan) -> - match classifiedSpan.ClassificationType with - | ClassificationTypeNames.ClassName - | ClassificationTypeNames.DelegateName - | ClassificationTypeNames.EnumName - | ClassificationTypeNames.InterfaceName - | ClassificationTypeNames.ModuleName - | ClassificationTypeNames.StructName - | ClassificationTypeNames.TypeParameterName - | ClassificationTypeNames.Identifier -> - match QuickParse.GetCompleteIdentifierIsland true (textLine.ToString()) textLineColumn with - | Some (islandIdentifier, islandColumn, isQuoted) -> - let qualifiers = if isQuoted then [islandIdentifier] else islandIdentifier.Split '.' |> Array.toList - Some (islandColumn, qualifiers, classifiedSpan.TextSpan) - | None -> None - | ClassificationTypeNames.Operator -> - let islandColumn = sourceText.Lines.GetLinePositionSpan(classifiedSpan.TextSpan).End.Character - Some (islandColumn, [""], classifiedSpan.TextSpan) + let attempt (position: int) = + let classifiedSpanOption = spans |> Seq.tryFind (fun classifiedSpan -> classifiedSpan.TextSpan.Contains position) + + match classifiedSpanOption with + | Some(classifiedSpan) -> + match classifiedSpan.ClassificationType with + | ClassificationTypeNames.ClassName + | ClassificationTypeNames.DelegateName + | ClassificationTypeNames.EnumName + | ClassificationTypeNames.InterfaceName + | ClassificationTypeNames.ModuleName + | ClassificationTypeNames.StructName + | ClassificationTypeNames.TypeParameterName + | ClassificationTypeNames.Identifier -> + match QuickParse.GetCompleteIdentifierIsland true (textLine.ToString()) textLineColumn with + | Some (islandIdentifier, islandColumn, isQuoted) -> + let qualifiers = if isQuoted then [islandIdentifier] else islandIdentifier.Split '.' |> Array.toList + Some (islandColumn, qualifiers, classifiedSpan.TextSpan) + | None -> None + | ClassificationTypeNames.Operator -> + let islandColumn = sourceText.Lines.GetLinePositionSpan(classifiedSpan.TextSpan).End.Character + Some (islandColumn, [""], classifiedSpan.TextSpan) + | _ -> None | _ -> None - | _ -> None + + match attempt position, symbolSearchKind with + | None, SymbolSearchKind.IncludeRightColumn -> attempt (position - 1) + | x, _ -> x /// Fix invalid span if it appears to have redundant suffix and prefix. let fixupSpan (sourceText: SourceText, span: TextSpan) : TextSpan = diff --git a/DocumentHighlights/DocumentHighlightsService.fs b/DocumentHighlights/DocumentHighlightsService.fs new file mode 100644 index 00000000000..2424a126745 --- /dev/null +++ b/DocumentHighlights/DocumentHighlightsService.fs @@ -0,0 +1,115 @@ +// Copyright (c) Microsoft Corporation. All Rights Reserved. Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. + +namespace Microsoft.VisualStudio.FSharp.Editor + +open System +open System.Composition +open System.Collections.Concurrent +open System.Collections.Generic +open System.Collections.Immutable +open System.Threading +open System.Threading.Tasks +open System.Linq +open System.Runtime.CompilerServices + +open Microsoft.CodeAnalysis +open Microsoft.CodeAnalysis.Completion +open Microsoft.CodeAnalysis.Classification +open Microsoft.CodeAnalysis.Editor +open Microsoft.CodeAnalysis.Editor.Implementation.ReferenceHighlighting +open Microsoft.CodeAnalysis.Host.Mef +open Microsoft.CodeAnalysis.Options +open Microsoft.CodeAnalysis.Text + +open Microsoft.VisualStudio.FSharp.LanguageService +open Microsoft.VisualStudio.Text +open Microsoft.VisualStudio.Text.Classification +open Microsoft.VisualStudio.Text.Tagging + +open Microsoft.FSharp.Compiler +open Microsoft.FSharp.Compiler.Parser +open Microsoft.FSharp.Compiler.Range +open Microsoft.FSharp.Compiler.SourceCodeServices +open System.Windows.Documents + +type internal FSharpHighlightSpan = + { IsDefinition: bool + TextSpan: TextSpan } + override this.ToString() = sprintf "%+A" this + +[] +[, FSharpCommonConstants.FSharpLanguageName)>] +type internal FSharpDocumentHighlightsService [] (checkerProvider: FSharpCheckerProvider, projectInfoManager: ProjectInfoManager) = + + /// Fix invalid spans if they appear to have redundant suffix and prefix. + static let fixInvalidSymbolSpans (sourceText: SourceText) (lastIdent: string) (spans: FSharpHighlightSpan []) = + spans + |> Seq.choose (fun (span: FSharpHighlightSpan) -> + let newLastIdent = sourceText.GetSubText(span.TextSpan).ToString() + let index = newLastIdent.LastIndexOf(lastIdent, StringComparison.Ordinal) + if index > 0 then + // Sometimes FCS returns a composite identifier for a short symbol, so we truncate the prefix + // Example: newLastIdent --> "x.Length", lastIdent --> "Length" + Some { span with TextSpan = TextSpan(span.TextSpan.Start + index, span.TextSpan.Length - index) } + elif index = 0 && newLastIdent.Length > lastIdent.Length then + // The returned symbol use is too long; we truncate its redundant suffix + // Example: newLastIdent --> "Length<'T>", lastIdent --> "Length" + Some { span with TextSpan = TextSpan(span.TextSpan.Start, lastIdent.Length) } + elif index = 0 then + Some span + else + // In the case of attributes, a returned symbol use may be a part of original text + // Example: newLastIdent --> "Sample", lastIdent --> "SampleAttribute" + let index = lastIdent.LastIndexOf(newLastIdent, StringComparison.Ordinal) + if index >= 0 then + Some span + else None) + |> Seq.distinctBy (fun span -> span.TextSpan.Start) + |> Seq.toArray + + static member GetDocumentHighlights(checker: FSharpChecker, documentKey: DocumentId, sourceText: SourceText, filePath: string, position: int, + defines: string list, options: FSharpProjectOptions, textVersionHash: int, cancellationToken: CancellationToken) : Async = + async { + let textLine = sourceText.Lines.GetLineFromPosition(position) + let textLinePos = sourceText.Lines.GetLinePosition(position) + let fcsTextLineNumber = textLinePos.Line + 1 + + match CommonHelpers.tryClassifyAtPosition(documentKey, sourceText, filePath, defines, position, SymbolSearchKind.IncludeRightColumn, cancellationToken) with + | Some (_, [], _) -> return [||] + | Some (islandEndColumn, qualifiers, _span) -> + let! _parseResults, checkFileAnswer = checker.ParseAndCheckFileInProject(filePath, textVersionHash, sourceText.ToString(), options) + match checkFileAnswer with + | FSharpCheckFileAnswer.Aborted -> return [||] + | FSharpCheckFileAnswer.Succeeded(checkFileResults) -> + let! symbolUse = checkFileResults.GetSymbolUseAtLocation(fcsTextLineNumber, islandEndColumn, textLine.ToString(), qualifiers) + match symbolUse with + | Some symbolUse -> + let! symbolUses = checkFileResults.GetUsesOfSymbolInFile(symbolUse.Symbol) + let lastIdent = List.last qualifiers + return + [| for symbolUse in symbolUses do + yield { IsDefinition = symbolUse.IsFromDefinition + TextSpan = CommonRoslynHelpers.FSharpRangeToTextSpan(sourceText, symbolUse.RangeAlternate) } |] + |> fixInvalidSymbolSpans sourceText lastIdent + | None -> return [||] + | None -> return [||] + } + + interface IDocumentHighlightsService with + member __.GetDocumentHighlightsAsync(document, position, _documentsToSearch, cancellationToken) : Task> = + async { + match projectInfoManager.TryGetOptionsForEditingDocumentOrProject(document) with + | Some options -> + let! sourceText = document.GetTextAsync(cancellationToken) |> Async.AwaitTask + let! textVersion = document.GetTextVersionAsync(cancellationToken) |> Async.AwaitTask + let defines = CompilerEnvironment.GetCompilationDefinesForEditing(document.Name, options.OtherOptions |> Seq.toList) + let! spans = FSharpDocumentHighlightsService.GetDocumentHighlights(checkerProvider.Checker, document.Id, sourceText, document.FilePath, position, defines, options, textVersion.GetHashCode(), cancellationToken) + + let highlightSpans = spans |> Array.map (fun span -> + let kind = if span.IsDefinition then HighlightSpanKind.Definition else HighlightSpanKind.Reference + HighlightSpan(span.TextSpan, kind)) + + return [| DocumentHighlights(document, highlightSpans.ToImmutableArray()) |].ToImmutableArray() + | None -> return ImmutableArray() + } + |> CommonRoslynHelpers.StartAsyncAsTask(cancellationToken) diff --git a/FSharp.Editor.fsproj b/FSharp.Editor.fsproj index a81f6a99af4..c7c58c6db64 100644 --- a/FSharp.Editor.fsproj +++ b/FSharp.Editor.fsproj @@ -34,13 +34,17 @@ - - - - - - - + + + + + + + + + + + diff --git a/Navigation/GoToDefinitionService.fs b/Navigation/GoToDefinitionService.fs index 94caa83cecd..64319f31a3a 100644 --- a/Navigation/GoToDefinitionService.fs +++ b/Navigation/GoToDefinitionService.fs @@ -53,7 +53,7 @@ type internal FSharpGoToDefinitionService let textLine = sourceText.Lines.GetLineFromPosition(position) let textLinePos = sourceText.Lines.GetLinePosition(position) let fcsTextLineNumber = textLinePos.Line + 1 // Roslyn line numbers are zero-based, FSharp.Compiler.Service line numbers are 1-based - match CommonHelpers.tryClassifyAtPosition(documentKey, sourceText, filePath, defines, position, cancellationToken) with + match CommonHelpers.tryClassifyAtPosition(documentKey, sourceText, filePath, defines, position, SymbolSearchKind.IncludeRightColumn, cancellationToken) with | Some (islandColumn, qualifiers, _) -> let! _parseResults, checkFileAnswer = checker.ParseAndCheckFileInProject(filePath, textVersionHash, sourceText.ToString(), options) match checkFileAnswer with diff --git a/QuickInfo/QuickInfoProvider.fs b/QuickInfo/QuickInfoProvider.fs index 467ade58cb2..5abd076bd06 100644 --- a/QuickInfo/QuickInfoProvider.fs +++ b/QuickInfo/QuickInfoProvider.fs @@ -85,7 +85,7 @@ type internal FSharpQuickInfoProvider //let qualifyingNames, partialName = QuickParse.GetPartialLongNameEx(textLine.ToString(), textLineColumn - 1) let defines = CompilerEnvironment.GetCompilationDefinesForEditing(filePath, options.OtherOptions |> Seq.toList) let tryClassifyAtPosition position = - CommonHelpers.tryClassifyAtPosition(documentId, sourceText, filePath, defines, position, cancellationToken) + CommonHelpers.tryClassifyAtPosition(documentId, sourceText, filePath, defines, position, SymbolSearchKind.DoesNotIncludeRightColumn, cancellationToken) let quickParseInfo = match tryClassifyAtPosition position with @@ -107,7 +107,8 @@ type internal FSharpQuickInfoProvider async { let! sourceText = document.GetTextAsync(cancellationToken) |> Async.AwaitTask let defines = projectInfoManager.GetCompilationDefinesForEditingDocument(document) - let classification = CommonHelpers.tryClassifyAtPosition(document.Id, sourceText, document.FilePath, defines, position, cancellationToken) + let classification = + CommonHelpers.tryClassifyAtPosition(document.Id, sourceText, document.FilePath, defines, position, SymbolSearchKind.DoesNotIncludeRightColumn, cancellationToken) match classification with | Some _ ->