forked from orbit-oss/chalk
See: https://github.com/chalk/chalk/pull/330/#issuecomment-471977551 The issue seems to have been fixed in newer Windows 10 builds. We're not interested in adding a conditional for older Windows versions as the fix severely complicates the codebase, and it also creates problems for consumers as it makes the output unpredictable.
204 lines
5.2 KiB
JavaScript
204 lines
5.2 KiB
JavaScript
'use strict';
|
|
const escapeStringRegexp = require('escape-string-regexp');
|
|
const ansiStyles = require('ansi-styles');
|
|
const {stdout: stdoutColor} = require('supports-color');
|
|
const template = require('./templates.js');
|
|
|
|
// `supportsColor.level` → `ansiStyles.color[name]` mapping
|
|
const levelMapping = [
|
|
'ansi',
|
|
'ansi',
|
|
'ansi256',
|
|
'ansi16m'
|
|
];
|
|
|
|
// `color-convert` models to exclude from the Chalk API due to conflicts and such
|
|
const skipModels = new Set(['gray']);
|
|
|
|
const styles = Object.create(null);
|
|
|
|
function applyOptions(object, options = {}) {
|
|
if (options.level > 3 || options.level < 0) {
|
|
throw new Error('The `level` option should be an integer from 0 to 3');
|
|
}
|
|
|
|
// Detect level if not set manually
|
|
const colorLevel = stdoutColor ? stdoutColor.level : 0;
|
|
object.level = options.level === undefined ? colorLevel : options.level;
|
|
object.enabled = 'enabled' in options ? options.enabled : object.level > 0;
|
|
}
|
|
|
|
class ChalkClass {
|
|
constructor(options) {
|
|
return chalkFactory(options);
|
|
}
|
|
}
|
|
|
|
function chalkFactory(options) {
|
|
const chalk = {};
|
|
applyOptions(chalk, options);
|
|
|
|
chalk.template = (...arguments_) => chalkTag(chalk.template, ...arguments_);
|
|
|
|
Object.setPrototypeOf(chalk, Chalk.prototype);
|
|
Object.setPrototypeOf(chalk.template, chalk);
|
|
|
|
chalk.template.constructor = () => {
|
|
throw new Error('`chalk.constructor()` is deprecated. Use `new chalk.Instance()` instead.');
|
|
};
|
|
|
|
chalk.template.Instance = ChalkClass;
|
|
|
|
return chalk.template;
|
|
}
|
|
|
|
function Chalk(options) {
|
|
return chalkFactory(options);
|
|
}
|
|
|
|
for (const [styleName, style] of Object.entries(ansiStyles)) {
|
|
style.closeRe = new RegExp(escapeStringRegexp(style.close), 'g');
|
|
|
|
styles[styleName] = {
|
|
get() {
|
|
return build.call(this, [...(this._styles || []), style], this._empty);
|
|
}
|
|
};
|
|
}
|
|
|
|
styles.visible = {
|
|
get() {
|
|
return build.call(this, this._styles || [], true);
|
|
}
|
|
};
|
|
|
|
ansiStyles.color.closeRe = new RegExp(escapeStringRegexp(ansiStyles.color.close), 'g');
|
|
for (const model of Object.keys(ansiStyles.color.ansi)) {
|
|
if (skipModels.has(model)) {
|
|
continue;
|
|
}
|
|
|
|
styles[model] = {
|
|
get() {
|
|
const {level} = this;
|
|
return function (...args) {
|
|
const open = ansiStyles.color[levelMapping[level]][model](...args);
|
|
const codes = {
|
|
open,
|
|
close: ansiStyles.color.close,
|
|
closeRe: ansiStyles.color.closeRe
|
|
};
|
|
return build.call(this, [...(this._styles || []), codes], this._empty);
|
|
};
|
|
}
|
|
};
|
|
}
|
|
|
|
ansiStyles.bgColor.closeRe = new RegExp(escapeStringRegexp(ansiStyles.bgColor.close), 'g');
|
|
for (const model of Object.keys(ansiStyles.bgColor.ansi)) {
|
|
if (skipModels.has(model)) {
|
|
continue;
|
|
}
|
|
|
|
const bgModel = 'bg' + model[0].toUpperCase() + model.slice(1);
|
|
styles[bgModel] = {
|
|
get() {
|
|
const {level} = this;
|
|
return function (...arguments_) {
|
|
const open = ansiStyles.bgColor[levelMapping[level]][model](...arguments_);
|
|
const codes = {
|
|
open,
|
|
close: ansiStyles.bgColor.close,
|
|
closeRe: ansiStyles.bgColor.closeRe
|
|
};
|
|
return build.call(this, [...(this._styles || []), codes], this._empty);
|
|
};
|
|
}
|
|
};
|
|
}
|
|
|
|
const proto = Object.defineProperties(() => {}, styles);
|
|
|
|
function build(_styles, _empty) {
|
|
const builder = (...arguments_) => applyStyle.call(builder, ...arguments_);
|
|
builder._styles = _styles;
|
|
builder._empty = _empty;
|
|
|
|
const self = this;
|
|
|
|
Object.defineProperty(builder, 'level', {
|
|
enumerable: true,
|
|
get() {
|
|
return self.level;
|
|
},
|
|
set(level) {
|
|
self.level = level;
|
|
}
|
|
});
|
|
|
|
Object.defineProperty(builder, 'enabled', {
|
|
enumerable: true,
|
|
get() {
|
|
return self.enabled;
|
|
},
|
|
set(enabled) {
|
|
self.enabled = enabled;
|
|
}
|
|
});
|
|
|
|
// `__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; // eslint-disable-line no-proto
|
|
|
|
return builder;
|
|
}
|
|
|
|
function applyStyle(...arguments_) {
|
|
let string = arguments_.join(' ');
|
|
|
|
if (!this.enabled || this.level <= 0 || !string) {
|
|
return this._empty ? '' : string;
|
|
}
|
|
|
|
for (const code of this._styles.slice().reverse()) {
|
|
// 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'.
|
|
string = code.open + string.replace(code.closeRe, code.open) + code.close;
|
|
|
|
// Close the styling before a linebreak and reopen
|
|
// after next line to fix a bleed issue on macOS
|
|
// https://github.com/chalk/chalk/pull/92
|
|
string = string.replace(/\r?\n/g, `${code.close}$&${code.open}`);
|
|
}
|
|
|
|
return string;
|
|
}
|
|
|
|
function chalkTag(chalk, ...strings) {
|
|
const [firstString] = strings;
|
|
|
|
if (!Array.isArray(firstString)) {
|
|
// If chalk() was called by itself or with a string,
|
|
// return the string itself as a string.
|
|
return strings.join(' ');
|
|
}
|
|
|
|
const arguments_ = strings.slice(1);
|
|
const parts = [firstString.raw[0]];
|
|
|
|
for (let i = 1; i < firstString.length; i++) {
|
|
parts.push(
|
|
String(arguments_[i - 1]).replace(/[{}\\]/g, '\\$&'),
|
|
String(firstString.raw[i])
|
|
);
|
|
}
|
|
|
|
return template(chalk, parts.join(''));
|
|
}
|
|
|
|
Object.defineProperties(Chalk.prototype, styles);
|
|
|
|
module.exports = Chalk(); // eslint-disable-line new-cap
|
|
module.exports.supportsColor = stdoutColor;
|
|
module.exports.default = module.exports; // For TypeScript
|