UNPKG

ansi-graphics

Version:
676 lines (610 loc) 22.7 kB
var util = require('util'), fs = require('fs'), spawn = require('child_process').spawn, events = require('events'), Canvas = require('canvas'), Image = Canvas.Image, GIFEncoder = require('gifencoder'), defs = require('./defs.js'); var copyObject = function(obj) { var ret = {}; for(var property in obj) if(Array.isArray(obj[property])) ret[property] = obj[property]; else if(typeof obj[property] == "object") ret[property] = copyObject(obj[property]); else if(typeof obj[property] != "undefined") ret[property] = obj[property]; return ret; } // Very shallow comparison var compareObjects = function(obj1, obj2) { var ret = true; for(var property in obj1) { if(obj1[property] === obj2[property]) continue; ret = false; break; } return ret; } var ANSI = function() { var self = this; events.EventEmitter.call(this); this.data = []; var width = 0; var height = 0; this.__defineGetter__( "width", function() { return width + 1; } ); this.__defineGetter__( "height", function() { return height + 1; } ); this.__defineGetter__( "pixelWidth", function() { return (9 * (width + 1)); } ); this.__defineGetter__( "pixelHeight", function() { return (16 * (height + 1)); } ); this.fromString = function(ansiString) { var plain = ""; var cursor = { 'x' : 0, 'y' : 0 }; var cursorStore = { 'x' : 0, 'y' : 0 }; var graphics = { 'bright' : false, 'blink' : false, 'foreground' : 37, 'background' : 40 }; while(ansiString.length > 0) { var regex = /^\u001b\[(\d*;?)*[a-zA-Z]/; var result = regex.exec(ansiString); if(result === null) { var chr = { 'cursor' : copyObject(cursor), 'graphics' : copyObject(graphics), 'chr' : ansiString.substr(0, 1) }; switch(chr.chr.charCodeAt(0)) { case 13: cursor.x = 0; break; case 10: cursor.y++; break; default: cursor.x++; if(cursor.x == 80) { cursor.x = 0; cursor.y++; } this.data.push(chr); break; } ansiString = ansiString.substr(1); } else { var ansiSequence = ansiString.substr(0, result[0].length).replace(/^\u001b\[/, ""); var cmd = ansiSequence.substr(ansiSequence.length - 1); var opts = ansiSequence.substr(0, ansiSequence.length - 1).split(";"); opts.forEach( function(e, i, a) { a[i] = parseInt(e); } ); ansiString = ansiString.substr(result[0].length); switch(cmd) { case 'A': if(isNaN(opts[0])) opts[0] = 1; cursor.y = Math.max(cursor.y - opts[0], 0); break; case 'B': if(isNaN(opts[0])) opts[0] = 1; cursor.y = cursor.y + opts[0]; break; case 'C': if(isNaN(opts[0])) opts[0] = 1; cursor.x = Math.min(cursor.x + opts[0], 79); break; case 'D': if(isNaN(opts[0])) opts[0] = 1; cursor.x = Math.max(cursor.x - opts[0], 0); break; case 'f': cursor.y = (isNaN(opts[0])) ? 1 : opts[0]; cursor.x = (opts.length < 2) ? 1 : opts[1]; break; case 'H': cursor.y = (isNaN(opts[0])) ? 1 : opts[0]; cursor.x = (opts.length < 2) ? 1 : opts[1]; break; case 'm': for(var o in opts) { var i = parseInt(opts[o]); if(opts[o] == 0) { graphics.foreground = 37; graphics.background = 40; graphics.bright = false; graphics.blink = false; } else if(opts[o] == 1) { graphics.bright = true; } else if(opts[o] == 5) { graphics.blink = true; } else if(opts[o] >= 30 && opts[o] <= 37) { graphics.foreground = opts[o]; } else if(opts[o] >= 40 && opts[o] <= 47) { graphics.background = opts[o]; } } break; case 's': cursorStore = copyObject(cursor); break; case 'u': cursor = copyObject(cursorStore); break; case 'J': if(opts.length == 1 && opts[0] == 2) { /* for(var d in this.data) { var o = copyObject(this.data[d]); o.chr = " "; this.data.push(o); cursor.y = 0; cursor.x = 0; } */ for(var y = 0; y < 24; y++) { for(var x = 0; x < 80; x++) { this.data.push( { 'cursor' : { 'x' : x, 'y' : y }, 'graphics' : { 'bright' : false, 'blink' : false, 'foreground' : 37, 'background' : 40 }, 'chr' : " " } ); } } } break; case 'K': for(var d in this.data) { if(this.data[d].cursor.y != cursor.y || this.data[d].cursor.x < cursor.x) continue; var o = copyObject(this.data[d]); o.chr = " "; this.data.push(o); } break; default: // Unknown or unimplemented command break; } } width = Math.max(cursor.x, width); height = Math.max(cursor.y, height); } } this.fromFile = function(fileName) { var contents = fs.readFileSync(fileName, { 'encoding' : 'binary' }); this.fromString(contents); } this.__defineGetter__( "matrix", function() { var ret = {}; for(var d = 0; d < self.data.length; d++) { if(typeof ret[self.data[d].cursor.y] == "undefined") ret[self.data[d].cursor.y] = {}; ret[self.data[d].cursor.y][self.data[d].cursor.x] = { 'graphics' : copyObject(self.data[d].graphics), 'chr' : self.data[d].chr }; } for(var y = 0; y <= height; y++) { if(typeof ret[y] == "undefined") ret[y] = {}; for(var x = 0; x <= width; x++) { if(typeof ret[y][x] != "undefined") continue; ret[y][x] = { 'graphics' : { 'bright' : false, 'blink' : false, 'foreground' : 37, 'background' : 40 }, 'chr' : " " } } } return ret; } ); this.__defineGetter__( "plainText", function() { var lines = []; var matrix = self.matrix; for(var y in matrix) { var line = ""; for(var x in matrix[y]) line += matrix[y][x].chr; lines.push(line); } return lines.join("\r\n") + "\r\n"; } ); this.__defineGetter__( "HTML", function() { var graphics = { 'bright' : false, 'blink' : false, 'foreground' : 37, 'background' : 40 }; var graphicsToSpan = function(graphics) { var span = util.format( '<span style="background-color: %s; color: %s;">', defs.Attributes[graphics.background].htmlLow, (graphics.bright) ? defs.Attributes[graphics.foreground].htmlHigh : defs.Attributes[graphics.foreground].htmlLow ); return span; } var lines = [ '<pre style="font-family: Courier New, Courier, monospace; font-style: normal; font-weight: normal; letter-spacing: -1px; line-height: 1;">', graphicsToSpan(graphics) ]; var matrix = self.matrix; for(var y in matrix) { var line = ""; for(var x in matrix[y]) { if(!compareObjects(matrix[y][x].graphics, graphics)) { line += "</span>" + graphicsToSpan(matrix[y][x].graphics); graphics = copyObject(matrix[y][x].graphics); } line += (typeof defs.ASCIItoHTML[matrix[y][x].chr.charCodeAt(0)] == "undefined") ? ((matrix[y][x].chr == " ") ? "&nbsp;" : matrix[y][x].chr) : "&#" + defs.ASCIItoHTML[matrix[y][x].chr.charCodeAt(0)].entityNumber + ";"; } lines.push(line); } lines.push("</span>"); lines.push("</pre>\n"); return lines.join("\n"); } ); this.__defineGetter__( "binary", function() { var matrix = self.matrix; var bin = []; var width = 0; for(var y in matrix) { for(var x in matrix[y]) { width = Math.max(x, width); bin.push(matrix[y][x].chr.charCodeAt(0)); bin.push( defs.Attributes[matrix[y][x].graphics.foreground].attribute|defs.Attributes[matrix[y][x].graphics.background].attribute|((matrix[y][x].graphics.bright)?defs.Attributes[1].attribute:0)|((matrix[y][x].graphics.blink)?defs.Attributes[5].attribute:0) ); } } return new Buffer(bin); } ); this.toGIF = function(options) { if(typeof options != "object") options = {}; if(typeof options.scale != "number") options.scale = 1; var encoder = new GIFEncoder( Math.ceil(this.pixelWidth * options.scale), Math.ceil(this.pixelHeight * options.scale) ); var rs = encoder.createReadStream(); encoder.start(); encoder.setRepeat( (typeof options.loop != "boolean" || !options.loop) ? -1 : 0 ); encoder.setDelay( (typeof options.delay != "number") ? 40 : Math.round(options.delay) ); encoder.setQuality( (typeof options.quality != "number") ? 20 : Math.min(20, options.quality) ); var frames = (typeof options.charactersPerFrame != "number") ? 20 : Math.round(options.charactersPerFrame); var canvas = new ansiCanvas( this.pixelWidth, this.pixelHeight, options.scale ); for(var d = 0; d < self.data.length; d++) { canvas.putCharacter( self.data[d].cursor.x, self.data[d].cursor.y, self.data[d].chr.charCodeAt(0), defs.Attributes[self.data[d].graphics.foreground].attribute|((self.data[d].graphics.bright)?defs.Attributes[1].attribute:0), (defs.Attributes[self.data[d].graphics.background].attribute>>4) ); if(d % frames == 0) encoder.addFrame(canvas.context); } encoder.setDelay(5000); // Dwell on the last frame for a while. Make configurable? encoder.addFrame(canvas.context); encoder.finish(); return rs.read(); } this.toPNG = function(options) { if(typeof options != "object") options = {}; var matrix = self.matrix; var canvas = new ansiCanvas( this.pixelWidth, this.pixelHeight, (typeof options.scale == "number") ? options.scale : 1, (typeof options.quality == "number" && options.quality >= 0 && options.quality <= 4) ? options.quality : 4 ); for(var y in matrix) { for(var x in matrix[y]) { canvas.putCharacter( x, y, matrix[y][x].chr.charCodeAt(0), defs.Attributes[matrix[y][x].graphics.foreground].attribute|((matrix[y][x].graphics.bright)?defs.Attributes[1].attribute:0), (defs.Attributes[matrix[y][x].graphics.background].attribute>>4) ); } } return canvas.canvas.toBuffer(); } this.toVideo = function(options, callback) { if(arguments.length == 1 && typeof options != "function") this.emit("error", "ANSI.toMovie: Invalid callback"); else if(arguments.length == 1) var callback = options; if(arguments.length > 1 && (typeof options != "object" || typeof callback != "function")) this.emit("error", "ANSI.toMovie: Invalid arguments"); var movie = new Buffer(0); var canvas = new ansiCanvas( this.pixelWidth, this.pixelHeight, (typeof options.scale == "number") ? options.scale : 1 ); var child = spawn( 'ffmpeg', [ '-y', '-loglevel', 'quiet', '-f', 'image2pipe', '-c:v', 'png', '-r', (typeof options.frameRate != "number") ? 30 : options.frameRate, '-i', 'pipe:0', '-f', (typeof options.format == "undefined") ? "webm" : options.format, '-filter:v', 'setpts=' + ((typeof options.speed != "number") ? 1 : options.speed) + '*PTS', 'pipe:1' ] ); child.on( "close", function() { callback(movie); } ); child.stderr.on( "data", function(data) { console.log(data.toString()); } ); child.stdout.on( "data", function(data) { movie = Buffer.concat([movie, data]); } ); for(var d = 0; d < self.data.length; d++) { canvas.putCharacter( self.data[d].cursor.x, self.data[d].cursor.y, self.data[d].chr.charCodeAt(0), defs.Attributes[self.data[d].graphics.foreground].attribute|((self.data[d].graphics.bright)?defs.Attributes[1].attribute:0), (defs.Attributes[self.data[d].graphics.background].attribute>>4) ); if(d % ((typeof options.charactersPerFrame != "number") ? 20 : options.charactersPerFrame) == 0) child.stdin.write(canvas.canvas.toBuffer()); } child.stdin.write(canvas.canvas.toBuffer()); child.stdin.end(); } } util.inherits(ANSI, events.EventEmitter); // Lazily ported and modified from my old HTML5 ANSI editor // Could be simplified and folded into ANSI.toGIF() at some point var ansiCanvas = function(width, height, scale, quality) { var foregroundCanvas, foregroundContext, backgroundCanvas, backgroundContext, mergeCanvas, mergeContext; var properties = { 'width' : width, 'height' : height, 'scale' : (typeof scale != "number") ? 1 : scale, 'quality' : (typeof quality != "number" || quality < 1 || quality > 5) ? 4 : quality - 1, 'characters' : [], 'spriteSheet' : new Image(), 'spriteWidth' : 9, 'spriteHeight' : 16, 'colors' : [ "#000000", "#0000A8", "#00A800", "#00A8A8", "#A80000", "#A800A8", "#A85400", "#A8A8A8", "#545454", "#5454FC", "#54FC54", "#54FCFC", "#FC5454", "#FC54FC", "#FCFC54", "#FFFFFF" ], 'qualityMap' : [ 'bilinear', 'nearest', 'better', 'good', 'fast' ] }; var merge = function() { mergeContext.drawImage( backgroundCanvas, 0, 0, properties.width, properties.height, 0, 0, Math.ceil(properties.width * properties.scale), Math.ceil(properties.height * properties.scale) ); mergeContext.drawImage( foregroundCanvas, 0, 0, properties.width, properties.height, 0, 0, Math.ceil(properties.width * properties.scale), Math.ceil(properties.height * properties.scale) ); } this.__defineGetter__( "canvas", function() { merge(); return mergeCanvas; } ); this.__defineGetter__( "context", function() { merge(); return mergeContext; } ); var initSpriteSheet = function() { properties.spriteSheet.src = ""; for( var y = 0; y <= properties.spriteSheet.height; y = y + properties.spriteHeight ) { for( var x = 0; x < properties.spriteSheet.width; x = x + properties.spriteWidth ) { properties.characters.push( { 'x' : x, 'y' : y } ); } } } var initCanvas = function() { backgroundCanvas = new Canvas(properties.width, properties.height); backgroundContext = backgroundCanvas.getContext('2d'); backgroundContext.fillStyle = properties.colors[0]; backgroundContext.fillRect(0, 0, properties.width, properties.height); foregroundCanvas = new Canvas(properties.width, properties.height); foregroundContext = foregroundCanvas.getContext('2d'); mergeCanvas = new Canvas(Math.ceil(properties.width * properties.scale), Math.ceil(properties.height * properties.scale)); mergeContext = mergeCanvas.getContext('2d'); mergeContext.patternQuality = properties.qualityMap[properties.quality]; mergeContext.filter = properties.qualityMap[properties.quality]; } this.putCharacter = function(x, y, character, foregroundColor, backgroundColor) { x = Math.floor( (x * properties.spriteWidth) / properties.spriteWidth ) * properties.spriteWidth; y = Math.floor( (y * properties.spriteHeight) / properties.spriteHeight ) * properties.spriteHeight; foregroundContext.clearRect( x, y, properties.spriteWidth, properties.spriteHeight ); foregroundContext.drawImage( properties.spriteSheet, properties.characters[character].x, properties.characters[character].y, properties.spriteWidth, properties.spriteHeight, x, y, properties.spriteWidth, properties.spriteHeight ); foregroundContext.globalCompositeOperation = 'source-atop'; foregroundContext.fillStyle = properties.colors[foregroundColor]; foregroundContext.fillRect( x, y, properties.spriteWidth, properties.spriteHeight ); foregroundContext.globalCompositeOperation = 'source-over'; backgroundContext.fillStyle = properties.colors[backgroundColor]; backgroundContext.fillRect( x, y, properties.spriteWidth, properties.spriteHeight ); } initSpriteSheet(); initCanvas(); } module.exports = ANSI;