Skip to content

Commit

Permalink
chore: optimize code to reduce the size and improve performance
Browse files Browse the repository at this point in the history
  • Loading branch information
webdiscus committed Jul 18, 2024
1 parent d700050 commit 0560948
Show file tree
Hide file tree
Showing 11 changed files with 124 additions and 125 deletions.
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,11 @@
# Change log

## 3.3.1 (2024-07-18)

- chore: optimize code to reduce the size by ~600 bytes,
- chore: minify `index.d.ts` to reduce the size by ~200 bytes,
- chore: increase performance, e.g. using chained styles, 70.000.000 -> 80.000.000 ops/sec

## 3.3.0 (2024-07-14)

- feat(BREAKING CHANGE): remove `old` named import DEPRECATED in `v2.0.0` (2023-11-03).
Expand Down
60 changes: 30 additions & 30 deletions bench/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,8 @@ let fixture;
log(hex('#F88').inverse.bold` -= Benchmark =- `);

bench('Using 1 style (red)').
add(packages['chalk'], () => chalk.red('foo')).
add(packages['ansis'], () => ansis.red('foo')).
add(packages['colors'], () => colorsJs.red('foo')).
add(packages['colorette'], () => colorette.red('foo')).
add(packages['picocolors'], () => picocolors.red('foo')).
Expand All @@ -99,11 +101,11 @@ bench('Using 1 style (red)').
add(packages['ansi-colors'], () => ansiColors.red('foo')).
add(packages['kleur'], () => kleur.red('foo')).
add(packages['kolorist'], () => kolorist.red('foo')).
add(packages['chalk'], () => chalk.red('foo')).
add(packages['ansis'], () => ansis.red('foo')).
run();

bench(`Using 2 styles (red, bold)`).
add(packages['chalk'], () => chalk.red.bold('foo')).
add(packages['ansis'], () => ansis.red.bold('foo')).
add(packages['colors'], () => colorsJs.red.bold('foo')).
add(packages['colorette'], () => colorette.red(colorette.bold('foo'))).
add(packages['picocolors'], () => picocolors.red(picocolors.bold('foo'))).
Expand All @@ -112,11 +114,11 @@ bench(`Using 2 styles (red, bold)`).
add(packages['ansi-colors'], () => ansiColors.red.bold('foo')).
add(packages['kleur'], () => kleur.red().bold('foo')).
add(packages['kolorist'], () => kolorist.red(kolorist.bold('foo'))).
add(packages['chalk'], () => chalk.red.bold('foo')).
add(packages['ansis'], () => ansis.red.bold('foo')).
run();

bench(`Using 3 styles (red, bold, underline)`).
add(packages['chalk'], () => chalk.red.bold.underline('foo')).
add(packages['ansis'], () => ansis.red.bold.underline('foo')).
add(packages['colors'], () => colorsJs.red.bold.underline('foo')).
add(packages['colorette'], () => colorette.red(colorette.bold(colorette.underline('foo')))).
add(packages['picocolors'], () => picocolors.red(picocolors.bold(picocolors.underline('foo')))).
Expand All @@ -125,27 +127,27 @@ bench(`Using 3 styles (red, bold, underline)`).
add(packages['ansi-colors'], () => ansiColors.red.bold.underline('foo')).
add(packages['kleur'], () => kleur.red().bold().underline('foo')).
add(packages['kolorist'], () => kolorist.red(kolorist.bold(kolorist.underline('foo')))).
add(packages['chalk'], () => chalk.red.bold.underline('foo')).
add(packages['ansis'], () => ansis.red.bold.underline('foo')).
run();

bench(`Using 5 styles (bgWhite red, bold, italic, underline)`).
add(packages['colors'], () => colorsJs.bgWhite.red.bold.italic.underline('foo')).
add(packages['colorette'], () => colorette.bgWhite(colorette.red(colorette.bold(colorette.italic(colorette.underline('foo')))))).
add(packages['picocolors'], () => picocolors.bgWhite(picocolors.red(picocolors.bold(picocolors.italic(picocolors.underline('foo')))))).
add(packages['cli-color'], () => cliColor.bgWhite.red.bold.italic.underline('foo')).
add(packages['colors-cli'], () => colorCli.white_b.red.bold.italic.underline('foo')).
add(packages['ansi-colors'], () => ansiColors.bgWhite.red.bold.italic.underline('foo')).
add(packages['kleur'], () => kleur.bgWhite().red().bold().italic().underline()('foo')).
add(packages['kolorist'], () => kolorist.bgWhite(kolorist.red(kolorist.bold(kolorist.italic(kolorist.underline('foo')))))).
add(packages['chalk'], () => chalk.bgWhite.red.bold.italic.underline('foo')).
add(packages['ansis'], () => ansis.bgWhite.red.bold.italic.underline('foo')).
bench(`Using 4 styles (bgWhite red, bold, underline)`).
add(packages['chalk'], () => chalk.bgWhite.red.bold.underline('foo')).
add(packages['ansis'], () => ansis.bgWhite.red.bold.underline('foo')).
add(packages['colors'], () => colorsJs.bgWhite.red.bold.underline('foo')).
add(packages['colorette'], () => colorette.bgWhite(colorette.red(colorette.bold(colorette.underline('foo'))))).
add(packages['picocolors'], () => picocolors.bgWhite(picocolors.red(picocolors.bold(picocolors.underline('foo'))))).
add(packages['cli-color'], () => cliColor.bgWhite.red.bold.underline('foo')).
add(packages['colors-cli'], () => colorCli.white_b.red.bold.underline('foo')).
add(packages['ansi-colors'], () => ansiColors.bgWhite.red.bold.underline('foo')).
add(packages['kleur'], () => kleur.bgWhite().red().bold().underline()('foo')).
add(packages['kolorist'], () => kolorist.bgWhite(kolorist.red(kolorist.bold(kolorist.underline('foo'))))).
run();

// Colorette bench
// https://github.com/jorgebucaran/colorette/blob/main/bench/index.js
fixture = createFixture(vendors, coloretteBench);
bench('Colorette bench').
add(vendors[8].name, () => fixture[8](vendors[8].lib)).
add(vendors[9].name, () => fixture[9](vendors[9].lib)).
add(vendors[0].name, () => fixture[0](vendors[0].lib)).
add(vendors[1].name, () => fixture[1](vendors[1].lib)).
add(vendors[2].name, () => fixture[2](vendors[2].lib)).
Expand All @@ -154,12 +156,12 @@ bench('Colorette bench').
add(vendors[5].name, () => fixture[5](vendors[5].lib)).
add(vendors[6].name, () => fixture[6](vendors[6].lib)).
add(vendors[7].name, () => fixture[7](vendors[7].lib)).
add(vendors[8].name, () => fixture[8](vendors[8].lib)).
add(vendors[9].name, () => fixture[9](vendors[9].lib)).
run();

// Base colors
bench('Base colors').
add(packages['chalk'], () => baseColors.forEach((style) => chalk[style]('foo'))).
add(packages['ansis'], () => baseColors.forEach((style) => ansis[style]('foo'))).
add(packages['colors'], () => baseColors.forEach((style) => colorsJs[style]('foo'))).
add(packages['colorette'], () => baseColors.forEach((style) => colorette[style]('foo'))).
add(packages['picocolors'], () => baseColors.forEach((style) => picocolors[style]('foo'))).
Expand All @@ -168,12 +170,12 @@ bench('Base colors').
add(packages['ansi-colors'], () => baseColors.forEach((style) => ansiColors[style]('foo'))).
add('kleur/colors', () => baseColors.forEach((style) => kleurColors[style]('foo'))).
add(packages['kleur'], () => baseColors.forEach((style) => kleur[style]('foo'))).
add(packages['chalk'], () => baseColors.forEach((style) => chalk[style]('foo'))).
add(packages['ansis'], () => baseColors.forEach((style) => ansis[style]('foo'))).
run();

// Chained styles
bench('Chained styles').
add(packages['chalk'], () => baseColors.forEach((style) => chalk[style].bold.underline.italic('foo'))).
add(packages['ansis'], () => baseColors.forEach((style) => ansis[style].bold.underline.italic('foo'))).
add(packages['colors'], () => baseColors.forEach((style) => colorsJs[style].bold.underline.italic('foo'))).
add('colorette (not supported)', () => baseColors.forEach((style) => colorette[style].bold.underline.italic('foo'))).
add('picocolors (not supported)', () =>
Expand All @@ -185,13 +187,13 @@ bench('Chained styles').
add('kleur/colors (not supported)', () =>
baseColors.forEach((style) => kleurColors[style].bold.underline.italic('foo')),
).
// add(packages['kleur'], () => baseColors.forEach((style) => kleur[style]().bold().underline().italic('foo'))). // alternate syntax
add(packages['chalk'], () => baseColors.forEach((style) => chalk[style].bold.underline.italic('foo'))).
add(packages['ansis'], () => baseColors.forEach((style) => ansis[style].bold.underline.italic('foo'))).
add(packages['kleur'], () => baseColors.forEach((style) => kleur[style]().bold().underline().italic('foo'))). // alternate syntax
run();

// Nested calls
bench('Nested calls').
add(packages['chalk'], () => baseColors.forEach((style) => chalk[style](chalk.bold(chalk.underline(chalk.italic('foo')))))).
add(packages['ansis'], () => baseColors.forEach((style) => ansis[style](ansis.bold(ansis.underline(ansis.italic('foo')))))).
add(packages['colors'], () =>
baseColors.forEach((style) => colorsJs[style](colorsJs.bold(colorsJs.underline(colorsJs.italic('foo'))))),
).
Expand All @@ -216,13 +218,13 @@ bench('Nested calls').
),
).
add(packages['kleur'], () => baseColors.forEach((style) => kleur[style](kleur.bold(kleur.underline(kleur.italic('foo')))))).
add(packages['chalk'], () => baseColors.forEach((style) => chalk[style](chalk.bold(chalk.underline(chalk.italic('foo')))))).
add(packages['ansis'], () => baseColors.forEach((style) => ansis[style](ansis.bold(ansis.underline(ansis.italic('foo')))))).
run();

// Nested styles
fixture = createFixture(vendors, nestedFixture);
bench('Nested styles').
add(packages['chalk'], () => fixture[7](chalk)).
add(packages['ansis'], () => fixture[8](ansis)).
add(packages['colors'], () => fixture[9](colorsJs)).
add(packages['colorette'], () => fixture[0](colorette)).
add(packages['picocolors'], () => fixture[1](picocolors)).
Expand All @@ -231,13 +233,13 @@ bench('Nested styles').
add(packages['ansi-colors'], () => fixture[4](ansiColors)).
add('kleur/colors', () => fixture[5](kleurColors)).
add(packages['kleur'], () => fixture[6](kleur)).
add(packages['chalk'], () => fixture[7](chalk)).
add(packages['ansis'], () => fixture[8](ansis)).
run();

// Deep nested styles
fixture = createFixture(vendors, deepNestedFixture);
bench('Deep nested styles').
add(packages['chalk'], () => fixture[7](chalk)).
add(packages['ansis'], () => fixture[8](ansis)).
add(packages['colors'], () => fixture[9](colorsJs)).
add(packages['colorette'], () => fixture[0](colorette)).
add(packages['picocolors'], () => fixture[1](picocolors)).
Expand All @@ -246,8 +248,6 @@ bench('Deep nested styles').
add(packages['ansi-colors'], () => fixture[4](ansiColors)).
add('kleur/colors', () => fixture[5](kleurColors)).
add(packages['kleur'], () => fixture[6](kleur)).
add(packages['chalk'], () => fixture[7](chalk)).
add(packages['ansis'], () => fixture[8](ansis)).
run();

// Check support of correct break style at new line
Expand Down
Binary file added docs/img/screenshot-readme-npm.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "ansis",
"version": "3.3.0",
"version": "3.3.1",
"description": "Colorize terminal output with ANSI colors & styles",
"keywords": [
"ansi",
Expand Down
2 changes: 1 addition & 1 deletion package.npm.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "ansis",
"version": "3.3.0",
"version": "3.3.1",
"description": "Colorize terminal output with ANSI colors & styles",
"keywords": [
"ansi",
Expand Down
39 changes: 24 additions & 15 deletions rollup.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,21 @@ import { minify } from 'terser';
const ecma = 2019;

export default [
// remove comments from d.ts file
{
input: 'src/index.d.ts',
output: [
{
file: 'dist/index.d.ts',
format: 'es',
},
],
plugins: [
cleanup({ extensions: ['ts'] }),
dts(),
],
},

{
input: 'src/index.js',
output: [
Expand Down Expand Up @@ -39,27 +54,21 @@ export default [
{
src: 'src/index.mjs',
dest: 'dist/',
transform: async (contents, name) => (await minify(contents.toString())).code,
transform: async (contents, name) => (await minify(contents.toString(), { ecma: 2015 })).code,
},

// minify d.ts file generated after cleanup
{
src: 'dist/index.d.ts',
dest: 'dist/',
transform: (contents, name) => { return contents.toString().replaceAll(/\n/g, '');},
},

{ src: 'package.npm.json', dest: 'dist/', rename: 'package.json' },
{ src: 'README.npm.md', dest: 'dist/', rename: 'README.md' },
{ src: 'LICENSE', dest: 'dist/' },
],
}),
],
},

{
input: 'src/index.d.ts',
output: [
{
file: 'dist/index.d.ts',
format: 'es',
},
],
plugins: [
cleanup({ extensions: ['ts'] }),
dts(),
],
},
];
9 changes: 0 additions & 9 deletions src/ansi-codes.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,10 +9,6 @@ const esc = isSupported ? (open, close) => ({ open: `\x1b[${open}m`, close: `\x1
const closeCode = 39;
const bgCloseCode = 49;

const ESC = '\x1b';
const BEL = '\x07';
const ZWSP = '\u200B';

// defaults, true color
let fnAnsi256 = (code) => esc(`38;5;${code}`, closeCode);
let fnBgAnsi256 = (code) => esc(`48;5;${code}`, bgCloseCode);
Expand Down Expand Up @@ -122,11 +118,6 @@ export const styleMethods = {
// note: the `...` operator is too slow
//bgHex: (hex) => fnBgRgb(...hexToRgb(hex)),
bgHex: createHexFn(fnBgRgb),

// reserved for future: hyperlink (OSC 8) is not widely supported (works in iTerm)
// link: hasColor
// ? (url) => ({ open: ESC + ']8;;' + url + BEL, close: ESC + ']8;;' + BEL })
// : (url) => ({ open: '', close: `(${ZWSP}${url}${ZWSP})` }),
};

export const rgb = fnRgb;
2 changes: 1 addition & 1 deletion src/index.d.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
type ColorExtend = Record<string, string | { open: string, close: string }>
type ColorExtend = Record<string, string | { open: string, close: string }>;

interface Ansis {
/**
Expand Down
87 changes: 54 additions & 33 deletions src/index.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { baseStyles, styleMethods, rgb, isSupported } from './ansi-codes.js';
import { hexToRgb, replaceAll } from './utils.js';
import { hexToRgb } from './utils.js';

/**
* @typedef {Object} AnsisProps
Expand All @@ -17,36 +17,6 @@ const ESC = '\x1b';
const LF = '\x0a';
const styles = {};

/**
* Wrap the string with styling and reset codes.
*
* @param {string | Array<String> | number} strings A string or template literals.
* @param {Array<String>} values The values of the template literals.
* @param {AnsisProps} props
* @returns {string}
*/
const wrap = (strings, values, props) => {
if (!strings) return '';

const { _a: openStack, _b: closeStack } = props;
// convert the number to the string
let string = strings.raw != null ? String.raw(strings, ...values) : strings + '';

if (string.includes(ESC)) {
while (props != null) {
string = replaceAll(string, props.close, props.open); // much faster than native replaceAll
//string = string.replaceAll(props.close, props.open); // too slow!
props = props._p;
}
}

if (string.includes(LF)) {
string = string.replace(regexLFCR, closeStack + '$1' + openStack);
}

return openStack + string + closeStack;
};

/**
* @param {Object} self
* @param {AnsisProps} self._p
Expand All @@ -56,7 +26,58 @@ const wrap = (strings, values, props) => {
* @returns {Ansis}
*/
const createStyle = ({ _p: props }, { open, close }) => {
const style = (strings, ...values) => wrap(strings, values, style._p);
/**
* Wrap the string with ANSI codes.
* @param {string} strings The normal or template string.
* @param {array} values The values of the template string.
* @return {string}
*/
const style = (strings, ...values) => {
if (!strings) return '';

let props = style._p;
let { _a: openStack, _b: closeStack } = props;

let str = strings.raw != null
// render template strings
? String.raw(strings, ...values)
// convert the number to the string
: '' + strings;

// on node.js, the performance of `includes()` and `~indexOf()` is the same, no difference
if (str.includes(ESC)) {
while (props != null) {
// this implementation is over 30% faster than String.replaceAll()
// -- begin replaceAll
let search = props.close;
let searchLength = search.length;

// the `visible` style has empty open/close props
if (searchLength) {
let lastPos = 0;
let result = '';
let pos;

while (~(pos = str.indexOf(search, lastPos))) {
result += str.slice(lastPos, pos) + props.open;
lastPos = pos + searchLength;
}

if (lastPos) str = result + str.slice(lastPos);
}
// -- end replaceAll

props = props._p;
}
}

if (str.includes(LF)) {
str = str.replace(regexLFCR, closeStack + '$1' + openStack);
}

return openStack + str + closeStack;
};

let openStack = open;
let closeStack = close;

Expand All @@ -74,7 +95,7 @@ const createStyle = ({ _p: props }, { open, close }) => {
};

const Ansis = function() {
const self = (str) => str + '';
const self = (str) => '' + str;

/**
* Whether the output supports ANSI color and styles.
Expand Down
Loading

0 comments on commit 0560948

Please sign in to comment.