Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Use type-only imports in auto-imports when it would be an error not to, and use auto-imports in “implement interface” fix #36615

Merged
merged 8 commits into from
Feb 11, 2020

Conversation

andrewbranch
Copy link
Member

@andrewbranch andrewbranch commented Feb 4, 2020

  1. When importsNotUsedAsValues is "error", an auto-import at a type position will use a type-only import if a regular import would result in an error.
  2. Encapsulates the fix-all logic for auto-imports in order to use it in other code fixes that would otherwise write type annotations with ImportTypeNodes, like “infer from usage,” “implement interface,” and “implement abstract class members.” (Infer from usage had an ad-hoc fix to use auto-imports already back at Allow inferFromUsage to do auto-imports #33915, but I realized that if it needs to import more than one type (or the same type multiple times), it would write duplicate imports since the code fix was being reevaluated fresh for each ImportTypeNode.)

Fixes #34995
Fixes #35078 but doesn’t create a new option—I would rather wait and see if anyone is actually asking for an opt-out than start by introducing another option that nobody wants.

@andrewbranch andrewbranch requested a review from sandersn February 6, 2020 18:37
Copy link
Member

@sandersn sandersn left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Some initial comments. I'm still reading importFixes and helpers

Copy link
Contributor

@mjbvz mjbvz left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Description of new behavior sounds good to me!

Copy link
Member

@sandersn sandersn left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good to me barring a few comments on documentation.

function getImportFixForSymbol(sourceFile: SourceFile, exportInfos: readonly SymbolExportInfo[], moduleSymbol: Symbol, symbolName: string, program: Program, position: number | undefined, preferTypeOnlyImport: boolean, host: LanguageServiceHost, preferences: UserPreferences) {
Debug.assert(exportInfos.some(info => info.moduleSymbol === moduleSymbol), "Some exportInfo should match the specified moduleSymbol");
// We sort the best codefixes first, so taking `first` is best.
return first(getFixForImport(exportInfos, symbolName, position, preferTypeOnlyImport, program, sourceFile, host, preferences));
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe getFixForImport should be renamed getImportFix

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The names in this file are, at least to me, confusing, repetitive, and unexpressive, but they’re roughly consistent in including an unnecessary “for.” Someday I’d like to do some housekeeping in here, but I think renaming just one function now might not be a useful step forward.

@jamiebuilds
Copy link

jamiebuilds commented Dec 1, 2020

Edit: ignore me, I had bad vscode settings. For any future person, don't do this:

"files.associations": {
  "*.tsx": "javascript"
}
ignore all this

@andrewbranch I think I'm gonna be the person to ask for an explicit option again. I keep having import("module").Type types generated whenever I have a file with zero other imports in the file (edit: never mind the number of imports does not appear to matter), and I don't ever want that behavior.

I think these are all the relevant options I have set:

{
  "target": "es2017",
  "module": "esnext",
  "isolatedModules": true,
  "importsNotUsedAsValues": "error",
  "moduleResolution": "node",
  "allowSyntheticDefaultImports": true,
  "esModuleInterop": true
}

Edit: I'm in hell, now I can't make this behavior go away. I created a new TS project with the default settings of tsc --init with ts@4.1.2 with all my extensions disabled.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
6 participants