/* jslint node: true */ 'use strict'; var events = require('events'); var util = require('util'); var miscUtil = require('./misc_util.js'); var ansi = require('./ansi_term.js'); exports.ANSIEscapeParser = ANSIEscapeParser; function ANSIEscapeParser(options) { var self = this; events.EventEmitter.call(this); this.column = 1; this.row = 1; this.style = 0x00; //this.style = { 0 : true }; this.scrollBack = 0; options = miscUtil.valueWithDefault(options, { mciReplaceChar : '', termHeight : 25, termWidth : 80, }); this.mciReplaceChar = miscUtil.valueWithDefault(options.mciReplaceChar, ''); this.termHeight = miscUtil.valueWithDefault(options.termHeight, 25); this.termWidth = miscUtil.valueWithDefault(options.termWidth, 80); function getArgArray(array) { var i = array.length; while(i--) { array[i] = parseInt(array[i], 10); } return array; } self.moveCursor = function(cols, rows) { self.column += cols; self.row += rows; self.column = Math.max(self.column, 1); self.column = Math.min(self.column, self.termWidth); self.row = Math.max(self.row, 1); self.row = Math.min(self.row, self.termHeight); self.emit('move cursor', self.column, self.row); self.rowUpdated(); }; self.saveCursorPosition = function() { self.savedPosition = { row : self.row, column : self.column }; }; self.restoreCursorPosition = function() { self.row = self.savedPosition.row; self.column = self.savedPosition.column; delete self.savedPosition; self.rowUpdated(); }; self.clearScreen = function() { // :TODO: should be doing something with row/column? self.emit('clear screen'); }; self.resetColor = function() { //self.fgColor = 7; //self.bgColor = 0; self.fgColor = 39; self.bgColor = 49; //self.style = { 0 : true }; //delete self.style; self.style = 0; }; self.rowUpdated = function() { self.emit('row update', self.row + self.scrollBack); }; function literal(text) { var CR = 0x0d; var LF = 0x0a; var charCode; var len = text.length; for(var i = 0; i < len; i++) { charCode = text.charCodeAt(i) & 0xff; // ensure 8 bit switch(charCode) { case CR : self.column = 1; break; case LF : self.row++; self.rowUpdated(); break; default : // wrap if(self.column === self.termWidth) { self.column = 1; self.row++; self.rowUpdated(); } else { self.column++; } break; } if(self.row === 26) { // :TODO: should be termHeight + 1 ? self.scrollBack++; self.row--; self.rowUpdated(); } } self.emit('chunk', text); } function getProcessedMCI(mci) { if(self.mciReplaceChar.length > 0) { var eraseColor = ansi.sgr(self.eraseColor.style, self.eraseColor.fgColor, self.eraseColor.bgColor); return eraseColor + new Array(mci.length + 1).join(self.mciReplaceChar); } else { return mci; } } function parseMCI(buffer) { var mciRe = /\%([A-Z]{2})([0-9]{1,2})?(?:\(([0-9A-Z,]+)\))*/g; var pos = 0; var match; var mciCode; var args; var id; do { pos = mciRe.lastIndex; match = mciRe.exec(buffer); if(null !== match) { if(match.index > pos) { literal(buffer.slice(pos, match.index)); } mciCode = match[1]; id = match[2] || null; if(match[3]) { args = match[3].split(','); } else { args = []; } // if MCI codes are changing, save off the current color var fullMciCode = mciCode + (id || ''); if(self.lastMciCode !== fullMciCode) { self.lastMciCode = fullMciCode; self.eraseColor = { flags : self.style, fgColor : self.fgColor, bgColor : self.bgColor, }; } self.emit('mci', mciCode, id, args); if(self.mciReplaceChar.length > 0) { self.emit('chunk', ansi.sgr(self.eraseColor.style, self.eraseColor.fgColor, self.eraseColor.bgColor)); literal(new Array(match[0].length + 1).join(self.mciReplaceChar)); } else { literal(match[0]); } //literal(getProcessedMCI(match[0])); //self.emit('chunk', getProcessedMCI(match[0])); } } while(0 !== mciRe.lastIndex); if(pos < buffer.length) { literal(buffer.slice(pos)); } } self.parse = function(buffer, savedRe) { // :TODO: ensure this conforms to ANSI-BBS / CTerm / bansi.txt for movement/etc. var re = /(?:\x1b\x5b)([\?=;0-9]*?)([ABCDHJKfhlmnpsu])/g; var pos = 0; var match; var opCode; var args; // ignore anything past EOF marker, if any buffer = buffer.split(String.fromCharCode(0x1a), 1)[0]; do { pos = re.lastIndex; match = re.exec(buffer); if(null !== match) { if(match.index > pos) { parseMCI(buffer.slice(pos, match.index)); } opCode = match[2]; args = getArgArray(match[1].split(';')); escape(opCode, args); self.emit('chunk', match[0]); } } while(0 !== re.lastIndex); if(pos < buffer.length) { parseMCI(buffer.slice(pos)); } self.emit('complete'); }; function escape(opCode, args) { var arg; var i; var len; switch(opCode) { // cursor up case 'A' : arg = args[0] || 1; self.moveCursor(0, -arg); break; // cursor down case 'B' : arg = args[0] || 1; self.moveCursor(0, arg); break; // cursor forward/right case 'C' : arg = args[0] || 1; self.moveCursor(arg, 0); break; // cursor back/left case 'D' : arg = args[0] || 1; self.moveCursor(-arg, 0); break; case 'f' : // horiz & vertical case 'H' : // cursor position self.row = args[0] || 1; self.column = args[1] || 1; self.rowUpdated(); break; // save position case 's' : self.saveCursorPosition(); break; // restore position case 'u' : self.restoreCursorPosition(); break; // set graphic rendition case 'm' : // :TODO: reset state here for new system for(i = 0, len = args.length; i < len; ++i) { arg = args[i]; // :TODO: finish this system // * style is map of styleName -> boolean // * change self.style -> self.styles // * Change all fg/bg/etc -> self.state.color { fg, bg, style{} } // * Change all refs to use this new system // * When passing color -> sgr, iterate enabled styles -> additional params // * view.getANSIColor() will need updated // * art.js will need updated /* if(ANSIEscapeParser.foregroundColors[arg]) { self.fgColor = arg;//ANSIEscapeParser.foregroundColors[arg]; } else if(ANSIEscapeParser.backgroundColors[arg]) { self.bgColor = arg;//ANSIEscapeParser.backgroundColors[arg]; } else if(39 === arg) { delete self.fgColor; } else if(49 === arg) { delete self.bgColor; } else if(ANSIEscapeParser.styles[arg]) { self.style = arg; } */ if(arg >= 30 && arg <= 37) { self.fgColor = arg; } else if(arg >= 40 && arg <= 47) { self.bgColor = arg; } else { self.style |= arg; if(0 === arg) { self.resetColor(); //self.style = 0; } } } break; // erase display/screen case 'J' : // :TODO: Handle others if(2 === args[0]) { self.clearScreen(); } break; } } this.resetColor(); } util.inherits(ANSIEscapeParser, events.EventEmitter); ANSIEscapeParser.foregroundColors = { 30 : 'black', 31 : 'red', 32 : 'green', 33 : 'yellow', 34 : 'blue', 35 : 'magenta', 36 : 'cyan', 37 : 'white', 90 : 'grey' }; Object.freeze(ANSIEscapeParser.foregroundColors); ANSIEscapeParser.backgroundColors = { 40 : 'black', 41 : 'red', 42 : 'green', 43 : 'yellow', 44 : 'blue', 45 : 'magenta', 46 : 'cyan', 47 : 'white' }; Object.freeze(ANSIEscapeParser.backgroundColors); ANSIEscapeParser.styles = { 0 : 'default', 1 : 'bright', 2 : 'dim', 5 : 'slow blink', 6 : 'fast blink', 7 : 'negative', 8 : 'concealed', 22 : 'normal', 27 : 'positive', }; Object.freeze(ANSIEscapeParser.styles);