| '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; |