| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485 | 'use strict';const Events = require('events');const colors = require('ansi-colors');const keypress = require('./keypress');const timer = require('./timer');const State = require('./state');const theme = require('./theme');const utils = require('./utils');const ansi = require('./ansi');/** * Base class for creating a new Prompt. * @param {Object} `options` Question object. */class Prompt extends Events {  constructor(options = {}) {    super();    this.name = options.name;    this.type = options.type;    this.options = options;    theme(this);    timer(this);    this.state = new State(this);    this.initial = [options.initial, options.default].find(v => v != null);    this.stdout = options.stdout || process.stdout;    this.stdin = options.stdin || process.stdin;    this.scale = options.scale || 1;    this.term = this.options.term || process.env.TERM_PROGRAM;    this.margin = margin(this.options.margin);    this.setMaxListeners(0);    setOptions(this);  }  async keypress(input, event = {}) {    this.keypressed = true;    let key = keypress.action(input, keypress(input, event), this.options.actions);    this.state.keypress = key;    this.emit('keypress', input, key);    this.emit('state', this.state.clone());    let fn = this.options[key.action] || this[key.action] || this.dispatch;    if (typeof fn === 'function') {      return await fn.call(this, input, key);    }    this.alert();  }  alert() {    delete this.state.alert;    if (this.options.show === false) {      this.emit('alert');    } else {      this.stdout.write(ansi.code.beep);    }  }  cursorHide() {    this.stdout.write(ansi.cursor.hide());    utils.onExit(() => this.cursorShow());  }  cursorShow() {    this.stdout.write(ansi.cursor.show());  }  write(str) {    if (!str) return;    if (this.stdout && this.state.show !== false) {      this.stdout.write(str);    }    this.state.buffer += str;  }  clear(lines = 0) {    let buffer = this.state.buffer;    this.state.buffer = '';    if ((!buffer && !lines) || this.options.show === false) return;    this.stdout.write(ansi.cursor.down(lines) + ansi.clear(buffer, this.width));  }  restore() {    if (this.state.closed || this.options.show === false) return;    let { prompt, after, rest } = this.sections();    let { cursor, initial = '', input = '', value = '' } = this;    let size = this.state.size = rest.length;    let state = { after, cursor, initial, input, prompt, size, value };    let codes = ansi.cursor.restore(state);    if (codes) {      this.stdout.write(codes);    }  }  sections() {    let { buffer, input, prompt } = this.state;    prompt = colors.unstyle(prompt);    let buf = colors.unstyle(buffer);    let idx = buf.indexOf(prompt);    let header = buf.slice(0, idx);    let rest = buf.slice(idx);    let lines = rest.split('\n');    let first = lines[0];    let last = lines[lines.length - 1];    let promptLine = prompt + (input ? ' ' + input : '');    let len = promptLine.length;    let after = len < first.length ? first.slice(len + 1) : '';    return { header, prompt: first, after, rest: lines.slice(1), last };  }  async submit() {    this.state.submitted = true;    this.state.validating = true;    // this will only be called when the prompt is directly submitted    // without initializing, i.e. when the prompt is skipped, etc. Otherwize,    // "options.onSubmit" is will be handled by the "initialize()" method.    if (this.options.onSubmit) {      await this.options.onSubmit.call(this, this.name, this.value, this);    }    let result = this.state.error || await this.validate(this.value, this.state);    if (result !== true) {      let error = '\n' + this.symbols.pointer + ' ';      if (typeof result === 'string') {        error += result.trim();      } else {        error += 'Invalid input';      }      this.state.error = '\n' + this.styles.danger(error);      this.state.submitted = false;      await this.render();      await this.alert();      this.state.validating = false;      this.state.error = void 0;      return;    }    this.state.validating = false;    await this.render();    await this.close();    this.value = await this.result(this.value);    this.emit('submit', this.value);  }  async cancel(err) {    this.state.cancelled = this.state.submitted = true;    await this.render();    await this.close();    if (typeof this.options.onCancel === 'function') {      await this.options.onCancel.call(this, this.name, this.value, this);    }    this.emit('cancel', await this.error(err));  }  async close() {    this.state.closed = true;    try {      let sections = this.sections();      let lines = Math.ceil(sections.prompt.length / this.width);      if (sections.rest) {        this.write(ansi.cursor.down(sections.rest.length));      }      this.write('\n'.repeat(lines));    } catch (err) { /* do nothing */ }    this.emit('close');  }  start() {    if (!this.stop && this.options.show !== false) {      this.stop = keypress.listen(this, this.keypress.bind(this));      this.once('close', this.stop);    }  }  async skip() {    this.skipped = this.options.skip === true;    if (typeof this.options.skip === 'function') {      this.skipped = await this.options.skip.call(this, this.name, this.value);    }    return this.skipped;  }  async initialize() {    let { format, options, result } = this;    this.format = () => format.call(this, this.value);    this.result = () => result.call(this, this.value);    if (typeof options.initial === 'function') {      this.initial = await options.initial.call(this, this);    }    if (typeof options.onRun === 'function') {      await options.onRun.call(this, this);    }    // if "options.onSubmit" is defined, we wrap the "submit" method to guarantee    // that "onSubmit" will always called first thing inside the submit    // method, regardless of how it's handled in inheriting prompts.    if (typeof options.onSubmit === 'function') {      let onSubmit = options.onSubmit.bind(this);      let submit = this.submit.bind(this);      delete this.options.onSubmit;      this.submit = async() => {        await onSubmit(this.name, this.value, this);        return submit();      };    }    await this.start();    await this.render();  }  render() {    throw new Error('expected prompt to have a custom render method');  }  run() {    return new Promise(async(resolve, reject) => {      this.once('submit', resolve);      this.once('cancel', reject);      if (await this.skip()) {        this.render = () => {};        return this.submit();      }      await this.initialize();      this.emit('run');    });  }  async element(name, choice, i) {    let { options, state, symbols, timers } = this;    let timer = timers && timers[name];    state.timer = timer;    let value = options[name] || state[name] || symbols[name];    let val = choice && choice[name] != null ? choice[name] : await value;    if (val === '') return val;    let res = await this.resolve(val, state, choice, i);    if (!res && choice && choice[name]) {      return this.resolve(value, state, choice, i);    }    return res;  }  async prefix() {    let element = await this.element('prefix') || this.symbols;    let timer = this.timers && this.timers.prefix;    let state = this.state;    state.timer = timer;    if (utils.isObject(element)) element = element[state.status] || element.pending;    if (!utils.hasColor(element)) {      let style = this.styles[state.status] || this.styles.pending;      return style(element);    }    return element;  }  async message() {    let message = await this.element('message');    if (!utils.hasColor(message)) {      return this.styles.strong(message);    }    return message;  }  async separator() {    let element = await this.element('separator') || this.symbols;    let timer = this.timers && this.timers.separator;    let state = this.state;    state.timer = timer;    let value = element[state.status] || element.pending || state.separator;    let ele = await this.resolve(value, state);    if (utils.isObject(ele)) ele = ele[state.status] || ele.pending;    if (!utils.hasColor(ele)) {      return this.styles.muted(ele);    }    return ele;  }  async pointer(choice, i) {    let val = await this.element('pointer', choice, i);    if (typeof val === 'string' && utils.hasColor(val)) {      return val;    }    if (val) {      let styles = this.styles;      let focused = this.index === i;      let style = focused ? styles.primary : val => val;      let ele = await this.resolve(val[focused ? 'on' : 'off'] || val, this.state);      let styled = !utils.hasColor(ele) ? style(ele) : ele;      return focused ? styled : ' '.repeat(ele.length);    }  }  async indicator(choice, i) {    let val = await this.element('indicator', choice, i);    if (typeof val === 'string' && utils.hasColor(val)) {      return val;    }    if (val) {      let styles = this.styles;      let enabled = choice.enabled === true;      let style = enabled ? styles.success : styles.dark;      let ele = val[enabled ? 'on' : 'off'] || val;      return !utils.hasColor(ele) ? style(ele) : ele;    }    return '';  }  body() {    return null;  }  footer() {    if (this.state.status === 'pending') {      return this.element('footer');    }  }  header() {    if (this.state.status === 'pending') {      return this.element('header');    }  }  async hint() {    if (this.state.status === 'pending' && !this.isValue(this.state.input)) {      let hint = await this.element('hint');      if (!utils.hasColor(hint)) {        return this.styles.muted(hint);      }      return hint;    }  }  error(err) {    return !this.state.submitted ? (err || this.state.error) : '';  }  format(value) {    return value;  }  result(value) {    return value;  }  validate(value) {    if (this.options.required === true) {      return this.isValue(value);    }    return true;  }  isValue(value) {    return value != null && value !== '';  }  resolve(value, ...args) {    return utils.resolve(this, value, ...args);  }  get base() {    return Prompt.prototype;  }  get style() {    return this.styles[this.state.status];  }  get height() {    return this.options.rows || utils.height(this.stdout, 25);  }  get width() {    return this.options.columns || utils.width(this.stdout, 80);  }  get size() {    return { width: this.width, height: this.height };  }  set cursor(value) {    this.state.cursor = value;  }  get cursor() {    return this.state.cursor;  }  set input(value) {    this.state.input = value;  }  get input() {    return this.state.input;  }  set value(value) {    this.state.value = value;  }  get value() {    let { input, value } = this.state;    let result = [value, input].find(this.isValue.bind(this));    return this.isValue(result) ? result : this.initial;  }  static get prompt() {    return options => new this(options).run();  }}function setOptions(prompt) {  let isValidKey = key => {    return prompt[key] === void 0 || typeof prompt[key] === 'function';  };  let ignore = [    'actions',    'choices',    'initial',    'margin',    'roles',    'styles',    'symbols',    'theme',    'timers',    'value'  ];  let ignoreFn = [    'body',    'footer',    'error',    'header',    'hint',    'indicator',    'message',    'prefix',    'separator',    'skip'  ];  for (let key of Object.keys(prompt.options)) {    if (ignore.includes(key)) continue;    if (/^on[A-Z]/.test(key)) continue;    let option = prompt.options[key];    if (typeof option === 'function' && isValidKey(key)) {      if (!ignoreFn.includes(key)) {        prompt[key] = option.bind(prompt);      }    } else if (typeof prompt[key] !== 'function') {      prompt[key] = option;    }  }}function margin(value) {  if (typeof value === 'number') {    value = [value, value, value, value];  }  let arr = [].concat(value || []);  let pad = i => i % 2 === 0 ? '\n' : ' ';  let res = [];  for (let i = 0; i < 4; i++) {    let char = pad(i);    if (arr[i]) {      res.push(char.repeat(arr[i]));    } else {      res.push('');    }  }  return res;}module.exports = Prompt;
 |