1
- import { stdin , stdout } from 'node:process' ;
2
- import readline , { type Key , type ReadLine } from 'node:readline' ;
3
- import type { Readable , Writable } from 'node:stream' ;
4
- import { WriteStream } from 'node:tty' ;
5
- import { cursor , erase } from 'sisteransi' ;
6
- import wrap from 'wrap-ansi' ;
7
- import { strLength } from '../utils' ;
8
-
9
- import { ALIASES , CANCEL_SYMBOL , KEYS , diffLines , hasAliasKey , setRawMode } from '../utils' ;
10
-
11
- import type { ClackEvents , ClackState , InferSetType } from '../types' ;
1
+ import { stdin , stdout } from "node:process" ;
2
+ import readline , { type Key , type ReadLine } from "node:readline" ;
3
+ import type { Readable , Writable } from "node:stream" ;
4
+ import { WriteStream } from "node:tty" ;
5
+ import { cursor , erase } from "sisteransi" ;
6
+ import wrap from "wrap-ansi" ;
7
+ import { strLength } from "../utils" ;
8
+
9
+ import {
10
+ ALIASES ,
11
+ CANCEL_SYMBOL ,
12
+ KEYS ,
13
+ diffLines ,
14
+ hasAliasKey ,
15
+ setRawMode ,
16
+ } from "../utils" ;
17
+
18
+ import type { ClackEvents , ClackState , InferSetType } from "../types" ;
12
19
13
20
export interface PromptOptions < Self extends Prompt > {
14
- render ( this : Omit < Self , ' prompt' > ) : string | undefined ;
21
+ render ( this : Omit < Self , " prompt" > ) : string | undefined ;
15
22
placeholder ?: string ;
16
23
initialValue ?: any ;
17
24
validate ?: ( ( value : any ) => string | undefined ) | undefined ;
@@ -20,9 +27,7 @@ export interface PromptOptions<Self extends Prompt> {
20
27
debug ?: boolean ;
21
28
}
22
29
23
- export type State = 'initial' | 'active' | 'cancel' | 'submit' | 'error' ;
24
-
25
- export type LineOption = 'firstLine' | 'newLine' | 'lastLine' ;
30
+ export type LineOption = "firstLine" | "newLine" | "lastLine" ;
26
31
27
32
export interface FormatLineOptions {
28
33
/**
@@ -71,7 +76,8 @@ export interface FormatLineOptions {
71
76
style : ( line : string ) => string ;
72
77
}
73
78
74
- export interface FormatOptions extends Record < LineOption , Partial < FormatLineOptions > > {
79
+ export interface FormatOptions
80
+ extends Record < LineOption , Partial < FormatLineOptions > > {
75
81
/**
76
82
* Shorthand to define values for each line
77
83
* @example
@@ -109,15 +115,18 @@ export default class Prompt {
109
115
protected output : Writable ;
110
116
111
117
private rl ! : ReadLine ;
112
- private opts : Omit < PromptOptions < Prompt > , ' render' | ' input' | ' output' > ;
113
- private _render : ( context : Omit < Prompt , ' prompt' > ) => string | undefined ;
118
+ private opts : Omit < PromptOptions < Prompt > , " render" | " input" | " output" > ;
119
+ private _render : ( context : Omit < Prompt , " prompt" > ) => string | undefined ;
114
120
private _track = false ;
115
- private _prevFrame = '' ;
116
- private _subscribers = new Map < string , { cb : ( ...args : any ) => any ; once ?: boolean } [ ] > ( ) ;
121
+ private _prevFrame = "" ;
122
+ private _subscribers = new Map <
123
+ string ,
124
+ { cb : ( ...args : any ) => any ; once ?: boolean } [ ]
125
+ > ( ) ;
117
126
protected _cursor = 0 ;
118
127
119
- public state : ClackState = ' initial' ;
120
- public error = '' ;
128
+ public state : ClackState = " initial" ;
129
+ public error = "" ;
121
130
public value : any ;
122
131
123
132
constructor ( options : PromptOptions < Prompt > , trackValue = true ) {
@@ -177,7 +186,10 @@ export default class Prompt {
177
186
* @param event - The event name
178
187
* @param data - The data to pass to the callback
179
188
*/
180
- public emit < T extends keyof ClackEvents > ( event : T , ...data : Parameters < ClackEvents [ T ] > ) {
189
+ public emit < T extends keyof ClackEvents > (
190
+ event : T ,
191
+ ...data : Parameters < ClackEvents [ T ] >
192
+ ) {
181
193
const cbs = this . _subscribers . get ( event ) ?? [ ] ;
182
194
const cleanup : ( ( ) => void ) [ ] = [ ] ;
183
195
@@ -199,9 +211,9 @@ export default class Prompt {
199
211
const sink = new WriteStream ( 0 ) ;
200
212
sink . _write = ( chunk , encoding , done ) => {
201
213
if ( this . _track ) {
202
- this . value = this . rl . line . replace ( / \t / g, '' ) ;
214
+ this . value = this . rl . line . replace ( / \t / g, "" ) ;
203
215
this . _cursor = this . rl . cursor ;
204
- this . emit ( ' value' , this . value ) ;
216
+ this . emit ( " value" , this . value ) ;
205
217
}
206
218
done ( ) ;
207
219
} ;
@@ -211,7 +223,7 @@ export default class Prompt {
211
223
input : this . input ,
212
224
output : sink ,
213
225
tabSize : 2 ,
214
- prompt : '' ,
226
+ prompt : "" ,
215
227
escapeCodeTimeout : 50 ,
216
228
} ) ;
217
229
readline . emitKeypressEvents ( this . input , this . rl ) ;
@@ -220,80 +232,80 @@ export default class Prompt {
220
232
this . rl . write ( this . opts . initialValue ) ;
221
233
}
222
234
223
- this . input . on ( ' keypress' , this . onKeypress ) ;
235
+ this . input . on ( " keypress" , this . onKeypress ) ;
224
236
setRawMode ( this . input , true ) ;
225
- this . output . on ( ' resize' , this . render ) ;
237
+ this . output . on ( " resize" , this . render ) ;
226
238
227
239
this . render ( ) ;
228
240
229
- this . once ( ' submit' , ( ) => {
241
+ this . once ( " submit" , ( ) => {
230
242
this . output . write ( cursor . show ) ;
231
- this . output . off ( ' resize' , this . render ) ;
243
+ this . output . off ( " resize" , this . render ) ;
232
244
setRawMode ( this . input , false ) ;
233
245
resolve ( this . value ) ;
234
246
} ) ;
235
- this . once ( ' cancel' , ( ) => {
247
+ this . once ( " cancel" , ( ) => {
236
248
this . output . write ( cursor . show ) ;
237
- this . output . off ( ' resize' , this . render ) ;
249
+ this . output . off ( " resize" , this . render ) ;
238
250
setRawMode ( this . input , false ) ;
239
251
resolve ( CANCEL_SYMBOL ) ;
240
252
} ) ;
241
253
} ) ;
242
254
}
243
255
244
256
private onKeypress ( char : string , key ?: Key ) {
245
- if ( this . state === ' error' ) {
246
- this . state = ' active' ;
257
+ if ( this . state === " error" ) {
258
+ this . state = " active" ;
247
259
}
248
260
if ( key ?. name && ! this . _track && ALIASES . has ( key . name ) ) {
249
- this . emit ( ' cursor' , ALIASES . get ( key . name ) ) ;
261
+ this . emit ( " cursor" , ALIASES . get ( key . name ) ) ;
250
262
}
251
263
if ( key ?. name && KEYS . has ( key . name as InferSetType < typeof KEYS > ) ) {
252
- this . emit ( ' cursor' , key . name as InferSetType < typeof KEYS > ) ;
264
+ this . emit ( " cursor" , key . name as InferSetType < typeof KEYS > ) ;
253
265
}
254
- if ( char && ( char . toLowerCase ( ) === 'y' || char . toLowerCase ( ) === 'n' ) ) {
255
- this . emit ( ' confirm' , char . toLowerCase ( ) === 'y' ) ;
266
+ if ( char && ( char . toLowerCase ( ) === "y" || char . toLowerCase ( ) === "n" ) ) {
267
+ this . emit ( " confirm" , char . toLowerCase ( ) === "y" ) ;
256
268
}
257
- if ( char === '\t' && this . opts . placeholder ) {
269
+ if ( char === "\t" && this . opts . placeholder ) {
258
270
if ( ! this . value ) {
259
271
this . rl . write ( this . opts . placeholder ) ;
260
- this . emit ( ' value' , this . opts . placeholder ) ;
272
+ this . emit ( " value" , this . opts . placeholder ) ;
261
273
}
262
274
}
263
275
if ( char ) {
264
- this . emit ( ' key' , char . toLowerCase ( ) ) ;
276
+ this . emit ( " key" , char . toLowerCase ( ) ) ;
265
277
}
266
278
267
- if ( key ?. name === ' return' ) {
279
+ if ( key ?. name === " return" ) {
268
280
if ( this . opts . validate ) {
269
281
const problem = this . opts . validate ( this . value ) ;
270
282
if ( problem ) {
271
283
this . error = problem ;
272
- this . state = ' error' ;
284
+ this . state = " error" ;
273
285
this . rl . write ( this . value ) ;
274
286
}
275
287
}
276
- if ( this . state !== ' error' ) {
277
- this . state = ' submit' ;
288
+ if ( this . state !== " error" ) {
289
+ this . state = " submit" ;
278
290
}
279
291
}
280
292
281
- if ( hasAliasKey ( [ char , key ?. name , key ?. sequence ] , ' cancel' ) ) {
282
- this . state = ' cancel' ;
293
+ if ( hasAliasKey ( [ char , key ?. name , key ?. sequence ] , " cancel" ) ) {
294
+ this . state = " cancel" ;
283
295
}
284
- if ( this . state === ' submit' || this . state === ' cancel' ) {
285
- this . emit ( ' finalize' ) ;
296
+ if ( this . state === " submit" || this . state === " cancel" ) {
297
+ this . emit ( " finalize" ) ;
286
298
}
287
299
this . render ( ) ;
288
- if ( this . state === ' submit' || this . state === ' cancel' ) {
300
+ if ( this . state === " submit" || this . state === " cancel" ) {
289
301
this . close ( ) ;
290
302
}
291
303
}
292
304
293
305
protected close ( ) {
294
306
this . input . unpipe ( ) ;
295
- this . input . removeListener ( ' keypress' , this . onKeypress ) ;
296
- this . output . write ( '\n' ) ;
307
+ this . input . removeListener ( " keypress" , this . onKeypress ) ;
308
+ this . output . write ( "\n" ) ;
297
309
setRawMode ( this . input , false ) ;
298
310
this . rl . close ( ) ;
299
311
this . emit ( `${ this . state } ` , this . value ) ;
@@ -302,32 +314,43 @@ export default class Prompt {
302
314
303
315
private restoreCursor ( ) {
304
316
const lines =
305
- wrap ( this . _prevFrame , process . stdout . columns , { hard : true } ) . split ( '\n' ) . length - 1 ;
317
+ wrap ( this . _prevFrame , process . stdout . columns , { hard : true } ) . split ( "\n" )
318
+ . length - 1 ;
306
319
this . output . write ( cursor . move ( - 999 , lines * - 1 ) ) ;
307
320
}
308
321
309
322
public format ( text : string , options ?: Partial < FormatOptions > ) : string {
310
- const getLineOption = < TLine extends LineOption , TKey extends keyof FormatLineOptions > (
323
+ const getLineOption = <
324
+ TLine extends LineOption ,
325
+ TKey extends keyof FormatLineOptions
326
+ > (
311
327
line : TLine ,
312
328
key : TKey
313
329
) : NonNullable < FormatOptions [ TLine ] [ TKey ] > => {
314
330
return (
315
- key === 'style'
316
- ? options ?. [ line ] ?. [ key ] ?? options ?. default ?. [ key ] ?? ( ( line ) => line )
317
- : options ?. [ line ] ?. [ key ] ?? options ?. [ line ] ?. sides ?? options ?. default ?. [ key ] ?? ''
331
+ key === "style"
332
+ ? options ?. [ line ] ?. [ key ] ??
333
+ options ?. default ?. [ key ] ??
334
+ ( ( line ) => line )
335
+ : options ?. [ line ] ?. [ key ] ??
336
+ options ?. [ line ] ?. sides ??
337
+ options ?. default ?. [ key ] ??
338
+ ""
318
339
) as NonNullable < FormatOptions [ TLine ] [ TKey ] > ;
319
340
} ;
320
- const getLineOptions = ( line : LineOption ) : Omit < FormatLineOptions , 'sides' > => {
341
+ const getLineOptions = (
342
+ line : LineOption
343
+ ) : Omit < FormatLineOptions , "sides" > => {
321
344
return {
322
- start : getLineOption ( line , ' start' ) ,
323
- end : getLineOption ( line , ' end' ) ,
324
- style : getLineOption ( line , ' style' ) ,
345
+ start : getLineOption ( line , " start" ) ,
346
+ end : getLineOption ( line , " end" ) ,
347
+ style : getLineOption ( line , " style" ) ,
325
348
} ;
326
349
} ;
327
350
328
- const firstLine = getLineOptions ( ' firstLine' ) ;
329
- const newLine = getLineOptions ( ' newLine' ) ;
330
- const lastLine = getLineOptions ( ' lastLine' ) ;
351
+ const firstLine = getLineOptions ( " firstLine" ) ;
352
+ const newLine = getLineOptions ( " newLine" ) ;
353
+ const lastLine = getLineOptions ( " lastLine" ) ;
331
354
332
355
const emptySlots =
333
356
Math . max (
@@ -344,7 +367,7 @@ export default class Prompt {
344
367
345
368
for ( const paragraph of paragraphs ) {
346
369
const words = paragraph . split ( / \s / g) ;
347
- let currentLine = '' ;
370
+ let currentLine = "" ;
348
371
349
372
for ( const word of words ) {
350
373
if ( strLength ( currentLine + word ) + emptySlots + 1 <= maxWidth ) {
@@ -371,7 +394,9 @@ export default class Prompt {
371
394
372
395
return formattedLines
373
396
. map ( ( line , i , ar ) => {
374
- const opt = < TPosition extends Exclude < keyof FormatLineOptions , 'sides' > > (
397
+ const opt = <
398
+ TPosition extends Exclude < keyof FormatLineOptions , "sides" >
399
+ > (
375
400
position : TPosition
376
401
) : FormatLineOptions [ TPosition ] => {
377
402
return (
@@ -386,27 +411,31 @@ export default class Prompt {
386
411
: newLine [ position ]
387
412
) as FormatLineOptions [ TPosition ] ;
388
413
} ;
389
- const startLine = opt ( ' start' ) ;
390
- const endLine = opt ( ' end' ) ;
391
- const styleLine = opt ( ' style' ) ;
414
+ const startLine = opt ( " start" ) ;
415
+ const endLine = opt ( " end" ) ;
416
+ const styleLine = opt ( " style" ) ;
392
417
// only format the line without the leading space.
393
418
const leadingSpaceRegex = / ^ \s / ;
394
419
const styledLine = leadingSpaceRegex . test ( line )
395
420
? ` ${ styleLine ( line . slice ( 1 ) ) } `
396
421
: styleLine ( line ) ;
397
422
const fullLine =
398
- styledLine + ' ' . repeat ( Math . max ( minWidth - strLength ( styledLine ) - emptySlots , 0 ) ) ;
399
- return [ startLine , fullLine , endLine ] . join ( ' ' ) ;
423
+ styledLine +
424
+ " " . repeat (
425
+ Math . max ( minWidth - strLength ( styledLine ) - emptySlots , 0 )
426
+ ) ;
427
+ return [ startLine , fullLine , endLine ] . join ( " " ) ;
400
428
} )
401
- . join ( '\n' ) ;
429
+ . join ( "\n" ) ;
402
430
}
403
431
404
- private _prevFrame = '' ;
405
432
private render ( ) {
406
- const frame = wrap ( this . _render ( this ) ?? '' , process . stdout . columns , { hard : true } ) ;
433
+ const frame = wrap ( this . _render ( this ) ?? "" , process . stdout . columns , {
434
+ hard : true ,
435
+ } ) ;
407
436
if ( frame === this . _prevFrame ) return ;
408
437
409
- if ( this . state === ' initial' ) {
438
+ if ( this . state === " initial" ) {
410
439
this . output . write ( cursor . hide ) ;
411
440
} else {
412
441
const diff = diffLines ( this . _prevFrame , frame ) ;
@@ -416,7 +445,7 @@ export default class Prompt {
416
445
const diffLine = diff [ 0 ] ;
417
446
this . output . write ( cursor . move ( 0 , diffLine ) ) ;
418
447
this . output . write ( erase . lines ( 1 ) ) ;
419
- const lines = frame . split ( '\n' ) ;
448
+ const lines = frame . split ( "\n" ) ;
420
449
this . output . write ( lines [ diffLine ] ) ;
421
450
this . _prevFrame = frame ;
422
451
this . output . write ( cursor . move ( 0 , lines . length - diffLine - 1 ) ) ;
@@ -427,9 +456,9 @@ export default class Prompt {
427
456
const diffLine = diff [ 0 ] ;
428
457
this . output . write ( cursor . move ( 0 , diffLine ) ) ;
429
458
this . output . write ( erase . down ( ) ) ;
430
- const lines = frame . split ( '\n' ) ;
459
+ const lines = frame . split ( "\n" ) ;
431
460
const newLines = lines . slice ( diffLine ) ;
432
- this . output . write ( newLines . join ( '\n' ) ) ;
461
+ this . output . write ( newLines . join ( "\n" ) ) ;
433
462
this . _prevFrame = frame ;
434
463
return ;
435
464
}
@@ -438,8 +467,8 @@ export default class Prompt {
438
467
}
439
468
440
469
this . output . write ( frame ) ;
441
- if ( this . state === ' initial' ) {
442
- this . state = ' active' ;
470
+ if ( this . state === " initial" ) {
471
+ this . state = " active" ;
443
472
}
444
473
this . _prevFrame = frame ;
445
474
}
0 commit comments