From 9ec554e4b3dc8d6f2506c3056d18b1b8c522f000 Mon Sep 17 00:00:00 2001 From: Eugene Auduchinok Date: Wed, 23 Mar 2022 18:47:17 +0300 Subject: [PATCH 1/3] Completion: fixes for records * Fix for fields in empty records * Fix for env items to be used as copyInfo --- src/fsharp/NameResolution.fs | 16 +++--- src/fsharp/NameResolution.fsi | 2 + src/fsharp/service/FSharpCheckerResults.fs | 41 ++++++++++++--- src/fsharp/service/ServiceParsedInputOps.fs | 25 +++++++-- src/fsharp/service/ServiceParsedInputOps.fsi | 3 +- tests/service/CompletionTests.fs | 55 +++++++++++++++++--- 6 files changed, 117 insertions(+), 25 deletions(-) diff --git a/src/fsharp/NameResolution.fs b/src/fsharp/NameResolution.fs index a77d3ba75d2..0cfbf0dbfbb 100644 --- a/src/fsharp/NameResolution.fs +++ b/src/fsharp/NameResolution.fs @@ -4473,6 +4473,14 @@ let rec ResolvePartialLongIdentInModuleOrNamespaceForRecordFields (ncenv: NameRe | _ -> [] ) +let getRecordFieldsInScope nenv = + nenv.eFieldLabels + |> Seq.collect (fun (KeyValue(_, v)) -> v) + |> Seq.map (fun fref -> + let typeInsts = fref.TyconRef.TyparsNoRange |> List.map mkTyparTy + Item.RecdField(RecdFieldInfo(typeInsts, fref))) + |> List.ofSeq + /// allowObsolete - specifies whether we should return obsolete types & modules /// as (no other obsolete items are returned) let rec ResolvePartialLongIdentToClassOrRecdFields (ncenv: NameResolver) (nenv: NameResolutionEnv) m ad plid (allowObsolete: bool) = @@ -4516,13 +4524,7 @@ and ResolvePartialLongIdentToClassOrRecdFieldsImpl (ncenv: NameResolver) (nenv: |> Seq.map (ItemOfTyconRef ncenv m) |> Seq.toList - let recdFields = - nenv.eFieldLabels - |> Seq.collect (fun (KeyValue(_, v)) -> v) - |> Seq.map (fun fref -> - let typeInsts = fref.TyconRef.TyparsNoRange |> List.map mkTyparTy - Item.RecdField(RecdFieldInfo(typeInsts, fref))) - |> List.ofSeq + let recdFields = getRecordFieldsInScope nenv mods @ recdTyCons @ recdFields diff --git a/src/fsharp/NameResolution.fsi b/src/fsharp/NameResolution.fsi index 0b96de1fac1..cd36ab6362a 100755 --- a/src/fsharp/NameResolution.fsi +++ b/src/fsharp/NameResolution.fsi @@ -557,6 +557,8 @@ val internal ResolveField : TcResultsSink -> NameResolver - /// Resolve a long identifier occurring in an expression position val internal ResolveExprLongIdent : TcResultsSink -> NameResolver -> range -> AccessorDomain -> NameResolutionEnv -> TypeNameResolutionInfo -> Ident list -> ResultOrException +val internal getRecordFieldsInScope: NameResolutionEnv -> Item list + /// Resolve a (possibly incomplete) long identifier to a loist of possible class or record fields val internal ResolvePartialLongIdentToClassOrRecdFields: NameResolver -> NameResolutionEnv -> range -> AccessorDomain -> string list -> bool -> Item list diff --git a/src/fsharp/service/FSharpCheckerResults.fs b/src/fsharp/service/FSharpCheckerResults.fs index d945fc786a6..9dd97642506 100644 --- a/src/fsharp/service/FSharpCheckerResults.fs +++ b/src/fsharp/service/FSharpCheckerResults.fs @@ -908,6 +908,21 @@ type internal TypeCheckInfo let toCompletionItems (items: ItemWithInst list, denv: DisplayEnv, m: range ) = items |> List.map DefaultCompletionItem, denv, m + /// Find record fields in the best naming environment. + let GetEnvironmentLookupResolutionsIncludingRecordFieldsAtPosition cursorPos envItems = + // An empty record expression may be completed into something like these: + // { XXX = ... } + // { xxx with XXX ... } + // Provide both expression items in scope and available record fields. + let (nenv, _), m = GetBestEnvForPos cursorPos + + let fieldItems = getRecordFieldsInScope nenv |> List.map ItemWithNoInst + let fieldCompletionItems, _, _ as fieldsResult = (fieldItems, nenv.DisplayEnv, m) |> toCompletionItems + + match envItems with + | Some(items, denv, m) -> Some(fieldCompletionItems @ items, denv, m) + | _ -> Some(fieldsResult) + /// Get the auto-complete items at a particular location. let GetDeclItemsForNamesAtPosition(parseResultsOpt: FSharpParseFileResults option, origLongIdentOpt: string list option, residueOpt:string option, lastDotPos: int option, line:int, lineStr:string, colAtEndOfNamesAndResidue, filterCtors, resolveOverloads, @@ -951,13 +966,25 @@ type internal TypeCheckInfo |> Option.map toCompletionItems // Completion at ' { XXX = ... } " - | Some(CompletionContext.RecordField(RecordContext.New(plid, _))) -> - // { x. } can be either record construction or computation expression. Try to get all visible record fields first - match GetClassOrRecordFieldsEnvironmentLookupResolutions(mkPos line loc, plid) |> toCompletionItems with - | [],_,_ -> - // no record fields found, return completion list as if we were outside any computation expression - GetDeclaredItems (parseResultsOpt, lineStr, origLongIdentOpt, colAtEndOfNamesAndResidue, residueOpt, lastDotPos, line, loc, filterCtors,resolveOverloads, false, fun() -> []) - | result -> Some(result) + | Some(CompletionContext.RecordField(RecordContext.New((plid, _), isFirstField))) -> + if isFirstField then + let cursorPos = mkPos line loc + let envItems = GetDeclaredItems (parseResultsOpt, lineStr, origLongIdentOpt, colAtEndOfNamesAndResidue, residueOpt, lastDotPos, line, loc, filterCtors,resolveOverloads, false, fun () -> []) + GetEnvironmentLookupResolutionsIncludingRecordFieldsAtPosition cursorPos envItems + else + // { x. } can be either record construction or computation expression. Try to get all visible record fields first + match GetClassOrRecordFieldsEnvironmentLookupResolutions(mkPos line loc, plid) |> toCompletionItems with + | [],_,_ -> + // no record fields found, return completion list as if we were outside any computation expression + GetDeclaredItems (parseResultsOpt, lineStr, origLongIdentOpt, colAtEndOfNamesAndResidue, residueOpt, lastDotPos, line, loc, filterCtors,resolveOverloads, false, fun() -> []) + | result -> Some(result) + + // Completion at '{ ... }' + | Some(CompletionContext.RecordField RecordContext.Empty) -> + let cursorPos = mkPos line loc + let envItems = GetDeclaredItems (parseResultsOpt, lineStr, origLongIdentOpt, colAtEndOfNamesAndResidue, residueOpt, lastDotPos, line, loc, filterCtors,resolveOverloads, false, fun () -> []) + GetEnvironmentLookupResolutionsIncludingRecordFieldsAtPosition cursorPos envItems + // Completion at ' { XXX = ... with ... } " | Some(CompletionContext.RecordField(RecordContext.CopyOnUpdate(r, (plid, _)))) -> diff --git a/src/fsharp/service/ServiceParsedInputOps.fs b/src/fsharp/service/ServiceParsedInputOps.fs index bf027df2038..29f2b1fcf02 100644 --- a/src/fsharp/service/ServiceParsedInputOps.fs +++ b/src/fsharp/service/ServiceParsedInputOps.fs @@ -43,7 +43,8 @@ type InheritanceContext = type RecordContext = | CopyOnUpdate of range: range * path: CompletionPath | Constructor of typeName: string - | New of path: CompletionPath + | Empty + | New of path: CompletionPath * isFirstField: bool | Declaration of isInIdentifier: bool [] @@ -956,7 +957,10 @@ module ParsedInput = Some (CompletionContext.ParameterList args) | _ -> defaultTraverse expr - + + | SynExpr.Record(None, None, [], _) -> + Some(CompletionContext.RecordField RecordContext.Empty) + | _ -> defaultTraverse expr member _.VisitRecordField(path, copyOpt, field) = @@ -965,7 +969,22 @@ module ParsedInput = match path with | SyntaxNode.SynExpr _ :: SyntaxNode.SynBinding _ :: SyntaxNode.SynMemberDefn _ :: SyntaxNode.SynTypeDefn(SynTypeDefn(typeInfo=SynComponentInfo(longId=[id]))) :: _ -> RecordContext.Constructor(id.idText) - | _ -> RecordContext.New completionPath + + | SyntaxNode.SynExpr(SynExpr.Record(None, _, fields, _)) :: _ -> + let isFirstField = + match field, fields with + | Some contextLid, SynExprRecordField(fieldName = lid, _) :: _ -> contextLid.Range = lid.Range + | _ -> false + + RecordContext.New(completionPath, isFirstField) + + // Unfinished `{ xxx }` expression considered a record field by the tree walker. + | SyntaxNode.SynExpr(SynExpr.ComputationExpr _) :: _ -> + RecordContext.New(completionPath, true) + + | _ -> + RecordContext.New(completionPath, false) + match field with | Some field -> match parseLid field with diff --git a/src/fsharp/service/ServiceParsedInputOps.fsi b/src/fsharp/service/ServiceParsedInputOps.fsi index 2180b700416..f037f8c0645 100644 --- a/src/fsharp/service/ServiceParsedInputOps.fsi +++ b/src/fsharp/service/ServiceParsedInputOps.fsi @@ -18,7 +18,8 @@ type public InheritanceContext = type public RecordContext = | CopyOnUpdate of range: range * path: CompletionPath | Constructor of typeName: string - | New of path: CompletionPath + | Empty + | New of path: CompletionPath * isFirstField: bool | Declaration of isInIdentifier: bool [] diff --git a/tests/service/CompletionTests.fs b/tests/service/CompletionTests.fs index 432633dc16e..4409f73737b 100644 --- a/tests/service/CompletionTests.fs +++ b/tests/service/CompletionTests.fs @@ -1,18 +1,59 @@ module FSharp.Compiler.Service.Tests.CompletionTests open FSharp.Compiler.EditorServices -open FsUnit open NUnit.Framework +let getCompletionInfo lineText (line, column) source = + let parseResults, checkResults = getParseAndCheckResults source + let plid = QuickParse.GetPartialLongNameEx(lineText, column) + checkResults.GetDeclarationListInfo(Some parseResults, line, lineText, plid) + +let getCompletionItemNames (completionInfo: DeclarationListInfo) = + completionInfo.Items |> Array.map (fun item -> item.Name) + +let assertHasItemWithNames names (completionInfo: DeclarationListInfo) = + let itemNames = getCompletionItemNames completionInfo |> set + + for name in names do + Assert.That(Set.contains name itemNames, name) + + [] let ``Expr - record - field 01 - anon module`` () = - let parseResults, checkResults = getParseAndCheckResults """ -type Record = { Field: int} + let info = getCompletionInfo "{ Fi }" (4, 3) """ +type Record = { Field: int } { Fi } """ - let lineText = "{ Fi }" - let plid = QuickParse.GetPartialLongNameEx(lineText, 3) - let info = checkResults.GetDeclarationListInfo(Some parseResults, 4, lineText, plid) + assertHasItemWithNames ["Field"] info - info.Items |> Array.exists (fun item -> item.Name = "Field") |> shouldEqual true +[] +let ``Expr - record - field 02 - anon module`` () = + let info = getCompletionInfo "{ Fi }" (6, 3) """ +type Record = { Field: int } + +let record = { Field = 1 } + +{ Fi } +""" + assertHasItemWithNames ["Field"] info + +[] +let ``Expr - record - empty 01`` () = + let info = getCompletionInfo "{ }" (4, 2) """ +type Record = { Field: int } + +{ } +""" + assertHasItemWithNames ["Field"] info + +[] +let ``Expr - record - empty 02`` () = + let info = getCompletionInfo "{ }" (6, 2) """ +type Record = { Field: int } + +let record = { Field = 1 } + +{ } +""" + assertHasItemWithNames ["Field"; "record"] info From d55158935d6d0e49fdbdf72bd7eea0b9ca2cd105 Mon Sep 17 00:00:00 2001 From: Eugene Auduchinok Date: Wed, 23 Mar 2022 23:10:12 +0300 Subject: [PATCH 2/3] Update surface area --- ...harp.CompilerService.SurfaceArea.netstandard.expected | 9 ++++++++- .../Tests.LanguageService.Completion.fs | 6 +++--- 2 files changed, 11 insertions(+), 4 deletions(-) diff --git a/tests/FSharp.Compiler.Service.Tests/FSharp.CompilerService.SurfaceArea.netstandard.expected b/tests/FSharp.Compiler.Service.Tests/FSharp.CompilerService.SurfaceArea.netstandard.expected index 5e980c2c2ec..2d142e167a0 100644 --- a/tests/FSharp.Compiler.Service.Tests/FSharp.CompilerService.SurfaceArea.netstandard.expected +++ b/tests/FSharp.Compiler.Service.Tests/FSharp.CompilerService.SurfaceArea.netstandard.expected @@ -3501,11 +3501,14 @@ FSharp.Compiler.EditorServices.RecordContext+CopyOnUpdate: System.Tuple`2[Micros FSharp.Compiler.EditorServices.RecordContext+CopyOnUpdate: System.Tuple`2[Microsoft.FSharp.Collections.FSharpList`1[System.String],Microsoft.FSharp.Core.FSharpOption`1[System.String]] path FSharp.Compiler.EditorServices.RecordContext+Declaration: Boolean get_isInIdentifier() FSharp.Compiler.EditorServices.RecordContext+Declaration: Boolean isInIdentifier +FSharp.Compiler.EditorServices.RecordContext+New: Boolean get_isFirstField() +FSharp.Compiler.EditorServices.RecordContext+New: Boolean isFirstField FSharp.Compiler.EditorServices.RecordContext+New: System.Tuple`2[Microsoft.FSharp.Collections.FSharpList`1[System.String],Microsoft.FSharp.Core.FSharpOption`1[System.String]] get_path() FSharp.Compiler.EditorServices.RecordContext+New: System.Tuple`2[Microsoft.FSharp.Collections.FSharpList`1[System.String],Microsoft.FSharp.Core.FSharpOption`1[System.String]] path FSharp.Compiler.EditorServices.RecordContext+Tags: Int32 Constructor FSharp.Compiler.EditorServices.RecordContext+Tags: Int32 CopyOnUpdate FSharp.Compiler.EditorServices.RecordContext+Tags: Int32 Declaration +FSharp.Compiler.EditorServices.RecordContext+Tags: Int32 Empty FSharp.Compiler.EditorServices.RecordContext+Tags: Int32 New FSharp.Compiler.EditorServices.RecordContext: Boolean Equals(FSharp.Compiler.EditorServices.RecordContext) FSharp.Compiler.EditorServices.RecordContext: Boolean Equals(System.Object) @@ -3513,15 +3516,19 @@ FSharp.Compiler.EditorServices.RecordContext: Boolean Equals(System.Object, Syst FSharp.Compiler.EditorServices.RecordContext: Boolean IsConstructor FSharp.Compiler.EditorServices.RecordContext: Boolean IsCopyOnUpdate FSharp.Compiler.EditorServices.RecordContext: Boolean IsDeclaration +FSharp.Compiler.EditorServices.RecordContext: Boolean IsEmpty FSharp.Compiler.EditorServices.RecordContext: Boolean IsNew FSharp.Compiler.EditorServices.RecordContext: Boolean get_IsConstructor() FSharp.Compiler.EditorServices.RecordContext: Boolean get_IsCopyOnUpdate() FSharp.Compiler.EditorServices.RecordContext: Boolean get_IsDeclaration() +FSharp.Compiler.EditorServices.RecordContext: Boolean get_IsEmpty() FSharp.Compiler.EditorServices.RecordContext: Boolean get_IsNew() +FSharp.Compiler.EditorServices.RecordContext: FSharp.Compiler.EditorServices.RecordContext Empty FSharp.Compiler.EditorServices.RecordContext: FSharp.Compiler.EditorServices.RecordContext NewConstructor(System.String) FSharp.Compiler.EditorServices.RecordContext: FSharp.Compiler.EditorServices.RecordContext NewCopyOnUpdate(FSharp.Compiler.Text.Range, System.Tuple`2[Microsoft.FSharp.Collections.FSharpList`1[System.String],Microsoft.FSharp.Core.FSharpOption`1[System.String]]) FSharp.Compiler.EditorServices.RecordContext: FSharp.Compiler.EditorServices.RecordContext NewDeclaration(Boolean) -FSharp.Compiler.EditorServices.RecordContext: FSharp.Compiler.EditorServices.RecordContext NewNew(System.Tuple`2[Microsoft.FSharp.Collections.FSharpList`1[System.String],Microsoft.FSharp.Core.FSharpOption`1[System.String]]) +FSharp.Compiler.EditorServices.RecordContext: FSharp.Compiler.EditorServices.RecordContext NewNew(System.Tuple`2[Microsoft.FSharp.Collections.FSharpList`1[System.String],Microsoft.FSharp.Core.FSharpOption`1[System.String]], Boolean) +FSharp.Compiler.EditorServices.RecordContext: FSharp.Compiler.EditorServices.RecordContext get_Empty() FSharp.Compiler.EditorServices.RecordContext: FSharp.Compiler.EditorServices.RecordContext+Constructor FSharp.Compiler.EditorServices.RecordContext: FSharp.Compiler.EditorServices.RecordContext+CopyOnUpdate FSharp.Compiler.EditorServices.RecordContext: FSharp.Compiler.EditorServices.RecordContext+Declaration diff --git a/vsintegration/tests/UnitTests/LegacyLanguageService/Tests.LanguageService.Completion.fs b/vsintegration/tests/UnitTests/LegacyLanguageService/Tests.LanguageService.Completion.fs index 83f3a87487c..b79ef889d4b 100644 --- a/vsintegration/tests/UnitTests/LegacyLanguageService/Tests.LanguageService.Completion.fs +++ b/vsintegration/tests/UnitTests/LegacyLanguageService/Tests.LanguageService.Completion.fs @@ -786,15 +786,15 @@ for i in 0..a."] ] let useCases = [ - "let _ = (* MARKER*){X", "(* MARKER*){X", [], ["XX"] + "let _ = (* MARKER*){X }", "(* MARKER*){X", [], ["XX"] "let _ = {(* MARKER*)Mod. = 1; O", "(* MARKER*)Mod.", ["XX"; "YY"], ["System"] "let _ = {(* MARKER*)Mod.Rec. ", "(* MARKER*)Mod.Rec.", ["XX"; "YY"], ["System"] + "let _ = (* MARKER*){Mod.XX = 1; }", "(* MARKER*){Mod.XX = 1; ", ["Mod"], ["XX"; "abs"] ] for (code, marker, should, shouldnot) in useCases do let code = prologue @ [code] - let shouldnot = shouldnot @ ["abs"] - AssertCtrlSpaceCompleteContains code marker should ["abs"] + AssertCtrlSpaceCompleteContains code marker should shouldnot [] [] From c9c1350e34f85182d8132e9a254e6547f304f068 Mon Sep 17 00:00:00 2001 From: Eugene Auduchinok Date: Thu, 24 Mar 2022 18:36:28 +0300 Subject: [PATCH 3/3] Fix qualified field lookup --- src/fsharp/NameResolution.fs | 14 ++++++++------ src/fsharp/NameResolution.fsi | 2 +- src/fsharp/service/FSharpCheckerResults.fs | 19 +++++++++---------- 3 files changed, 18 insertions(+), 17 deletions(-) diff --git a/src/fsharp/NameResolution.fs b/src/fsharp/NameResolution.fs index 0cfbf0dbfbb..2b4e828eb46 100644 --- a/src/fsharp/NameResolution.fs +++ b/src/fsharp/NameResolution.fs @@ -4483,20 +4483,21 @@ let getRecordFieldsInScope nenv = /// allowObsolete - specifies whether we should return obsolete types & modules /// as (no other obsolete items are returned) -let rec ResolvePartialLongIdentToClassOrRecdFields (ncenv: NameResolver) (nenv: NameResolutionEnv) m ad plid (allowObsolete: bool) = - ResolvePartialLongIdentToClassOrRecdFieldsImpl ncenv nenv OpenQualified m ad plid allowObsolete +let rec ResolvePartialLongIdentToClassOrRecdFields (ncenv: NameResolver) (nenv: NameResolutionEnv) m ad plid (allowObsolete: bool) (fieldsOnly: bool) = + ResolvePartialLongIdentToClassOrRecdFieldsImpl ncenv nenv OpenQualified m ad plid allowObsolete fieldsOnly -and ResolvePartialLongIdentToClassOrRecdFieldsImpl (ncenv: NameResolver) (nenv: NameResolutionEnv) fullyQualified m ad plid allowObsolete = +and ResolvePartialLongIdentToClassOrRecdFieldsImpl (ncenv: NameResolver) (nenv: NameResolutionEnv) fullyQualified m ad plid allowObsolete fieldsOnly = let g = ncenv.g match plid with | id :: plid when id = "global" -> // this is deliberately not the mangled name // dive deeper - ResolvePartialLongIdentToClassOrRecdFieldsImpl ncenv nenv FullyQualified m ad plid allowObsolete + ResolvePartialLongIdentToClassOrRecdFieldsImpl ncenv nenv FullyQualified m ad plid allowObsolete fieldsOnly | [] -> // empty plid - return namespaces\modules\record types\accessible fields + if fieldsOnly then getRecordFieldsInScope nenv else let mods = let moduleOrNamespaceRefs = @@ -4524,7 +4525,8 @@ and ResolvePartialLongIdentToClassOrRecdFieldsImpl (ncenv: NameResolver) (nenv: |> Seq.map (ItemOfTyconRef ncenv m) |> Seq.toList - let recdFields = getRecordFieldsInScope nenv + let recdFields = + getRecordFieldsInScope nenv mods @ recdTyCons @ recdFields @@ -4540,7 +4542,7 @@ and ResolvePartialLongIdentToClassOrRecdFieldsImpl (ncenv: NameResolver) (nenv: let qualifiedFields = match rest with - | [] -> + | [] when not fieldsOnly -> // get record types accessible in given nenv let tycons = LookupTypeNameInEnvNoArity OpenQualified id nenv tycons diff --git a/src/fsharp/NameResolution.fsi b/src/fsharp/NameResolution.fsi index cd36ab6362a..3386fe231b7 100755 --- a/src/fsharp/NameResolution.fsi +++ b/src/fsharp/NameResolution.fsi @@ -560,7 +560,7 @@ val internal ResolveExprLongIdent : TcResultsSink -> NameResolver - val internal getRecordFieldsInScope: NameResolutionEnv -> Item list /// Resolve a (possibly incomplete) long identifier to a loist of possible class or record fields -val internal ResolvePartialLongIdentToClassOrRecdFields: NameResolver -> NameResolutionEnv -> range -> AccessorDomain -> string list -> bool -> Item list +val internal ResolvePartialLongIdentToClassOrRecdFields: NameResolver -> NameResolutionEnv -> range -> AccessorDomain -> string list -> bool -> bool -> Item list /// Return the fields for the given class or record val internal ResolveRecordOrClassFieldsOfType : NameResolver -> range -> AccessorDomain -> TType -> bool -> Item list diff --git a/src/fsharp/service/FSharpCheckerResults.fs b/src/fsharp/service/FSharpCheckerResults.fs index 9dd97642506..17c6164d7ea 100644 --- a/src/fsharp/service/FSharpCheckerResults.fs +++ b/src/fsharp/service/FSharpCheckerResults.fs @@ -633,9 +633,9 @@ type internal TypeCheckInfo GetEnvironmentLookupResolutions(nenv, ad, m, plid, filterCtors, showObsolete) /// Find record fields in the best naming environment. - let GetClassOrRecordFieldsEnvironmentLookupResolutions(cursorPos, plid) = + let GetClassOrRecordFieldsEnvironmentLookupResolutions(cursorPos, plid, fieldsOnly) = let (nenv, ad),m = GetBestEnvForPos cursorPos - let items = ResolvePartialLongIdentToClassOrRecdFields ncenv nenv m ad plid false + let items = ResolvePartialLongIdentToClassOrRecdFields ncenv nenv m ad plid false fieldsOnly let items = items |> List.map ItemWithNoInst let items = items |> RemoveDuplicateItems g let items = items |> RemoveExplicitlySuppressed g @@ -909,14 +909,14 @@ type internal TypeCheckInfo items |> List.map DefaultCompletionItem, denv, m /// Find record fields in the best naming environment. - let GetEnvironmentLookupResolutionsIncludingRecordFieldsAtPosition cursorPos envItems = + let GetEnvironmentLookupResolutionsIncludingRecordFieldsAtPosition cursorPos plid envItems = // An empty record expression may be completed into something like these: // { XXX = ... } // { xxx with XXX ... } // Provide both expression items in scope and available record fields. let (nenv, _), m = GetBestEnvForPos cursorPos - let fieldItems = getRecordFieldsInScope nenv |> List.map ItemWithNoInst + let fieldItems, _, _ = GetClassOrRecordFieldsEnvironmentLookupResolutions(cursorPos, plid, true) let fieldCompletionItems, _, _ as fieldsResult = (fieldItems, nenv.DisplayEnv, m) |> toCompletionItems match envItems with @@ -970,10 +970,10 @@ type internal TypeCheckInfo if isFirstField then let cursorPos = mkPos line loc let envItems = GetDeclaredItems (parseResultsOpt, lineStr, origLongIdentOpt, colAtEndOfNamesAndResidue, residueOpt, lastDotPos, line, loc, filterCtors,resolveOverloads, false, fun () -> []) - GetEnvironmentLookupResolutionsIncludingRecordFieldsAtPosition cursorPos envItems + GetEnvironmentLookupResolutionsIncludingRecordFieldsAtPosition cursorPos plid envItems else // { x. } can be either record construction or computation expression. Try to get all visible record fields first - match GetClassOrRecordFieldsEnvironmentLookupResolutions(mkPos line loc, plid) |> toCompletionItems with + match GetClassOrRecordFieldsEnvironmentLookupResolutions(mkPos line loc, plid, false) |> toCompletionItems with | [],_,_ -> // no record fields found, return completion list as if we were outside any computation expression GetDeclaredItems (parseResultsOpt, lineStr, origLongIdentOpt, colAtEndOfNamesAndResidue, residueOpt, lastDotPos, line, loc, filterCtors,resolveOverloads, false, fun() -> []) @@ -983,14 +983,13 @@ type internal TypeCheckInfo | Some(CompletionContext.RecordField RecordContext.Empty) -> let cursorPos = mkPos line loc let envItems = GetDeclaredItems (parseResultsOpt, lineStr, origLongIdentOpt, colAtEndOfNamesAndResidue, residueOpt, lastDotPos, line, loc, filterCtors,resolveOverloads, false, fun () -> []) - GetEnvironmentLookupResolutionsIncludingRecordFieldsAtPosition cursorPos envItems - + GetEnvironmentLookupResolutionsIncludingRecordFieldsAtPosition cursorPos [] envItems // Completion at ' { XXX = ... with ... } " | Some(CompletionContext.RecordField(RecordContext.CopyOnUpdate(r, (plid, _)))) -> match GetRecdFieldsForExpr(r) with | None -> - Some (GetClassOrRecordFieldsEnvironmentLookupResolutions(mkPos line loc, plid)) + Some (GetClassOrRecordFieldsEnvironmentLookupResolutions(mkPos line loc, plid, false)) |> Option.map toCompletionItems | Some (items, denv, m) -> Some (List.map ItemWithNoInst items, denv, m) @@ -998,7 +997,7 @@ type internal TypeCheckInfo // Completion at ' { XXX = ... with ... } " | Some(CompletionContext.RecordField(RecordContext.Constructor(typeName))) -> - Some(GetClassOrRecordFieldsEnvironmentLookupResolutions(mkPos line loc, [typeName])) + Some(GetClassOrRecordFieldsEnvironmentLookupResolutions(mkPos line loc, [typeName], false)) |> Option.map toCompletionItems // No completion at '...: string'