chalk/index.js

100 lines
2.3 KiB
JavaScript
Raw Normal View History

2013-08-03 02:16:26 +02:00
'use strict';
var escapeStringRegexp = require('escape-string-regexp');
2014-06-04 01:43:07 +02:00
var ansiStyles = require('ansi-styles');
var stripAnsi = require('strip-ansi');
2014-06-24 21:34:11 +02:00
var hasAnsi = require('has-ansi');
2014-06-14 03:49:42 +02:00
var supportsColor = require('supports-color');
2013-08-03 02:16:26 +02:00
var defineProps = Object.defineProperties;
2013-12-13 19:36:43 +01:00
var chalk = module.exports;
2013-08-29 16:14:18 +02:00
// use bright blue on Windows as the normal blue color is illegible
if (process.platform === 'win32') {
ansiStyles.blue.open = '\u001b[94m';
}
function build(_styles) {
var builder = function builder() {
return applyStyle.apply(builder, arguments);
};
builder._styles = _styles;
// __proto__ is used because we must return a function, but there is
// no way to create a function with a different prototype.
builder.__proto__ = proto;
return builder;
}
2013-08-03 02:16:26 +02:00
var styles = (function () {
var ret = {};
2014-06-04 01:43:07 +02:00
Object.keys(ansiStyles).forEach(function (key) {
ansiStyles[key].closeRe = new RegExp(escapeStringRegexp(ansiStyles[key].close), 'g');
2013-08-03 02:16:26 +02:00
ret[key] = {
get: function () {
return build(this._styles.concat(key));
2013-08-03 02:16:26 +02:00
}
};
});
return ret;
})();
var proto = defineProps(function chalk() {}, styles);
function applyStyle() {
// support varags, but simply cast to string in case there's only one arg
var args = arguments;
var argsLen = args.length;
var str = argsLen !== 0 && String(arguments[0]);
if (argsLen > 1) {
// don't slice `arguments`, it prevents v8 optimizations
for (var a = 1; a < argsLen; a++) {
str += ' ' + args[a];
}
}
if (!chalk.enabled || !str) {
return str;
}
/*jshint validthis: true */
var nestedStyles = this._styles;
var i = nestedStyles.length;
while (i--) {
var code = ansiStyles[nestedStyles[i]];
// Replace any instances already present with a re-opening code
// otherwise only the part of the string until said closing code
// will be colored, and the rest will simply be 'plain'.
str = code.open + str.replace(code.closeRe, code.open) + code.close;
}
return str;
}
2013-08-03 02:16:26 +02:00
function init() {
var ret = {};
Object.keys(styles).forEach(function (name) {
ret[name] = {
get: function () {
return build([name]);
2013-08-03 02:16:26 +02:00
}
2013-12-16 19:25:01 +01:00
};
2013-08-03 02:16:26 +02:00
});
return ret;
}
2013-12-13 19:36:43 +01:00
defineProps(chalk, init());
2014-06-04 01:43:07 +02:00
chalk.styles = ansiStyles;
2014-06-24 21:34:11 +02:00
chalk.hasColor = hasAnsi;
chalk.stripColor = stripAnsi;
2014-06-14 03:49:42 +02:00
chalk.supportsColor = supportsColor;
2013-08-03 02:16:26 +02:00
// detect mode if not set manually
if (chalk.enabled === undefined) {
chalk.enabled = chalk.supportsColor;
}