## Using TypeScript Deno supports both JavaScript and TypeScript as first class languages at runtime. This means it requires fully qualified module names, including the extension (or a server providing the correct media type). In addition, Deno has no "magical" module resolution. Instead, imported modules are specified as files (including extensions) or fully qualified URL imports. Typescript modules can be directly imported. E.g. ``` import { Response } from "https://deno.land/std@$STD_VERSION/http/server.ts"; import { queue } from "./collections.ts"; ``` ### `--no-check` option When using `deno run`, `deno test`, `deno cache`, `deno info`, or `deno bundle` you can specify the `--no-check` flag to disable TypeScript type checking. This can significantly reduce the time that program startup takes. This can be very useful when type checking is provided by your editor and you want startup time to be as fast as possible (for example when restarting the program automatically with a file watcher). Because `--no-check` does not do TypeScript type checking we can not automatically remove type only imports and exports as this would require type information. For this purpose TypeScript provides the [`import type` and `export type` syntax](https://www.typescriptlang.org/docs/handbook/release-notes/typescript-3-8.html#type-only-imports-and-exports). To export a type in a different file use `export type { AnInterface } from "./mod.ts";`. To import a type use `import type { AnInterface } from "./mod.ts";`. You can check that you are using `import type` and `export type` where necessary by setting the `isolatedModules` TypeScript compiler option to `true`. You can see an example `tsconfig.json` with this option [in the standard library](https://github.com/denoland/deno/blob/master/std/tsconfig_test.json). Because there is no type information when using `--no-check`, `const enum` is not supported because it is type-directed. `--no-check` also does not support the legacy `import =` and `export =` syntax. ### Using external type definitions The out of the box TypeScript compiler though relies on both extension-less modules and the Node.js module resolution logic to apply types to JavaScript modules. In order to bridge this gap, Deno supports three ways of referencing type definition files without having to resort to "magic" resolution. #### Compiler hint If you are importing a JavaScript module, and you know where the type definition for that module is located, you can specify the type definition at import. This takes the form of a compiler hint. Compiler hints inform Deno the location of `.d.ts` files and the JavaScript code that is imported that they relate to. The hint is `@deno-types` and when specified the value will be used in the compiler instead of the JavaScript module. For example, if you had `foo.js`, but you know that alongside of it was `foo.d.ts` which was the types for the file, the code would look like this: ```ts // @deno-types="./foo.d.ts" import * as foo from "./foo.js"; ``` The value follows the same resolution logic as importing a module, meaning the file needs to have an extension and is relative to the current module. Remote specifiers are also allowed. The hint affects the next `import` statement (or `export ... from` statement) where the value of the `@deno-types` will be substituted at compile time instead of the specified module. Like in the above example, the Deno compiler will load `./foo.d.ts` instead of `./foo.js`. Deno will still load `./foo.js` when it runs the program. #### Triple-slash reference directive in JavaScript files If you are hosting modules which you want to be consumed by Deno, and you want to inform Deno about the location of the type definitions, you can utilize a triple-slash directive in the actual code. For example, if you have a JavaScript module and you would like to provide Deno with the location of the type definition which happens to be alongside that file, your JavaScript module named `foo.js` might look like this: ```js /// export const foo = "foo"; ``` Deno will see this, and the compiler will use `foo.d.ts` when type checking the file, though `foo.js` will be loaded at runtime. The resolution of the value of the directive follows the same resolution logic as importing a module, meaning the file needs to have an extension and is relative to the current file. Remote specifiers are also allowed. #### X-TypeScript-Types custom header If you are hosting modules which you want to be consumed by Deno, and you want to inform Deno the location of the type definitions, you can use a custom HTTP header of `X-TypeScript-Types` to inform Deno of the location of that file. The header works in the same way as the triple-slash reference mentioned above, it just means that the content of the JavaScript file itself does not need to be modified, and the location of the type definitions can be determined by the server itself. **Not all type definitions are supported.** Deno will use the compiler hint to load the indicated `.d.ts` files, but some `.d.ts` files contain unsupported features. Specifically, some `.d.ts` files expect to be able to load or reference type definitions from other packages using the module resolution logic. For example a type reference directive to include `node`, expecting to resolve to some path like `./node_modules/@types/node/index.d.ts`. Since this depends on non-relative "magical" resolution, Deno cannot resolve this. **Why not use the triple-slash type reference in TypeScript files?** The TypeScript compiler supports triple-slash directives, including a type reference directive. If Deno used this, it would interfere with the behavior of the TypeScript compiler. Deno only looks for the directive in JavaScript (and JSX) files. ### Custom TypeScript Compiler Options In the Deno ecosystem, all strict flags are enabled in order to comply with TypeScript's ideal of being `strict` by default. However, in order to provide a way to support customization a configuration file such as `tsconfig.json` might be provided to Deno on program execution. You need to explicitly tell Deno where to look for this configuration by setting the `-c` (or `--config`) argument when executing your application. ```shell deno run -c tsconfig.json mod.ts ``` Following are the currently allowed settings and their default values in Deno: ```json { "compilerOptions": { "allowJs": false, "allowUmdGlobalAccess": false, "allowUnreachableCode": false, "allowUnusedLabels": false, "alwaysStrict": true, "assumeChangesOnlyAffectDirectDependencies": false, "checkJs": false, "disableSizeLimit": false, "generateCpuProfile": "profile.cpuprofile", "jsx": "react", "jsxFactory": "React.createElement", "lib": [], "noFallthroughCasesInSwitch": false, "noImplicitAny": true, "noImplicitReturns": true, "noImplicitThis": true, "noImplicitUseStrict": false, "noStrictGenericChecks": false, "noUnusedLocals": false, "noUnusedParameters": false, "preserveConstEnums": false, "removeComments": false, "resolveJsonModule": true, "strict": true, "strictBindCallApply": true, "strictFunctionTypes": true, "strictNullChecks": true, "strictPropertyInitialization": true, "suppressExcessPropertyErrors": false, "suppressImplicitAnyIndexErrors": false, "useDefineForClassFields": false } } ``` For documentation on allowed values and use cases please visit the [typescript docs](https://www.typescriptlang.org/docs/handbook/compiler-options.html). **Note**: Any options not listed above are either not supported by Deno or are listed as deprecated/experimental in the TypeScript documentation.