import type Position from "./position"; import repeating from "repeating"; import trimRight from "trim-right"; /** * Buffer for collecting generated output. */ export default class Buffer { constructor(position: Position, format: Object) { this.printedCommentStarts = {}; this.parenPushNewlineState = null; this.position = position; this._indent = format.indent.base; this.format = format; this.buf = ""; this.last = ""; } parenPushNewlineState: ?Object; buf: string; position: Position; _indent: number; format: Object; /** * Description */ catchUp(node) { // catch up to this nodes newline if we're behind if (node.loc && this.format.retainLines && this.buf) { while (this.position.line < node.loc.start.line) { this._push("\n"); } } } /** * Get the current trimmed buffer. */ get() { return trimRight(this.buf); } /** * Get the current indent. */ getIndent() { if (this.format.compact || this.format.concise) { return ""; } else { return repeating(this.format.indent.style, this._indent); } } /** * Get the current indent size. */ indentSize() { return this.getIndent().length; } /** * Increment indent size. */ indent() { this._indent++; } /** * Decrement indent size. */ dedent() { this._indent--; } /** * Add a semicolon to the buffer. */ semicolon() { this.push(";"); } /** * Ensure last character is a semicolon. */ ensureSemicolon() { if (!this.isLast(";")) this.semicolon(); } /** * Add a right brace to the buffer. */ rightBrace() { this.newline(true); if (this.format.minified && !this._lastPrintedIsEmptyStatement) { this._removeLast(";"); } this.push("}"); } /** * Add a keyword to the buffer. */ keyword(name: string) { this.push(name); this.space(); } /** * Add a space to the buffer unless it is compact (override with force). */ space(force?: boolean) { if (!force && this.format.compact) return; if (force || this.buf && !this.isLast(" ") && !this.isLast("\n")) { this.push(" "); } } /** * Remove the last character. */ removeLast(cha: string) { if (this.format.compact) return; return this._removeLast(cha); } _removeLast(cha: string) { if (!this._isLast(cha)) return; this.buf = this.buf.slice(0, -1); this.last = this.buf[this.buf.length - 1]; this.position.unshift(cha); } /** * Set some state that will be modified if a newline has been inserted before any * non-space characters. * * This is to prevent breaking semantics for terminatorless separator nodes. eg: * * return foo; * * returns `foo`. But if we do: * * return * foo; * * `undefined` will be returned and not `foo` due to the terminator. */ startTerminatorless(): Object { return this.parenPushNewlineState = { printed: false }; } /** * Print an ending parentheses if a starting one has been printed. */ endTerminatorless(state: Object) { if (state.printed) { this.dedent(); this.newline(); this.push(")"); } } /** * Add a newline (or many newlines), maintaining formatting. * Strips multiple newlines if removeLast is true. */ newline(i?: boolean | number, removeLast?: boolean) { if (this.format.retainLines || this.format.compact) return; if (this.format.concise) { this.space(); return; } // never allow more than two lines if (this.endsWith("\n\n")) return; if (typeof i === "boolean") removeLast = i; if (typeof i !== "number") i = 1; i = Math.min(2, i); if (this.endsWith("{\n") || this.endsWith(":\n")) i--; if (i <= 0) return; // remove the last newline if (removeLast) { this.removeLast("\n"); } this.removeLast(" "); this._removeSpacesAfterLastNewline(); this._push(repeating("\n", i)); } /** * If buffer ends with a newline and some spaces after it, trim those spaces. */ _removeSpacesAfterLastNewline() { let lastNewlineIndex = this.buf.lastIndexOf("\n"); if (lastNewlineIndex >= 0 && this.buf.trimRight().length <= lastNewlineIndex) { this.buf = this.buf.substring(0, lastNewlineIndex + 1); this.last = "\n"; } } /** * Push a string to the buffer, maintaining indentation and newlines. */ push(str: string, noIndent?: boolean) { if (!this.format.compact && this._indent && !noIndent && str !== "\n") { // we have an indent level and we aren't pushing a newline let indent = this.getIndent(); // replace all newlines with newlines with the indentation str = str.replace(/\n/g, `\n${indent}`); // we've got a newline before us so prepend on the indentation if (this.isLast("\n")) this._push(indent); } this._push(str); } /** * Push a string to the buffer. */ _push(str) { // see startTerminatorless() instance method let parenPushNewlineState = this.parenPushNewlineState; if (parenPushNewlineState) { for (let i = 0; i < str.length; i++) { let cha = str[i]; // we can ignore spaces since they wont interupt a terminatorless separator if (cha === " ") continue; this.parenPushNewlineState = null; if (cha === "\n" || cha === "/") { // we're going to break this terminator expression so we need to add a parentheses this._push("("); this.indent(); parenPushNewlineState.printed = true; } break; } } // this.position.push(str); this.buf += str; this.last = str[str.length - 1]; } /** * Test if the buffer ends with a string. */ endsWith(str: string): boolean { if (str.length === 1) { return this.last === str; } else { return this.buf.slice(-str.length) === str; } } /** * Test if a character is last in the buffer. */ isLast(cha: string) { if (this.format.compact) return false; return this._isLast(cha); } _isLast(cha: string) { let last = this.last; if (Array.isArray(cha)) { return cha.indexOf(last) >= 0; } else { return cha === last; } } }