-
Notifications
You must be signed in to change notification settings - Fork 12.6k
Type Checking JavaScript Files
TypeScript 2.3 and later support type-checking and reporting errors in .js
files with --checkJs
.
You can skip checking some files by adding // @ts-nocheck
comment to them; conversely, you can choose to check only a few .js
files by adding a // @ts-check
comment to them without setting --checkJs
.
You can also ignore errors on specific lines by adding // @ts-ignore
on the preceding line.
Here are some notable differences on how checking work in .js
file from .ts
file:
In a .js
file, types can often be inferred just like in .ts
files.
Likewise, when types can't be inferred, they can be specified using JSDoc the same way that type annotations do in a .ts
file.
Just like Typescript, --noImplicitAny
will give you errors on the places that the compiler could not infer a type.
(With the exception of open-ended object literals; see below for details.)
JSDoc annotations adorning a declaration will be used to set the type of that declaration. For example:
/** @type {number} */
var x;
x = 0; // OK
x = false; // Error: boolean is not assignable to number
You can find the full list of supported JSDoc patterns in the JSDoc support in JavaScript documentation.
ES2015/ES6 does not have a means for declaring properties on classes. Properties are dynamically assigned, just like in the case of object literals.
In a .js
file, property declarations are inferred from assignments to the properties inside the class body.
The type of properties is the type given in the constructor, unless it's not defined there, or the type in the constructor is undefined or null.
In that case, the type is the union of the types of all the right-hand values in these assignments.
Properties defined in the constructor are always assumed to exist, where as ones defined just in methods, getters, or setters are considered optional.
class C {
constructor() {
this.x = 0
}
method() {
this.x = false // error, x is a number
this.y = 'ok' // ok, but y could also be undefined
}
method2() {
this.y = true // also, ok, y's type is string | boolean | undefined
}
}
If the property type can't be inferred, annotate the assignment in the constructor with JSDoc to specify the type. You don't even have to give a value if it will be initialised later:
class C {
constructor() {
/** @type {number | undefined} */
this.prop = undefined;
/** @type {number | undefined} */
this.count;
}
}
let c = new C();
c.prop = 0; // OK
c.count = "string"; // Error: string is not assignable to number|undefined
If properties are never set in the class body, they are considered unknown. If your class has properties that are only read from, consider adding an initialization in the constructor to undefined, e.g. this.prop = undefined;
.
Any variable, parameter or property that is initialized with null or undefined will have type any, even if strict null checks is turned on. Any variable, parameter or property that is initialized with [] will have type any[], even if strict null checks is turned on. The only exception is for properties that have multiple initializers as described above.
Right down to methods. Lots of detail needs to go here. Kind of surprised there was none.
Constructor functions are namespaces:
function Outer() {
this.y = 2
}
Outer.Inner = class {
}
TODO: Other kinds of initializers
TODO: Much more
In a .js
files CommonJS module format is allowed as an input module format. Assignments to exports
, and module.exports
are recognized as export declarations. Similarly, require
function calls are recognized as module imports. For example:
// same as `import module "fs"`
const fs = require("fs");
// same as `export function readFile`
module.exports.readFile = function(f) {
return fs.readFileSync(f);
}
The module support in Javascript is much more syntactically forgiving than Typescript's module support. Most combinations of assignments and declarations are supported.
By default object literals in variable declarations provide the type of a declaration. No new members can be added that were not specified in the original initialization. This rule is relaxed in a .js
file; object literals have an open-ended type, allowing adding and looking up properties that were not defined originally. For instance:
var obj = { a: 1 };
obj.b = 2; // Allowed
Object literals get a default index signature [x:string]: any
that allows them to be treated as open maps instead of closed objects.
Similar to other special JS checking behaviors, this behavior can be changed by specifying a JSDoc type for the variable. For example:
/** @type {{a: number}} */
var obj = { a: 1 };
obj.b = 2; // Error, type {a: number} does not have property b
Since there is no way to specify optionality on parameters in JS (without specifying a default value), all function parameters in .js
file are considered optional. Calls with fewer arguments are allowed.
It is important to note that it is an error to call a function with too many arguments.
For instance:
function bar(a, b){
console.log(a + " " + b);
}
bar(1); // OK, second argument considered optional
bar(1, 2);
bar(1, 2, 3); // Error, too many arguments
JSDoc annotated functions are excluded from this rule. Use JSDoc optional parameter syntax to express optionality. e.g.:
/**
* @param {string} [somebody] - Somebody's name.
*/
function sayHello(somebody) {
if (!somebody) {
somebody = 'John Doe';
}
alert('Hello ' + somebody);
}
sayHello();
A function whose body has a reference to the arguments
reference is implicitly considered to have a var-arg parameter (i.e. (...arg: any[]) => any
). Use JSDoc var-arg syntax to specify the type of the arguments.
TODO: Give an example of the correct jsdoc
An unspecified generic type parameter defaults to any
. There are few places where this happens:
For instance, React.Component
is defined to have two generic type parameters, Props
and State
.
In a .js
file, there is no legal way to specify these in the extends clause. By default the type arguments will be any
:
import { Component } from "react";
class MyComponent extends Component {
render() {
this.props.b; // Allowed, since this.props is of type any
}
}
Use JSDoc @augments
to specify the types explicitly. for instance:
import { Component } from "react";
/**
* @augments {Component<{a: number}, State>}
*/
class MyComponent extends Component {
render() {
this.props.b; // Error: b does not exist on {a:number}
}
}
An unspecified generic type argument in JSDoc defaults to any:
/** @type{Array} */
var x = [];
x.push(1); // OK
x.push("string"); // OK, x is of type Array<any>
/** @type{Array.<number>} */
var y = [];
y.push(1); // OK
y.push("string"); // Error, string is not assignable to number
A call to generic functions uses arguments to infer the generic type parameters. Sometimes this process fails to infer any types, mainly because of lack on inference sources; in these cases, the generic type parameters will default to any
. For example:
var p = new Promise((resolve, reject) => { reject() });
p; // Promise<any>;
News
Debugging TypeScript
- Performance
- Performance-Tracing
- Debugging-Language-Service-in-VS-Code
- Getting-logs-from-TS-Server-in-VS-Code
- JavaScript-Language-Service-in-Visual-Studio
- Providing-Visual-Studio-Repro-Steps
Contributing to TypeScript
- Contributing to TypeScript
- TypeScript Design Goals
- Coding Guidelines
- Useful Links for TypeScript Issue Management
- Writing Good Design Proposals
- Compiler Repo Notes
- Deployment
Building Tools for TypeScript
- Architectural Overview
- Using the Compiler API
- Using the Language Service API
- Standalone Server (tsserver)
- TypeScript MSBuild In Depth
- Debugging Language Service in VS Code
- Writing a Language Service Plugin
- Docker Quickstart
FAQs
The Main Repo