README.md
May 31, 2026 Β· View on GitHub

ANSI color library for use in terminals, CI environments, and Chromium-based browsers.
Ansis is focused on small size and speed while providing rich functionality and handling edge cases.
// Named imports - for cleaner, more readable code
import { red, cyan, bold, hex, rgb } from 'ansis';
// Clean chained syntax with template literals - no extra parentheses
console.log(bold.bgRed` FAIL `);
// Nested templates - no string concatenation needed
console.log(red`β Error: ${cyan`file.js`} not found`);
// Truecolor: hex and rgb
console.log(hex('#FF75D1').bold`Pink`);
console.log(rgb(224, 17, 95).italic`Ruby`);
π Shortcuts
π Getting Started π Ansis vs styleText() π Migrating from βοΈ Alternatives β
Compare features
π» CLI Environment π§ͺ CLI Testing βοΈ Compatibility π§Troubleshooting
π Upgrading to v4 Β· New features Β· Breaking changes
βοΈ Star History
β¨ Highlights
π¨ Colors - 16 Β· 256 Β· Truecolor (hex/rgb) Β· Named colors (orange, pink ...)
βοΈ Syntax - Chained Β· Template literals Β· Nested templates
βοΈ Works - ESM Β· CJS Β· TS Β· Node 10+ Β· Bun Β· Deno Β· CI Β· Chromium browsers
π§ Smart - Auto color detection Β· Fallback (truecolor β 256 β 16 β b&w) Β· NO_COLOR Β· FORCE_COLOR
π¦ Tiny - 5.8 kB Β· Drop-in replacement for Chalk (44 kB)
β‘ Performance
Ansis is the fastest when using 2 or more styles, which is the common real-world use case.
| Library | 1 style | 2+ styles |
|---|---|---|
ansis | 60M ops/sec | π 60M ops/sec |
picocolors | π 109M ops/sec | 58M ops/sec |
chalk | 55M ops/sec | 47M ops/sec |
util.styleText | 0.5M ops/sec | 0.5M ops/sec |
π‘ Features
π¨ Colors & Styles
- ANSI styles:
dimbolditalicunderline...strikethrough - ANSI 16 colors:
redredBrightbgRedbgRedBright... - ANSI 256 colors:
fg(n)bg(n) - Truecolor:
hex('#FF75D1')rgb(224, 17, 95)bgHex()bgRgb() - Named truecolors:
orange()bgPink()... - viaextend()
βοΈ Syntax
- Chained styles:
red.bold.underline- no nested calls likered(bold(underline())) - Template literals:
red`text`- no parentheses - Nested templates:
red`Error: ${cyan`file.js`} not found`- no string concatenations
π οΈ Utilities
- Strip ANSI codes:
ansis.strip(red('text'))β plain'text' - Hyperlink:
blue.link('https://...', 'Click here'),link('https://...') - Raw escape codes:
open/close-`${red.open}Error${red.close} file not found`
π» Environment
- Auto color detection + Fallback: Truecolor β 256 β 16 β b&w
- ENV variables:
NO_COLORFORCE_COLORCOLORTERM - CLI flags:
--no-color--color - CLI testing: force color levels in tests
βοΈ Compatibility
- ESM Β· CJS Β· TypeScript Β· Bun Β· Deno Β· Next.js Β· CI (GitHub and others)
- Chromium browsers: Chrome Β· Edge Β· Opera Β· Brave Β· Vivaldi
- Drop-in replacement for
chalkansi-colorscoloretteand others
π― You might also like
ansilight- Truecolor syntax highlighter for the terminal with 256highlight.jsthemesflaget- CLI argument parsing. A smaller (5 kB) and faster alternative toyargs-parser(85 kB)
βοΈ Alternatives
chalk, picocolors, colorette, kleur, ansi-colors, kolorist, cli-color, colors-cli, colors.js, tinyrainbow
Since Node.js 22 supports ANSI styling natively via util.styleText(), it is recommended for simple use cases where 16 colors are enough and top performance is not critical.
See styleText() limitations.
β Compare features π Benchmarks π§© Handling edge cases
Install
npm install ansis
For Node.js 10β12+ use special build
npm install ansis@node10
Usage
ESM
import ansis, { red, bold, fg, hex, rgb } from 'ansis';
CJS
const ansis = require('ansis');
const { red, bold, fg, hex, rgb } = require('ansis');
Chained syntax
All colors, styles and functions are chainable. Each color or style can be combined in any order.
import { red, bold, hex } from 'ansis';
red.bold`text`;
hex('#FF75D1').bgCyan.bold`text`;
bold.hex('#FF75D1').bgCyan`text`;
Template literals
Omit parentheses to keep your code clean:
import { red, yellow, green } from 'ansis';
red`Error`; // no parentheses
red`Red ${yellow`Yellow ${green`Green`} Yellow`} Red`; // deep nested templates
Escape sequences work exactly as in regular strings:
red`Hello\nWorld`; // two lines in red
red`prev\\next`; // one line: prev\next
β top
ANSI Styles
dim bold italic underline strikethroughinverse visible hidden reset
ANSI 16 colors
There are 16 basic colors: 8 standard and 8 bright variants.
| Example | Color | Background | Bright Example | Bright Color | Bright Background |
|---|---|---|---|---|---|
![]() | black | bgBlack | ![]() | gray | bgGray |
![]() | red | bgRed | ![]() | redBright | bgRedBright |
![]() | green | bgGreen | ![]() | greenBright | bgGreenBright |
![]() | yellow | bgYellow | ![]() | yellowBright | bgYellowBright |
![]() | blue | bgBlue | ![]() | blueBright | bgBlueBright |
![]() | magenta | bgMagenta | ![]() | magentaBright | bgMagentaBright |
![]() | cyan | bgCyan | ![]() | cyanBright | bgCyanBright |
![]() | white | bgWhite | ![]() | whiteBright | bgWhiteBright |
ANSI 256 colors
- Foreground:
fg(code)-chalk.ansi256(code)equivalent - Background:
bg(code)-chalk.bgAnsi256(code)equivalent
import { bold, fg, bg } from 'ansis';
fg(96)`Bright Cyan`;
bg(105).fg(96)`Cyan text on magenta background`;
bold.fg(96).underline`Bold underline Bright Cyan`;
256 color codes
See ANSI color codes.
Fallback
If a terminal supports only 16 colors then ANSI 256 colors will be interpolated into base 16 colors.
Truecolor
- Foreground:
hex()rgb() - Background:
bgHex()bgRgb()
import { bold, hex, rgb, bgHex } from 'ansis';
hex('#E0115F').bold`Bold Ruby`;
rgb(224, 17, 95).italic`Italic Ruby`;
bold.hex('#E0115F').bgHex('#96C')`Ruby text on amethyst background`;
See also named truecolors.
Fallback
Ansis automatically interpolates to the best available color level.
Truecolor β 256 colors β 16 colors β no colors (b&w)

β top
Named truecolors
Register any hex color as a named style via extend().
Background methods bg* are generated automatically.
import { Ansis } from 'ansis';
const myTheme = {
orange: '#ffa500',
pink: '#ffc0cb',
};
const color = new Ansis().extend(myTheme);
color.orange.bold`orange bold`; // extended first in chain
color.bgOrange`orange background`; // auto-generated bg tag
color.pink`pink foreground`;
color.bgPink`pink background`; // auto-generated bg tag
color.red`built-in red still works`; // built-in remains intact
Warning
For extended styles, create a new Ansis instance and call extend() on it.
Tip
For all CSS named colors use css-color-names package.

Example: extend with CSS color names
import { Ansis } from 'ansis';
import colorNames from 'css-color-names'; // { pink: '#ffc0cb', orange: '#ffa500', ... }
const color = new Ansis().extend(colorNames);
color.pink('Pink foreground');
color.bgPink('Pink background'); // auto-generated bg
Tip
Need help picking a color name? Try the Name that Color tool - paste a hex and get its closest color name.
Hyperlink
OSC 8 hyperlinks are now supported by many terminal emulators.
Use link(url, text?) to create a terminal hyperlink. If text is omitted, the URL itself is used as the visible label.
link(url, text)- link URL + optional link textlink(url)- URL as both target and text
import { blue, link } from 'ansis';
link('https://example.com'); // URL and text are the same
blue.underline.link('https://example.com', 'Click here');
Important
Call link() last in the chain:
blue.underline.link(...); // β
blue.link(...).underline; // β
Warning
In unsupported terminals text is shown without a link.
β top
Color support
Ansis automatically detects the supported color level:
0β No color1β Basic ANSI (16 colors)2β Extended ANSI (256 colors)3β Truecolor (16 million colors)
Check the detected level at runtime:
import ansis from 'ansis';
console.log(ansis.level); // 0 | 1 | 2 | 3
console.log(ansis.isSupported()); // true -> level > 0 (at least 16 colors supported)
Create an Ansis instance directly when you need to override color detection:
import { Ansis } from 'ansis';
const noColor = new Ansis(0); // always plain text, no ANSI codes
const basicColor = new Ansis(1); // 16 colors; hex/rgb fall back to nearest
const autoColor = new Ansis(); // auto-detect using globalThis
console.log(noColor.red`foo`); // plain text, no ANSI codes
console.log(basicColor.hex('#FFAB40')`Orange`); // falls back to yellowBright
The constructor also accepts a mock globalThis object to control auto-detection in custom runtimes:
const color = new Ansis({
process: {
env: { FORCE_COLOR: '1' }, // COLORTERM, TERM, CI, NO_COLOR, FORCE_COLOR
argv: ['node', 'app.js'], // --no-color, --color
stdout: { isTTY: false },
platform: 'linux',
},
});
console.log(color.level); // 1
Example: disable colors via custom CLI flag, e.g. --save-to-log
import { Ansis } from 'ansis';
/**
* @param {boolean} noColors Disable colors for non-terminal output.
* @return {Ansis} Default or custom instance
*/
function safeAnsis(noColors) {
return noColors
? new Ansis(0) // disable colors
: new Ansis(); // auto/detect color support
}
// handle a special CLI flag to disable colors
const ansis = safeAnsis(process.argv.includes('--save-to-log'))
Auto-detection
Ansis detects color support from the runtime environment in this order:
-
Chromium browser-like runtimes
- detected first -> truecolor
-
COLORTERM(some terminals set it even when output is not TTY)truecoloror24bit-> truecoloransi256-> 256 colorsansi-> 16 colors
-
CI environment (not TTY)
- GitHub Actions -> truecolor
- other CI environments -> 16 colors
-
Terminal
- no TTY -> no colors
TERM=dumb-> no colorsPM2andNext.jsnon-TTY runtimes -> color output
-
Windows
- Windows terminals since Windows 10 build 14931 (released 2016) -> truecolor
-
256-color terminals
- known 256-color terminals -> 256 colors
-
Fallback
- unknown terminals -> 16 colors
Supported terminals and CI environments
| Terminal | ANSI 16 colors | ANSI 256 colors | True Color | env. TERM | env. COLORTERM | Specifically ENV variables |
|---|---|---|---|---|---|---|
| Azure CI | β | β | β | dumb | TF_BUILD AGENT_NAME | |
| GitHub CI | β | β | β | dumb | CI, GITHUB_ACTIONS | |
| GitTea CI | β | β | β | dumb | CI, GITEA_ACTIONS | |
| GitLab CI | β | β | β | dumb | CI, GITLAB_CI | |
| Travis CI | β | β | β | dumb | TRAVIS | |
| PM2 not isTTY | β [^1] | β [^1] | β [^1] | dumb | PM2_HOME pm_id | |
| JetBrains TeamCity >=2020.1.1 | β | β | β | TEAMCITY_VERSION | ||
| JetBrains IDEA | β | β | β | xterm-256color | TERMINAL_EMULATOR='JetBrains-JediTerm' | |
| VS Code | β | β | β | xterm-256color | truecolor | |
| Windows Terminal | β | β | β [^2] | |||
| Windows PowerShell | β | β | β [^2] | |||
| macOS Terminal | β | β | β | xterm-256color | ||
| iTerm | β | β | β | xterm-256color | truecolor | |
| Kitty | β | β | β | xterm-kitty | truecolor | |
| KDE Konsole | β | β | β | xterm-256color | truecolor |
- ^1 Colors supported depends on actual terminal.
- ^2: The Windows terminal supports true color since Windows 10 revision 14931 (2016-09-21).
See also:
Environment & CLI options
Ansis detects color support automatically, but you can override it with environment variables and CLI flags.
Environment variables
NO_COLOR
Set to any non-empty value (1, true) to disable color output (see no-color.org):
NO_COLOR=1 node app.js
FORCE_COLOR
Force or override color support via environment variable (see force-color.org).
| Value | Behavior |
|---|---|
0 or false | Disable colors (level 0) |
1 | Force 16 colors (level 1) |
2 | Force 256 colors (level 2) |
3 | Force truecolor (level 3) |
true or any other string | Auto-detect, with fallback to 16 colors if detection fails |
Important
Ansis differs from the original force-color.org rule in a few cases:
-
FORCE_COLOR=0disables colors.
This matches Node.js, but the original force-color.org rule enables colors when the value is a non-empty string, regardless of the actual value. -
FORCE_COLOR=trueand an empty value enable colors.
This matches Node.js, but the original force-color.org rule ignores an empty string.
COLORTERM
Hint the auto-detected color level using terminal emulator conventions:
| Value | Level |
|---|---|
ansi | 16 colors (level 1) |
ansi256 | 256 colors (level 2) |
truecolor or 24bit | Truecolor (level 3) |
CLI flags
Pass --no-color or --color directly to your script:
./app.js # auto-detect
./app.js --no-color # disable colors
./app.js --color # enable colors (useful when piping output)
Note
If multiple CLI color flags are present, the last one wins:
--color --no-color->--no-color--no-color --color->--color
Quick reference
node app.js # auto-detect
node app.js > log.txt # no colors (non-TTY)
NO_COLOR=1 node app.js # force off
FORCE_COLOR=0 node app.js # force off
FORCE_COLOR=1 node app.js > log.txt # force 16 colors
FORCE_COLOR=2 node app.js > log.txt # force 256 colors
FORCE_COLOR=3 node app.js > log.txt # force truecolor
FORCE_COLOR=true node app.js > log.txt # auto-detect, fallback to 16 colors
node app.js --no-color # disable via flag
node app.js --color > log.txt # auto-detect via flag, fallback to 16 colors
β top
Edge cases
Break style at New Line
Ansis and Chalk add a style break at each new line to correctly display multi-line text.
However, Picocolors doesn't handle this case.
ansis.bgRed('\n ERROR \n') + ansis.cyan('The file not found!') // β
chalk.bgRed('\n ERROR \n') + chalk.cyan('The file not found!') // β
pico.bgRed('\n ERROR \n') + pico.cyan('The file not found!') // β

Nested template strings
Only Ansis handles this very useful use case.
ansis.red`R ${ansis.green`G ${ansis.blue`B`} G`} R` // β
chalk.red`R ${chalk.green`G ${chalk.blue`B`} G`} R` // β
pico.red`R ${pico.green`G ${pico.blue`B`} G`} R` // β

Handling arguments
Compare how different libraries handle various input arguments in their functions.
ansis.red() // β
''
chalk.red() // β
''
pico.red() // β \e[31mundefined\e[39m
ansis.red(undefined) // β
''
chalk.red(undefined) // β \e[31mundefined\e[39m
pico.red(undefined) // β \e[31mundefined\e[39m
ansis.red(null) // β
''
chalk.red(null) // β \e[31mnull\e[39m
pico.red(null) // β \e[31mnull\e[39m
ansis.red('') // β
''
chalk.red('') // β
''
pico.red('') // β \e[31m\e[39m
ansis.reset() // β
\e[0m
chalk.reset() // β ''
pico.reset() // β \e[0mundefined\e[0m
| Library | c.reset() | c.red() | c.red(undefined) | c.red(null) | c.red('') |
|---|---|---|---|---|---|
ansis | β
\e[0m | β
'' | β
'' | β
'' | β
'' |
chalk | β'' | β
'' | β'undefined' | β'null' | β
'' |
picocolors | βundefined | β'undefined' | β'undefined' | β'null' | β'ESC' |
tinyrainbow | βundefined | β'undefined' | β'undefined' | β'null' | β'ESC' |
colorette | β'' | β
'' | β
'' | β'null' | β
'' |
kleur | β[object] | β[object] | β[object] | β'null' | β'ESC' |
ansi-colors | β'' | β
'' | β
'' | β
'' | β
'' |
kolorist | βundefined | β'undefined' | β'undefined' | β'null' | β'ESC' |
colors.js | β'' | β
'' | β'undefined' | β'null' | β
'' |
cli-color | β- | β'ESC' | β'ESC' | β'ESC' | β'ESC' |
colors-cli | β- | β Error | β'undefined' | β'null' | β'ESC' |
Legend:
- β
''- Returns an empty string without ANSI escape codes. This is the correct and expected behavior. - β
\e[0m- Returns the reset escape code. - β
'ESC'- Returns an empty string containing ANSI escape codes, e.g.,\e[31m\e[39m. - β
'undefined'- Returns the styled stringundefined. - β
'null'- Returns the styled stringnull. - β
[object]- Returns an object of the library instance. - β
-- The feature is not supported. - β
Error- Causes a fatal error.
Other arguments are correctly handled by all libraries:
c.red(0) // '0' in red
c.red(false) // 'false' in red
c.red(true) // 'true' in red
c.red(5/'1px') // 'NaN' in red
c.red(1/0) // 'Infinity' in red
β top
Ansis vs styleText()
Since Node v22, the built-in util.styleText()
has been officially introduced, supporting standard modifiers - the basic 16 colors and styles.
Where it works
Ansis
β
Node v10+
β
Chromium-based browsers and Safari
β οΈ Firefox DevTools don't render ANSI escape sequences
styleText
β
Node v22+ (native)
β No browser support (Node only)
Performance
In practical benchmarks, styleText() is 100x slower than Ansis:
ansis.red('text'); // 59.646.465 ops/sec
styleText('red', 'text'); // 579.832 ops/sec
See full benchmarks.
Color support detection
Ansis
- Auto-detects terminal, TTY, CI and browser color support with automatic fallback
- Supports
NO_COLORFORCE_COLORCOLORTERM--no-color--color ansis.levelreturns the detected color level
styleText
- Auto-detects terminal color support
- Supports only
NO_COLORFORCE_COLORNODE_DISABLE_COLORS
Simple styling
import { green } from 'ansis';
import { styleText } from 'node:util';
green`Success!`; // ansis
styleText('green', 'Success!');
green.bold`Success!`; // ansis
styleText(['green', 'bold'], 'Success!');
Nested styling
import { red, cyan } from 'ansis';
import { styleText } from 'node:util';
red`Error: ${cyan.bold`file.js`} not found!`; // ansis
styleText('red', `Error: ${styleText(['cyan', 'bold'], 'file.js')} not found!`);
Truecolor
Ansis: hex() rgb()
styleText: Limited to 16 ANSI colors.
β top
Compatibility
Check the minimum version of your tool required for compatibility with the latest Ansis.
| Tool | Version | Compatibility | Supports |
|---|---|---|---|
| Node.js | v14+ | β Full support | CJS, ESM |
| Deno | v2.0+ | β Full support | CJS, ESM |
| TypeScript/tsc | v5.0+ | β Full support | CJS, ESM |
| esbuild | v0.8+ | β Full support | CJS, ESM |
| swc | v1.2+ | β Full support | CJS, ESM, FAUX |
| tsup | v4.0+ | β Full support | CJS, ESM, FAUX |
| tsx | v3.0+ | β Full support | CJS, ESM |
| Rollup | v2.0+ | β Full support | CJS, ESM |
| Rolldown | v1.0.0-beta.8+ | β Full support | CJS, ESM |
| Vite | v2.5+ | β Full support | ESM |
| Turbo | v1.0+ | β Full support | CJS, ESM |
| Webpack | v5.0+ | β Full support | CJS, ESM |
Supports:
- CJS: CommonJS module support.
- ESM: ECMAScript module support.
- FAUX: Fake or non-standard approach to module resolution (seen in swc).
Browser Compatibility for ANSI Codes
| Browser | Version | Colors Supported |
|---|---|---|
| Chrome | v20+ | TrueColor (16M) |
| Safari | v10+ | TrueColor (16M) |
| Edge | v12+ | TrueColor (16M) |
| Opera | v12+ | TrueColor (16M) |
| Brave | v1.0+ | TrueColor (16M) |
| Vivaldi | v1.0+ | TrueColor (16M) |
Warning
Firefox doesn't natively support ANSI codes in the developer console.
βοΈ Star History
If you find this useful, please βοΈ the repo.















