Make auto-imports more likely to be valid for the file (including JS) & project settings #32684
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Fixes #31219
Fixes #29038
Reincarnation of #32150
The Problem
For a given combination of script kind (JS or TS), module target, module interop settings (
esModuleInterop
andallowSyntheticDefaultImports
), and exported symbol that you want to import, there are anywhere between zero and two correct-ish ways to write an import for that exported symbol. Prior to this PR,was always auto-imported as
This is never valid when either of the following are true:
It’s also potentially undesirable, while not wrong, when all of the following are true:
esModuleInterop
orallowSyntheticDefaultImports
is trueIn this scenario, both
import x = require("./x")
andimport x from "./x"
are valid, and it’s debatable which one should be preferred.The Solution
The No-Brainer Part: Use the Default Import When It’s The Only Valid Choice
When the module target is es2015+ and
esModuleInterop
orallowSyntheticDefaultImports
is enabled and the importing file is TypeScript, the default import is the only thing that works forexport =
. (When the importing file is JavaScript, the intent is also very likely to have a default import, even thoughconst x = require('./x')
is an option, so this logic applies to JS too.)The JavaScript Part: Use Simple Usage Heuristics
Relying heavily on compiler options to make decisions about what to do in a JavaScript file isn’t a great idea, because many users won’t have a tsconfig/jsconfig at all, and will just be going with the defaults. It’s highly likely that, if there are any signs of the ES module system in the file, they’re working with a bundler that does have ES module support, and we should essentially pretend
esModuleInterop
is turned on and give them the default import.If it doesn’t look like the importing JS file is an ES module, we’ll give them
const x = require("./x")
, which we didn’t support at all before this PR.Additionally, UMD imports are affected here, as JavaScript without
esModuleInterop
/allowSyntheticDefaultImports
/es2015+ module target was getting the TypeScript-specificimport x = require("./x")
. Now, if the file looks like an ES module and it’s JavaScript, it gets a namespace import (import * as x from "./x"
), and if it doesn’t look like a module it getsconst x = require("./x")
(although currently, I don’t believe that path ever gets taken since a non-module will be happy with using the UMD export as a global).The Other Part: Use More Sophisticated Heuristics
When the importing file is TypeScript and
esModuleInterop
orallowSyntheticDefaultImports
is enabled and the module target is CommonJS or AMD or UMD, bothimport x = require('./x')
andimport x from './x'
will work. The former is arguably more correct, but the latter is arguably preferable given the compiler options. After listening to differing opinions amongst the team, I decided to ignore all of them, including my own, and listen to the user’s opinion instead. We search for an existing default import that’s being used to import anexport =
, and if we find one, we give themimport x from './x'
. If we don‘t find one, we default toimport x = require('./x')
.The Part I Didn’t Do: JavaScript
require
For ES ModulesOn one hand, if we’re adding support for inserting
const x = require('./x')
for a CommonJS export in JavaScript when we think you may not be able to useimport
, you could argue that we should follow suit and also offerconst { y } = require('./x')
andconst z = require('./z').default
for named and default ESM exports, respectively. On the other hand, you could also argue that those are not a valid way to import ES modules, so we’re offering you the only thing that’s technically correct. On the other other hand, you could argue that in today’s ecosystem, modules that TypeScript thinks are ES modules are almost certainly actually CommonJS with interop stuff baked in, sorequire
would work and we should offer it in the name of pragmatism.I wasn’t sure what to do with this, so I left it as-is, which means in a JavaScript file you could end up taking an auto-import that gives you
const x = require('./x')
followed by one that gives youimport { y } from './y'
which is a little odd.