2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2020-05-25 18:32:34 +01:00
|
|
|
/** A module to print ANSI terminal colors. Inspired by chalk, kleur, and colors
|
2019-08-24 10:38:18 -07:00
|
|
|
* on npm.
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* import { bgBlue, red, bold } from "https://deno.land/std/fmt/colors.ts";
|
|
|
|
* console.log(bgBlue(red(bold("Hello world!"))));
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* This module supports `NO_COLOR` environmental variable disabling any coloring
|
|
|
|
* if `NO_COLOR` is set.
|
2020-05-25 18:32:34 +01:00
|
|
|
*
|
|
|
|
* This module is browser compatible. */
|
|
|
|
|
|
|
|
const noColor = globalThis.Deno?.noColor ?? true;
|
2019-01-19 21:09:35 +11:00
|
|
|
|
|
|
|
interface Code {
|
|
|
|
open: string;
|
|
|
|
close: string;
|
|
|
|
regexp: RegExp;
|
|
|
|
}
|
|
|
|
|
2020-05-09 20:29:44 +10:00
|
|
|
/** RGB 8-bits per channel. Each in range `0->255` or `0x00->0xff` */
|
|
|
|
interface Rgb {
|
|
|
|
r: number;
|
|
|
|
g: number;
|
|
|
|
b: number;
|
|
|
|
}
|
|
|
|
|
2019-02-10 01:13:44 +01:00
|
|
|
let enabled = !noColor;
|
2019-01-19 21:09:35 +11:00
|
|
|
|
2019-12-20 20:21:30 +00:00
|
|
|
export function setColorEnabled(value: boolean): void {
|
2019-02-10 01:13:44 +01:00
|
|
|
if (noColor) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-01-19 21:09:35 +11:00
|
|
|
enabled = value;
|
|
|
|
}
|
|
|
|
|
2019-12-20 20:21:30 +00:00
|
|
|
export function getColorEnabled(): boolean {
|
2019-01-19 21:09:35 +11:00
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
2020-05-09 20:29:44 +10:00
|
|
|
function code(open: number[], close: number): Code {
|
2019-01-19 21:09:35 +11:00
|
|
|
return {
|
2020-05-09 20:29:44 +10:00
|
|
|
open: `\x1b[${open.join(";")}m`,
|
2019-01-19 21:09:35 +11:00
|
|
|
close: `\x1b[${close}m`,
|
2020-03-29 04:03:49 +11:00
|
|
|
regexp: new RegExp(`\\x1b\\[${close}m`, "g"),
|
2019-01-19 21:09:35 +11:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-03-05 11:53:35 +11:00
|
|
|
function run(str: string, code: Code): string {
|
2019-01-19 21:09:35 +11:00
|
|
|
return enabled
|
|
|
|
? `${code.open}${str.replace(code.regexp, code.open)}${code.close}`
|
|
|
|
: str;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function reset(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([0], 0));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function bold(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([1], 22));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function dim(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([2], 22));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function italic(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([3], 23));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function underline(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([4], 24));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function inverse(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([7], 27));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function hidden(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([8], 28));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function strikethrough(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([9], 29));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function black(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([30], 39));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function red(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([31], 39));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function green(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([32], 39));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function yellow(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([33], 39));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function blue(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([34], 39));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function magenta(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([35], 39));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function cyan(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([36], 39));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function white(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([37], 39));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function gray(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([90], 39));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function bgBlack(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([40], 49));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function bgRed(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([41], 49));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function bgGreen(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([42], 49));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function bgYellow(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([43], 49));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function bgBlue(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([44], 49));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function bgMagenta(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([45], 49));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function bgCyan(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([46], 49));
|
2019-01-19 21:09:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export function bgWhite(str: string): string {
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(str, code([47], 49));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Special Color Sequences */
|
|
|
|
|
|
|
|
function clampAndTruncate(n: number, max = 255, min = 0): number {
|
|
|
|
return Math.trunc(Math.max(Math.min(n, max), min));
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Set text color using paletted 8bit colors.
|
|
|
|
* https://en.wikipedia.org/wiki/ANSI_escape_code#8-bit */
|
|
|
|
export function rgb8(str: string, color: number): string {
|
|
|
|
return run(str, code([38, 5, clampAndTruncate(color)], 39));
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Set background color using paletted 8bit colors.
|
|
|
|
* https://en.wikipedia.org/wiki/ANSI_escape_code#8-bit */
|
|
|
|
export function bgRgb8(str: string, color: number): string {
|
|
|
|
return run(str, code([48, 5, clampAndTruncate(color)], 49));
|
|
|
|
}
|
|
|
|
|
2020-05-21 00:29:59 +10:00
|
|
|
/** Set text color using 24bit rgb.
|
|
|
|
* `color` can be a number in range `0x000000` to `0xffffff` or
|
|
|
|
* an `Rgb`.
|
|
|
|
*
|
|
|
|
* To produce the color magenta:
|
|
|
|
*
|
|
|
|
* rgba24("foo", 0xff00ff);
|
|
|
|
* rgba24("foo", {r: 255, g: 0, b: 255});
|
|
|
|
*/
|
|
|
|
export function rgb24(str: string, color: number | Rgb): string {
|
|
|
|
if (typeof color === "number") {
|
|
|
|
return run(
|
|
|
|
str,
|
2020-07-14 15:24:17 -04:00
|
|
|
code(
|
|
|
|
[38, 2, (color >> 16) & 0xff, (color >> 8) & 0xff, color & 0xff],
|
|
|
|
39,
|
|
|
|
),
|
2020-05-21 00:29:59 +10:00
|
|
|
);
|
|
|
|
}
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(
|
|
|
|
str,
|
|
|
|
code(
|
|
|
|
[
|
|
|
|
38,
|
|
|
|
2,
|
|
|
|
clampAndTruncate(color.r),
|
|
|
|
clampAndTruncate(color.g),
|
|
|
|
clampAndTruncate(color.b),
|
|
|
|
],
|
2020-07-14 15:24:17 -04:00
|
|
|
39,
|
|
|
|
),
|
2020-05-09 20:29:44 +10:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-05-21 00:29:59 +10:00
|
|
|
/** Set background color using 24bit rgb.
|
|
|
|
* `color` can be a number in range `0x000000` to `0xffffff` or
|
|
|
|
* an `Rgb`.
|
|
|
|
*
|
|
|
|
* To produce the color magenta:
|
|
|
|
*
|
|
|
|
* bgRgba24("foo", 0xff00ff);
|
|
|
|
* bgRgba24("foo", {r: 255, g: 0, b: 255});
|
|
|
|
*/
|
|
|
|
export function bgRgb24(str: string, color: number | Rgb): string {
|
|
|
|
if (typeof color === "number") {
|
|
|
|
return run(
|
|
|
|
str,
|
2020-07-14 15:24:17 -04:00
|
|
|
code(
|
|
|
|
[48, 2, (color >> 16) & 0xff, (color >> 8) & 0xff, color & 0xff],
|
|
|
|
49,
|
|
|
|
),
|
2020-05-21 00:29:59 +10:00
|
|
|
);
|
|
|
|
}
|
2020-05-09 20:29:44 +10:00
|
|
|
return run(
|
|
|
|
str,
|
|
|
|
code(
|
|
|
|
[
|
|
|
|
48,
|
|
|
|
2,
|
|
|
|
clampAndTruncate(color.r),
|
|
|
|
clampAndTruncate(color.g),
|
|
|
|
clampAndTruncate(color.b),
|
|
|
|
],
|
2020-07-14 15:24:17 -04:00
|
|
|
49,
|
|
|
|
),
|
2020-05-09 20:29:44 +10:00
|
|
|
);
|
2018-12-19 15:30:44 +11:00
|
|
|
}
|
2020-05-19 20:19:26 +02:00
|
|
|
|
|
|
|
// https://github.com/chalk/ansi-regex/blob/2b56fb0c7a07108e5b54241e8faec160d393aedb/index.js
|
|
|
|
const ANSI_PATTERN = new RegExp(
|
|
|
|
[
|
|
|
|
"[\\u001B\\u009B][[\\]()#;?]*(?:(?:(?:[a-zA-Z\\d]*(?:;[-a-zA-Z\\d\\/#&.:=?%@~_]*)*)?\\u0007)",
|
|
|
|
"(?:(?:\\d{1,4}(?:;\\d{0,4})*)?[\\dA-PR-TZcf-ntqry=><~]))",
|
|
|
|
].join("|"),
|
2020-07-14 15:24:17 -04:00
|
|
|
"g",
|
2020-05-19 20:19:26 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
export function stripColor(string: string): string {
|
|
|
|
return string.replace(ANSI_PATTERN, "");
|
|
|
|
}
|