@@ -56,11 +56,13 @@ async function typeCheckFiles(
56
56
filenames : string [ ] ,
57
57
) {
58
58
const tscExec = path . join ( require . resolve ( 'typescript' ) , '..' , '..' , 'bin' , 'tsc' ) ;
59
+ const options = [ '--noEmit' , '--pretty' , '--moduleDetection' , 'force' ] ;
60
+ if ( filenames . find ( ( filename ) => filename . endsWith ( '.js' ) ) ) {
61
+ options . push ( '--checkJs' ) ;
62
+ }
59
63
const args = [
60
64
tscExec ,
61
- '--noEmit' ,
62
- '--checkJs' ,
63
- '--pretty' ,
65
+ ...options ,
64
66
path . join ( tempDir , 'electron.d.ts' ) ,
65
67
path . join ( tempDir , 'ambient.d.ts' ) ,
66
68
...filenames ,
@@ -113,7 +115,7 @@ async function main(workspaceRoot: string, globs: string[], { ignoreGlobs = [] }
113
115
try {
114
116
const filenames : string [ ] = [ ] ;
115
117
const originalFilenames = new Map < string , string > ( ) ;
116
- const windowTypeFilenames = new Set < string > ( ) ;
118
+ const isolateFilenames = new Set < string > ( ) ;
117
119
118
120
// Copy over the typings so that a relative path can be used
119
121
fs . copyFileSync ( path . join ( process . cwd ( ) , 'electron.d.ts' ) , path . join ( tempDir , 'electron.d.ts' ) ) ;
@@ -124,15 +126,19 @@ async function main(workspaceRoot: string, globs: string[], { ignoreGlobs = [] }
124
126
for ( const document of await workspace . getAllMarkdownDocuments ( ) ) {
125
127
const uri = URI . parse ( document . uri ) ;
126
128
const filepath = workspace . getWorkspaceRelativePath ( uri ) ;
127
- const jsCodeBlocks = ( await getCodeBlocks ( document ) ) . filter (
128
- ( code ) => code . lang && [ 'javascript ', 'js '] . includes ( code . lang . toLowerCase ( ) ) ,
129
+ const codeBlocks = ( await getCodeBlocks ( document ) ) . filter (
130
+ ( code ) =>
131
+ code . lang && [ 'javascript' , 'js' , 'typescript' , 'ts' ] . includes ( code . lang . toLowerCase ( ) ) ,
129
132
) ;
130
133
131
- for ( const codeBlock of jsCodeBlocks ) {
134
+ for ( const codeBlock of codeBlocks ) {
135
+ const isTypeScript =
136
+ codeBlock . lang && [ 'typescript' , 'ts' ] . includes ( codeBlock . lang . toLowerCase ( ) ) ;
132
137
const line = codeBlock . position ! . start . line ;
133
138
const indent = codeBlock . position ! . start . column - 1 ;
134
139
135
140
const tsNoCheck = codeBlock . meta ?. split ( ' ' ) . includes ( '@ts-nocheck' ) ;
141
+ const tsNoIsolate = codeBlock . meta ?. split ( ' ' ) . includes ( '@ts-noisolate' ) ;
136
142
const tsExpectErrorLines = codeBlock . meta
137
143
?. match ( / \B @ t s - e x p e c t - e r r o r = \[ ( [ \d , ] * ) \] \B / ) ?. [ 1 ]
138
144
. split ( ',' )
@@ -233,10 +239,12 @@ async function main(workspaceRoot: string, globs: string[], { ignoreGlobs = [] }
233
239
. join ( '\n' ) ,
234
240
) ;
235
241
236
- // If there are no require() lines, insert a default set of
242
+ // If there are no require() or import lines, insert a default set of
237
243
// imports so that most snippets will have what they need.
238
244
// This isn't foolproof and might cause name conflicts
239
- const imports = codeBlock . value . includes ( ' require(' ) ? '' : DEFAULT_IMPORTS ;
245
+ const imports = codeBlock . value . match ( / ^ \s * (?: i m p o r t .* f r o m ) | (?: .* = r e q u i r e ( ) ) / m)
246
+ ? ''
247
+ : DEFAULT_IMPORTS ;
240
248
241
249
// Insert the necessary number of blank lines so that the line
242
250
// numbers in output from tsc is accurate to the original file
@@ -248,7 +256,7 @@ async function main(workspaceRoot: string, globs: string[], { ignoreGlobs = [] }
248
256
tempDir ,
249
257
`${ filepath
250
258
. replace ( new RegExp ( path . sep . replace ( / \\ / g, '\\\\' ) , 'g' ) , '-' )
251
- . replace ( / \. / g, '-' ) } -${ line } .js `,
259
+ . replace ( / \. / g, '-' ) } -${ line } .${ isTypeScript ? 'ts' : 'js' } `,
252
260
) ;
253
261
254
262
// Blocks can have @ts -type={name:type} in their info
@@ -272,14 +280,16 @@ async function main(workspaceRoot: string, globs: string[], { ignoreGlobs = [] }
272
280
273
281
// Blocks can have @ts-window-type={name:type} in their
274
282
// info string to extend the Window object for a block
275
- if ( tsWindowTypeLines . length ) {
283
+ if (!tsNoIsolate && tsWindowTypeLines.length) {
276
284
const extraTypes = tsWindowTypeLines
277
285
.map((type) => ` $ { type [ 1 ] } : ${type [ 2 ] } ; `)
278
286
. join ( '\n' ) ;
279
287
// Needs an export {} at the end to make TypeScript happy
280
288
windowTypes = `declare global { \n interface Window { \n${extraTypes } \n } \n} \n\nexport { } ; \n\n`;
281
- fs . writeFileSync ( filename . replace ( / .j s $ / , '-window.d.ts' ) , windowTypes ) ;
282
- windowTypeFilenames . add ( filename ) ;
289
+ fs . writeFileSync ( filename . replace ( / .[ j t ] s $ / , '-window.d.ts' ) , windowTypes ) ;
290
+ isolateFilenames . add ( filename ) ;
291
+ } else if ( ! tsNoIsolate && code . match ( / ^ \s * d e c l a r e g l o b a l / m) ) {
292
+ isolateFilenames . add ( filename ) ;
283
293
} else {
284
294
filenames . push ( filename ) ;
285
295
}
@@ -291,13 +301,18 @@ async function main(workspaceRoot: string, globs: string[], { ignoreGlobs = [] }
291
301
292
302
fs . writeFileSync ( path . join ( tempDir , 'ambient.d.ts' ) , ambientModules ) ;
293
303
294
- // Files for code blocks with window types have to be processed separately
295
- // since window types are by nature global, and would bleed between blocks
296
- for ( const filename of windowTypeFilenames ) {
297
- const status = await typeCheckFiles ( tempDir , originalFilenames , [
298
- filename . replace ( / .j s $ / , '-window.d.ts' ) ,
299
- filename ,
300
- ] ) ;
304
+ // Files for code blocks with window type directives or 'declare global' need
305
+ // to be processed separately since window types are by nature global, and
306
+ // they would bleed between blocks otherwise, which can cause problems
307
+ for ( const filename of isolateFilenames ) {
308
+ const filenames = [ filename ] ;
309
+ const windowTypesFilename = filename . replace ( / .[ j t ] s $ / , '-window.d.ts' ) ;
310
+ try {
311
+ fs . statSync ( windowTypesFilename ) ;
312
+ filenames . unshift ( windowTypesFilename ) ;
313
+ } catch { }
314
+
315
+ const status = await typeCheckFiles ( tempDir , originalFilenames , filenames ) ;
301
316
errors = errors || status !== 0 ;
302
317
}
303
318
0 commit comments