Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(@clack/core,@clack/prompts): path prompt #148

Draft
wants to merge 28 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
16a3c34
feat(@clack/core): path prompt
orochaa Aug 24, 2023
f6b5042
feat(@clack/prompts): path prompt
orochaa Aug 24, 2023
a93b884
feat: free navigation
orochaa Aug 24, 2023
6a64200
chore: add changeset
orochaa Aug 24, 2023
c55a847
feat: max items
orochaa Aug 24, 2023
a1a6a94
feat: folder arrow
orochaa Aug 24, 2023
fb8262a
chore: lint
orochaa Aug 24, 2023
9395dbb
feat: only show dir option
orochaa Aug 24, 2023
c99fa86
chore: restore basic example
orochaa Aug 24, 2023
6ffd8c3
chore: add file selection example
orochaa Aug 24, 2023
ef2d7e1
fix: leading space
orochaa Aug 24, 2023
647cefd
refactor: remove unused event listener
orochaa Aug 25, 2023
2aaf6bb
refactor: improve cursor mapping
orochaa Aug 25, 2023
1a11247
refactor: cancel state
orochaa Aug 25, 2023
a28d2b1
Merge branch 'main' of https://github.com/natemoo-re/clack into path
orochaa Aug 26, 2023
9bc1e8b
Merge branch 'path' of https://github.com/Mist3rBru/clack into path
orochaa Aug 26, 2023
1bdf0bc
docs: add path example
orochaa Aug 27, 2023
154f9c2
feat: path input
orochaa Aug 29, 2023
9ed6904
Merge branch 'main' of https://github.com/natemoo-re/clack into path
orochaa Aug 29, 2023
0beb787
Merge branch 'path' of https://github.com/Mist3rBru/clack into path
orochaa Aug 29, 2023
c9e3043
refactor: prevent error on file root
orochaa Aug 29, 2023
b2c639c
chore: update docs
orochaa Aug 29, 2023
b118f43
feat: path validate
orochaa Aug 29, 2023
5487280
feat: shell like input
orochaa Nov 3, 2023
ec70704
feat: highlight current hint option
orochaa Nov 9, 2023
3f1db88
feat: maxHintOptions param
orochaa Nov 9, 2023
dec422f
feat: reverse hint option selection
orochaa Nov 9, 2023
8c854c1
feat: indentify directories on hintOptions
orochaa Nov 10, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .changeset/fresh-grapes-rule.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@clack/core': minor
---

Feat PathPrompt
5 changes: 5 additions & 0 deletions .changeset/modern-eggs-laugh.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@clack/prompts': minor
---

Feat path prompt
28 changes: 28 additions & 0 deletions examples/basic/file-selection.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
import * as p from '@clack/prompts';

(async () => {
const selectResult = await p.path({
type: 'select',
message: 'Pick a file with select component:',
initialValue: process.cwd(),
onlyShowDir: false,
maxItems: 15,
});
if (p.isCancel(selectResult)) {
p.cancel('File selection canceled');
process.exit(0);
}

const inputResult = await p.path({
type: 'text',
message: 'Pick other file with input component:',
onlyShowDir: false,
placeholder: process.cwd(),
});
if (p.isCancel(inputResult)) {
p.cancel('File input canceled');
process.exit(0);
}

console.log({ selectResult, inputResult });
})();
3 changes: 2 additions & 1 deletion examples/basic/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,8 @@
},
"scripts": {
"start": "jiti ./index.ts",
"spinner": "jiti ./spinner.ts"
"spinner": "jiti ./spinner.ts",
"file": "jiti ./file-selection.ts"
},
"devDependencies": {
"jiti": "^1.17.0"
Expand Down
1 change: 1 addition & 0 deletions packages/core/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ export { default as ConfirmPrompt } from './prompts/confirm';
export { default as GroupMultiSelectPrompt } from './prompts/group-multiselect';
export { default as MultiSelectPrompt } from './prompts/multi-select';
export { default as PasswordPrompt } from './prompts/password';
export { default as PathPrompt } from './prompts/path';
export { default as Prompt, isCancel } from './prompts/prompt';
export type { State } from './prompts/prompt';
export { default as SelectPrompt } from './prompts/select';
Expand Down
285 changes: 285 additions & 0 deletions packages/core/src/prompts/path.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,285 @@
import { readdirSync, statSync } from 'node:fs';
import { resolve } from 'node:path';
import { Key } from 'node:readline';
import color from 'picocolors';
import Prompt, { PromptOptions } from './prompt';

interface PathNode {
name: string;
children: PathNode[] | undefined;
}

type PathType = 'text' | 'select';

type PathOptions = PromptOptions<PathPrompt> &
(
| {
type: 'text';
onlyShowDir?: boolean;
placeholder?: string;
}
| {
type: 'select';
onlyShowDir?: boolean;
maxHintOptions?: number;
}
);

export default class PathPrompt extends Prompt {
public readonly type: PathType;
public readonly placeholder: string;
public readonly onlyShowDir: boolean;
public root: PathNode;
public hint: string;
public hintOptions: string[];
public valueWithHint: string;
public hintIndex: number;

private _cursorMap: number[];
private _maxHintOptions: number;

public get option() {
let aux: PathNode = this.root;
for (const index of this._cursorMap) {
if (aux.children && aux.children[index]) {
aux = aux.children[index];
}
}
return {
index: this._cursorMap[this._cursorMap.length - 1] ?? 0,
depth: this._cursorMap.length,
node: aux,
};
}

public get options(): PathNode[] {
let aux: PathNode = this.root;
let options: PathNode[] = [this.root];
for (const index of this._cursorMap) {
options = options.concat(aux.children ?? []);
if (aux.children && aux.children[index]) {
aux = aux.children[index];
}
}
return options;
}

public get cursor(): number {
return this.type === 'select' ? this._cursorMap.reduce((a, b) => a + b + 1, 0) : this._cursor;
}

private get _layer(): PathNode[] {
let aux: PathNode = this.root;
let options: PathNode[] = [];
for (const index of this._cursorMap) {
if (aux.children?.[index]) {
options = aux.children;
aux = aux.children[index];
} else {
break;
}
}
return options;
}

private get _selectedValue(): string {
const value: string[] = [];
let option: PathNode = this.root;
for (const index of this._cursorMap) {
if (option.children?.[index]) {
option = option.children[index];
value.push(option.name);
}
}
return resolve(this.root.name, ...value);
}

private _changeSelectValue(): void {
this.value = this._selectedValue;
}

private get _valueDir(): string {
return this.value.replace(/^(.*)\/.*/, '$1').replace(/\s+$/, '');
}

private get _valueEnd(): string {
return this.value.replace(/.*\/(.*)$/, '$1');
}

private get _hintOptions(): string[] {
return statSync(this._valueDir, { throwIfNoEntry: false })?.isDirectory()
? this._mapDir(this._valueDir)
.filter((node) => node.name.startsWith(this._valueEnd))
.slice(0, this._maxHintOptions)
.map((node) => node.name + (!!node.children ? '/' : ''))
: [];
}

private _changeInputHint(): void {
const hintOption = this._hintOptions[0] ?? '';
this.hintOptions = [];
this.hint = hintOption.replace(this._valueEnd, '');
}

private _changeInputValue(): void {
let value: string;
let hint: string;
const cursor = color.inverse(color.hidden('_'));
if (this.cursor >= this.value.length) {
value = this.value;
hint = this.hint
? `${color.inverse(this.hint.charAt(0))}${color.dim(this.hint.slice(1))}`
: cursor;
} else {
const s1 = this.value.slice(0, this.cursor);
const s2 = this.value.slice(this.cursor);
value = `${s1}${color.inverse(s2[0])}${s2.slice(1)}`;
hint = color.dim(this.hint);
}
this.valueWithHint = value + hint;
}

private _changeValue(): void {
if (this.type === 'select') {
this._changeSelectValue();
} else {
this._changeInputHint();
this._changeInputValue();
}
}

private _autocomplete(): void {
const complete = this.value ? this.hint : this.placeholder;
this.value += complete;
this._cursor = this.value.length;
this.hint = '';
this.hintOptions = [];
this.rl.write(complete);
this._changeInputValue();
}

private _suggestAutocomplete(step: number): void {
const hintOptions = this._hintOptions;
if (hintOptions.length <= 1) {
this._autocomplete();
} else if (this.hintOptions.length) {
this.hintIndex += step;
if (this.hintIndex >= this.hintOptions.length) {
this.hintIndex = 0;
} else if (this.hintIndex < 0) {
this.hintIndex = this.hintOptions.length - 1;
}
this.hint = this.hintOptions[this.hintIndex].replace(this._valueEnd, '');
this._changeInputValue();
} else {
this.hintIndex = -1;
this.hintOptions = hintOptions;
}
}

private _mapDir(path: string): PathNode[] {
return readdirSync(path, { withFileTypes: true })
.map((item) => ({
name: item.name,
children: item.isDirectory() ? [] : undefined,
}))
.filter((node) => {
return this.onlyShowDir ? !!node.children : true;
});
}

constructor(opts: PathOptions) {
super(opts, opts.type === 'text');

// General
this.type = opts.type;
this.onlyShowDir = opts.onlyShowDir ?? false;
this.value = opts.initialValue ?? '';

// Select
this._cursorMap = [0];
const cwd = opts.initialValue ?? process.cwd();
this.root = {
name: cwd,
children: this._mapDir(cwd),
};

// Text
this.placeholder = opts.placeholder ?? '';
this.hint = '';
this.hintOptions = [];
this.hintIndex = -1;
this._maxHintOptions =
'maxHintOptions' in opts && opts.maxHintOptions ? opts.maxHintOptions : Infinity;
this.valueWithHint = '';

this._changeValue();

this.on('cursor', (key) => {
if (this.type !== 'select') return;

switch (key) {
case 'up':
if (this._cursorMap.length) {
this._cursorMap = [
...this._cursorMap.slice(0, -1),
this.option.index > 0 ? this.option.index - 1 : this._layer.length - 1,
];
}
break;
case 'down':
if (this._cursorMap.length) {
this._cursorMap = [
...this._cursorMap.slice(0, -1),
this.option.index < this._layer.length - 1 ? this.option.index + 1 : 0,
];
}
break;
case 'right':
if (this.option.node.children) {
const children = this._mapDir(this._selectedValue);
this.option.node.children = children;
this._cursorMap = children.length ? [...this._cursorMap, 0] : this._cursorMap;
}
break;
case 'left':
const prevCursor = this._cursorMap;
this._cursorMap = this._cursorMap.slice(0, -1);
if (this.option.node.children?.length && this._cursorMap.length) {
this.option.node.children = [];
} else if (prevCursor.length === 0) {
const cwd = resolve(this.root.name, '..');
this.root = {
name: cwd,
children: this._mapDir(cwd),
};
}
break;
}
return this._changeSelectValue();
});

this.on('cursor', (key) => {
if (this.type !== 'text') return;

if (key === 'right' && this.cursor >= this.value.length) {
this._autocomplete();
}
});

this.on('key', (char: string, key: Key) => {
if (this.type !== 'text') return;

if (key.name === 'tab') {
this._suggestAutocomplete(key.shift ? -1 : 1);
} else {
this._changeInputHint();
this._changeInputValue();
}
});

this.on('finalize', () => {
this.value = this.value ? resolve(this.value) : '';
});
}
}
6 changes: 3 additions & 3 deletions packages/core/src/prompts/prompt.ts
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,7 @@ export type State = 'initial' | 'active' | 'cancel' | 'submit' | 'error';
export default class Prompt {
protected input: Readable;
protected output: Writable;
private rl!: ReadLine;
protected rl!: ReadLine;
private opts: Omit<PromptOptions<Prompt>, 'render' | 'input' | 'output'>;
private _track: boolean = false;
private _render: (context: Omit<Prompt, 'prompt'>) => string | void;
Expand Down Expand Up @@ -172,8 +172,8 @@ export default class Prompt {
this.emit('value', this.opts.placeholder);
}
}
if (char) {
this.emit('key', char.toLowerCase());
if (char || key) {
this.emit('key', char?.toLowerCase(), key);
}

if (key?.name === 'return') {
Expand Down
6 changes: 3 additions & 3 deletions packages/core/src/prompts/select-key.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,9 +14,9 @@ export default class SelectKeyPrompt<T extends { value: any }> extends Prompt {
const keys = this.options.map(({ value: [initial] }) => initial?.toLowerCase());
this.cursor = Math.max(keys.indexOf(opts.initialValue), 0);

this.on('key', (key) => {
if (!keys.includes(key)) return;
const value = this.options.find(({ value: [initial] }) => initial?.toLowerCase() === key);
this.on('key', (char) => {
if (!keys.includes(char)) return;
const value = this.options.find(({ value: [initial] }) => initial?.toLowerCase() === char);
if (value) {
this.value = value.value;
this.state = 'submit';
Expand Down
Loading