ansi-graphics
Version:
An ANSI graphics parser.
676 lines (610 loc) • 22.7 kB
JavaScript
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 == " ") ? " " : 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;