Skip to content

Commit f9cae47

Browse files
committed
chore: rebase branch
1 parent 69b7030 commit f9cae47

File tree

6 files changed

+346
-383
lines changed

6 files changed

+346
-383
lines changed

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -25,4 +25,4 @@
2525
"volta": {
2626
"node": "20.18.1"
2727
}
28-
}
28+
}

packages/core/src/index.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,9 @@ export { default as ConfirmPrompt } from "./prompts/confirm";
22
export { default as GroupMultiSelectPrompt } from "./prompts/group-multiselect";
33
export { default as MultiSelectPrompt } from "./prompts/multi-select";
44
export { default as PasswordPrompt } from "./prompts/password";
5-
export { default as Prompt, State } from "./prompts/prompt";
5+
export { default as Prompt } from "./prompts/prompt";
66
export { default as SelectPrompt } from "./prompts/select";
77
export { default as SelectKeyPrompt } from "./prompts/select-key";
88
export { default as TextPrompt } from "./prompts/text";
9+
export type { ClackState as State } from "./types";
910
export { block, isCancel, strLength, setGlobalAliases } from "./utils";

packages/core/src/prompts/prompt.ts

+109-80
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,24 @@
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";
1219

1320
export interface PromptOptions<Self extends Prompt> {
14-
render(this: Omit<Self, 'prompt'>): string | undefined;
21+
render(this: Omit<Self, "prompt">): string | undefined;
1522
placeholder?: string;
1623
initialValue?: any;
1724
validate?: ((value: any) => string | undefined) | undefined;
@@ -20,9 +27,7 @@ export interface PromptOptions<Self extends Prompt> {
2027
debug?: boolean;
2128
}
2229

23-
export type State = 'initial' | 'active' | 'cancel' | 'submit' | 'error';
24-
25-
export type LineOption = 'firstLine' | 'newLine' | 'lastLine';
30+
export type LineOption = "firstLine" | "newLine" | "lastLine";
2631

2732
export interface FormatLineOptions {
2833
/**
@@ -71,7 +76,8 @@ export interface FormatLineOptions {
7176
style: (line: string) => string;
7277
}
7378

74-
export interface FormatOptions extends Record<LineOption, Partial<FormatLineOptions>> {
79+
export interface FormatOptions
80+
extends Record<LineOption, Partial<FormatLineOptions>> {
7581
/**
7682
* Shorthand to define values for each line
7783
* @example
@@ -109,15 +115,18 @@ export default class Prompt {
109115
protected output: Writable;
110116

111117
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;
114120
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+
>();
117126
protected _cursor = 0;
118127

119-
public state: ClackState = 'initial';
120-
public error = '';
128+
public state: ClackState = "initial";
129+
public error = "";
121130
public value: any;
122131

123132
constructor(options: PromptOptions<Prompt>, trackValue = true) {
@@ -177,7 +186,10 @@ export default class Prompt {
177186
* @param event - The event name
178187
* @param data - The data to pass to the callback
179188
*/
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+
) {
181193
const cbs = this._subscribers.get(event) ?? [];
182194
const cleanup: (() => void)[] = [];
183195

@@ -199,9 +211,9 @@ export default class Prompt {
199211
const sink = new WriteStream(0);
200212
sink._write = (chunk, encoding, done) => {
201213
if (this._track) {
202-
this.value = this.rl.line.replace(/\t/g, '');
214+
this.value = this.rl.line.replace(/\t/g, "");
203215
this._cursor = this.rl.cursor;
204-
this.emit('value', this.value);
216+
this.emit("value", this.value);
205217
}
206218
done();
207219
};
@@ -211,7 +223,7 @@ export default class Prompt {
211223
input: this.input,
212224
output: sink,
213225
tabSize: 2,
214-
prompt: '',
226+
prompt: "",
215227
escapeCodeTimeout: 50,
216228
});
217229
readline.emitKeypressEvents(this.input, this.rl);
@@ -220,80 +232,80 @@ export default class Prompt {
220232
this.rl.write(this.opts.initialValue);
221233
}
222234

223-
this.input.on('keypress', this.onKeypress);
235+
this.input.on("keypress", this.onKeypress);
224236
setRawMode(this.input, true);
225-
this.output.on('resize', this.render);
237+
this.output.on("resize", this.render);
226238

227239
this.render();
228240

229-
this.once('submit', () => {
241+
this.once("submit", () => {
230242
this.output.write(cursor.show);
231-
this.output.off('resize', this.render);
243+
this.output.off("resize", this.render);
232244
setRawMode(this.input, false);
233245
resolve(this.value);
234246
});
235-
this.once('cancel', () => {
247+
this.once("cancel", () => {
236248
this.output.write(cursor.show);
237-
this.output.off('resize', this.render);
249+
this.output.off("resize", this.render);
238250
setRawMode(this.input, false);
239251
resolve(CANCEL_SYMBOL);
240252
});
241253
});
242254
}
243255

244256
private onKeypress(char: string, key?: Key) {
245-
if (this.state === 'error') {
246-
this.state = 'active';
257+
if (this.state === "error") {
258+
this.state = "active";
247259
}
248260
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));
250262
}
251263
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>);
253265
}
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");
256268
}
257-
if (char === '\t' && this.opts.placeholder) {
269+
if (char === "\t" && this.opts.placeholder) {
258270
if (!this.value) {
259271
this.rl.write(this.opts.placeholder);
260-
this.emit('value', this.opts.placeholder);
272+
this.emit("value", this.opts.placeholder);
261273
}
262274
}
263275
if (char) {
264-
this.emit('key', char.toLowerCase());
276+
this.emit("key", char.toLowerCase());
265277
}
266278

267-
if (key?.name === 'return') {
279+
if (key?.name === "return") {
268280
if (this.opts.validate) {
269281
const problem = this.opts.validate(this.value);
270282
if (problem) {
271283
this.error = problem;
272-
this.state = 'error';
284+
this.state = "error";
273285
this.rl.write(this.value);
274286
}
275287
}
276-
if (this.state !== 'error') {
277-
this.state = 'submit';
288+
if (this.state !== "error") {
289+
this.state = "submit";
278290
}
279291
}
280292

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";
283295
}
284-
if (this.state === 'submit' || this.state === 'cancel') {
285-
this.emit('finalize');
296+
if (this.state === "submit" || this.state === "cancel") {
297+
this.emit("finalize");
286298
}
287299
this.render();
288-
if (this.state === 'submit' || this.state === 'cancel') {
300+
if (this.state === "submit" || this.state === "cancel") {
289301
this.close();
290302
}
291303
}
292304

293305
protected close() {
294306
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");
297309
setRawMode(this.input, false);
298310
this.rl.close();
299311
this.emit(`${this.state}`, this.value);
@@ -302,32 +314,43 @@ export default class Prompt {
302314

303315
private restoreCursor() {
304316
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;
306319
this.output.write(cursor.move(-999, lines * -1));
307320
}
308321

309322
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+
>(
311327
line: TLine,
312328
key: TKey
313329
): NonNullable<FormatOptions[TLine][TKey]> => {
314330
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+
""
318339
) as NonNullable<FormatOptions[TLine][TKey]>;
319340
};
320-
const getLineOptions = (line: LineOption): Omit<FormatLineOptions, 'sides'> => {
341+
const getLineOptions = (
342+
line: LineOption
343+
): Omit<FormatLineOptions, "sides"> => {
321344
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"),
325348
};
326349
};
327350

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");
331354

332355
const emptySlots =
333356
Math.max(
@@ -344,7 +367,7 @@ export default class Prompt {
344367

345368
for (const paragraph of paragraphs) {
346369
const words = paragraph.split(/\s/g);
347-
let currentLine = '';
370+
let currentLine = "";
348371

349372
for (const word of words) {
350373
if (strLength(currentLine + word) + emptySlots + 1 <= maxWidth) {
@@ -371,7 +394,9 @@ export default class Prompt {
371394

372395
return formattedLines
373396
.map((line, i, ar) => {
374-
const opt = <TPosition extends Exclude<keyof FormatLineOptions, 'sides'>>(
397+
const opt = <
398+
TPosition extends Exclude<keyof FormatLineOptions, "sides">
399+
>(
375400
position: TPosition
376401
): FormatLineOptions[TPosition] => {
377402
return (
@@ -386,27 +411,31 @@ export default class Prompt {
386411
: newLine[position]
387412
) as FormatLineOptions[TPosition];
388413
};
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");
392417
// only format the line without the leading space.
393418
const leadingSpaceRegex = /^\s/;
394419
const styledLine = leadingSpaceRegex.test(line)
395420
? ` ${styleLine(line.slice(1))}`
396421
: styleLine(line);
397422
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(" ");
400428
})
401-
.join('\n');
429+
.join("\n");
402430
}
403431

404-
private _prevFrame = '';
405432
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+
});
407436
if (frame === this._prevFrame) return;
408437

409-
if (this.state === 'initial') {
438+
if (this.state === "initial") {
410439
this.output.write(cursor.hide);
411440
} else {
412441
const diff = diffLines(this._prevFrame, frame);
@@ -416,7 +445,7 @@ export default class Prompt {
416445
const diffLine = diff[0];
417446
this.output.write(cursor.move(0, diffLine));
418447
this.output.write(erase.lines(1));
419-
const lines = frame.split('\n');
448+
const lines = frame.split("\n");
420449
this.output.write(lines[diffLine]);
421450
this._prevFrame = frame;
422451
this.output.write(cursor.move(0, lines.length - diffLine - 1));
@@ -427,9 +456,9 @@ export default class Prompt {
427456
const diffLine = diff[0];
428457
this.output.write(cursor.move(0, diffLine));
429458
this.output.write(erase.down());
430-
const lines = frame.split('\n');
459+
const lines = frame.split("\n");
431460
const newLines = lines.slice(diffLine);
432-
this.output.write(newLines.join('\n'));
461+
this.output.write(newLines.join("\n"));
433462
this._prevFrame = frame;
434463
return;
435464
}
@@ -438,8 +467,8 @@ export default class Prompt {
438467
}
439468

440469
this.output.write(frame);
441-
if (this.state === 'initial') {
442-
this.state = 'active';
470+
if (this.state === "initial") {
471+
this.state = "active";
443472
}
444473
this._prevFrame = frame;
445474
}

0 commit comments

Comments
 (0)