From 7b61fa23938b8289e1e04e895174f405c4f388e4 Mon Sep 17 00:00:00 2001 From: TypeScript Bot Date: Wed, 30 Jun 2021 14:33:26 -0700 Subject: [PATCH] Cherry-pick PR #44724 into release-4.3 (#44822) Component commits: 55459467d6 Fix auto imports in opening JSX tag Co-authored-by: Andrew Branch --- src/services/completions.ts | 17 +++++---- .../fourslash/completionsImportFromJSXTag.ts | 35 +++++++++++++++++++ 2 files changed, 43 insertions(+), 9 deletions(-) create mode 100644 tests/cases/fourslash/completionsImportFromJSXTag.ts diff --git a/src/services/completions.ts b/src/services/completions.ts index 5ed1608fbcdcb..65ffbb5073e11 100644 --- a/src/services/completions.ts +++ b/src/services/completions.ts @@ -1275,10 +1275,9 @@ namespace ts.Completions { getTypeScriptMemberSymbols(); } else if (isRightOfOpenTag) { - const tagSymbols = typeChecker.getJsxIntrinsicTagNamesAt(location); - Debug.assertEachIsDefined(tagSymbols, "getJsxIntrinsicTagNames() should all be defined"); + symbols = typeChecker.getJsxIntrinsicTagNamesAt(location); + Debug.assertEachIsDefined(symbols, "getJsxIntrinsicTagNames() should all be defined"); tryGetGlobalSymbols(); - symbols = tagSymbols.concat(symbols); completionKind = CompletionKind.Global; keywordFilters = KeywordCompletionFilters.None; } @@ -1564,7 +1563,7 @@ namespace ts.Completions { const attrsType = jsxContainer && typeChecker.getContextualType(jsxContainer.attributes); if (!attrsType) return GlobalsSearch.Continue; const completionsType = jsxContainer && typeChecker.getContextualType(jsxContainer.attributes, ContextFlags.Completions); - symbols = filterJsxAttributes(getPropertiesForObjectExpression(attrsType, completionsType, jsxContainer!.attributes, typeChecker), jsxContainer!.attributes.properties); + symbols = concatenate(symbols, filterJsxAttributes(getPropertiesForObjectExpression(attrsType, completionsType, jsxContainer!.attributes, typeChecker), jsxContainer!.attributes.properties)); setSortTextToOptionalMember(); completionKind = CompletionKind.MemberLike; isNewIdentifierLocation = false; @@ -1621,7 +1620,7 @@ namespace ts.Completions { const symbolMeanings = (isTypeOnly ? SymbolFlags.None : SymbolFlags.Value) | SymbolFlags.Type | SymbolFlags.Namespace | SymbolFlags.Alias; - symbols = typeChecker.getSymbolsInScope(scopeNode, symbolMeanings); + symbols = concatenate(symbols, typeChecker.getSymbolsInScope(scopeNode, symbolMeanings)); Debug.assertEachIsDefined(symbols, "getSymbolsInScope() should all be defined"); for (const symbol of symbols) { if (!typeChecker.isArgumentsSymbol(symbol) && @@ -1957,7 +1956,7 @@ namespace ts.Completions { const existingMemberEscapedNames: Set<__String> = new Set(); existingMembers.forEach(s => existingMemberEscapedNames.add(s.escapedName)); - symbols = filter(members, s => !existingMemberEscapedNames.has(s.escapedName)); + symbols = concatenate(symbols, filter(members, s => !existingMemberEscapedNames.has(s.escapedName))); completionKind = CompletionKind.ObjectPropertyDeclaration; isNewIdentifierLocation = true; @@ -2046,7 +2045,7 @@ namespace ts.Completions { if (typeMembers && typeMembers.length > 0) { // Add filtered items to the completion list - symbols = filterObjectMembersList(typeMembers, Debug.checkDefined(existingMembers)); + symbols = concatenate(symbols, filterObjectMembersList(typeMembers, Debug.checkDefined(existingMembers))); } setSortTextToOptionalMember(); @@ -2082,7 +2081,7 @@ namespace ts.Completions { isNewIdentifierLocation = false; const exports = typeChecker.getExportsAndPropertiesOfModule(moduleSpecifierSymbol); const existing = new Set((namedImportsOrExports.elements as NodeArray).filter(n => !isCurrentlyEditingNode(n)).map(n => (n.propertyName || n.name).escapedText)); - symbols = exports.filter(e => e.escapedName !== InternalSymbolName.Default && !existing.has(e.escapedName)); + symbols = concatenate(symbols, exports.filter(e => e.escapedName !== InternalSymbolName.Default && !existing.has(e.escapedName))); return GlobalsSearch.Success; } @@ -2161,7 +2160,7 @@ namespace ts.Completions { type?.symbol && typeChecker.getPropertiesOfType(typeChecker.getTypeOfSymbolAtLocation(type.symbol, decl)) : type && typeChecker.getPropertiesOfType(type); }); - symbols = filterClassMembersList(baseSymbols, decl.members, classElementModifierFlags); + symbols = concatenate(symbols, filterClassMembersList(baseSymbols, decl.members, classElementModifierFlags)); } return GlobalsSearch.Success; diff --git a/tests/cases/fourslash/completionsImportFromJSXTag.ts b/tests/cases/fourslash/completionsImportFromJSXTag.ts new file mode 100644 index 0000000000000..b3c016b186e6b --- /dev/null +++ b/tests/cases/fourslash/completionsImportFromJSXTag.ts @@ -0,0 +1,35 @@ +/// + +// @jsx: react + +// @Filename: /types.d.ts +//// declare namespace JSX { +//// interface IntrinsicElements { a } +//// } + +// @Filename: /Box.tsx +//// export function Box(props: any) { return null; } + +// @Filename: /App.tsx +//// export function App() { +//// return ( +////
+//// +//// ) +//// } + +verify.applyCodeActionFromCompletion("", { + name: "Box", + source: "/Box", + description: `Import 'Box' from module "./Box"`, + newFileContent: `import { Box } from "./Box"; + +export function App() { + return ( +
+ + ) +}` +});