UNPKG

jsdoc-75lb

Version:

An API documentation generator for JavaScript.

130 lines (107 loc) 3.68 kB
'use strict'; var _ = require('lodash'); var util = require('./readline'); var cliWidth = require('cli-width'); var stripAnsi = require('strip-ansi'); var stringWidth = require('string-width'); // Prevent crashes on environments where the width can't be properly detected cliWidth.defaultWidth = 80; function height(content) { return content.split('\n').length; } function lastLine(content) { return _.last(content.split('\n')); } function normalizedCliWidth() { var width = cliWidth() || cliWidth.defaultWidth; if (process.platform === 'win32') { return width - 1; } return width; } var ScreenManager = module.exports = function (rl) { // These variables are keeping information to allow correct prompt re-rendering this.height = 0; this.extraLinesUnderPrompt = 0; this.rl = rl; }; ScreenManager.prototype.render = function (content, bottomContent) { this.rl.output.unmute(); this.clean(this.extraLinesUnderPrompt); /** * Write message to screen and setPrompt to control backspace */ var promptLine = lastLine(content); var rawPromptLine = stripAnsi(promptLine); // Remove the rl.line from our prompt. We can't rely on the content of // rl.line (mainly because of the password prompt), so just rely on it's // length. var prompt = promptLine; if (this.rl.line.length) { prompt = prompt.slice(0, -this.rl.line.length); } this.rl.setPrompt(prompt); // setPrompt will change cursor position, now we can get correct value var cursorPos = this.rl._getCursorPos(); content = forceLineReturn(content); if (bottomContent) { bottomContent = forceLineReturn(bottomContent); } // Manually insert an extra line if we're at the end of the line. // This prevent the cursor from appearing at the beginning of the // current line. if (rawPromptLine.length % normalizedCliWidth() === 0) { content = content + '\n'; } var fullContent = content + (bottomContent ? '\n' + bottomContent : ''); this.rl.output.write(fullContent); /** * Re-adjust the cursor at the correct position. */ // We need to consider parts of the prompt under the cursor as part of the bottom // content in order to correctly cleanup and re-render. var promptLineUpDiff = Math.floor(rawPromptLine.length / normalizedCliWidth()) - cursorPos.rows; var bottomContentHeight = promptLineUpDiff + (bottomContent ? height(bottomContent) : 0); if (bottomContentHeight > 0) { util.up(this.rl, bottomContentHeight); } // Reset cursor at the beginning of the line util.left(this.rl, stringWidth(lastLine(fullContent))); // Adjust cursor on the right util.right(this.rl, cursorPos.cols); /** * Set up state for next re-rendering */ this.extraLinesUnderPrompt = bottomContentHeight; this.height = height(fullContent); this.rl.output.mute(); }; ScreenManager.prototype.clean = function (extraLines) { if (extraLines > 0) { util.down(this.rl, extraLines); } util.clearLine(this.rl, this.height); }; ScreenManager.prototype.done = function () { this.rl.setPrompt(''); this.rl.output.unmute(); this.rl.output.write('\n'); }; function breakLines(lines) { // Break lines who're longuer than the cli width so we can normalize the natural line // returns behavior accross terminals. var width = normalizedCliWidth(); var regex = new RegExp( '(?:(?:\\033\[[0-9;]*m)*.?){1,' + width + '}', 'g' ); return lines.map(function (line) { var chunk = line.match(regex); // last match is always empty chunk.pop(); return chunk || ''; }); } function forceLineReturn(content) { return _.flatten(breakLines(content.split('\n'))).join('\n'); }