123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139 |
- 'use strict';
- const stringWidth = require('string-width');
- const chalk = require('chalk');
- const widestLine = require('widest-line');
- const cliBoxes = require('cli-boxes');
- const camelCase = require('camelcase');
- const ansiAlign = require('ansi-align');
- const termSize = require('term-size');
-
- const getObject = detail => {
- let object;
-
- if (typeof detail === 'number') {
- object = {
- top: detail,
- right: detail * 3,
- bottom: detail,
- left: detail * 3
- };
- } else {
- object = {
- top: 0,
- right: 0,
- bottom: 0,
- left: 0,
- ...detail
- };
- }
-
- return object;
- };
-
- const getBorderChars = borderStyle => {
- const sides = [
- 'topLeft',
- 'topRight',
- 'bottomRight',
- 'bottomLeft',
- 'vertical',
- 'horizontal'
- ];
-
- let chararacters;
-
- if (typeof borderStyle === 'string') {
- chararacters = cliBoxes[borderStyle];
-
- if (!chararacters) {
- throw new TypeError(`Invalid border style: ${borderStyle}`);
- }
- } else {
- for (const side of sides) {
- if (!borderStyle[side] || typeof borderStyle[side] !== 'string') {
- throw new TypeError(`Invalid border style: ${side}`);
- }
- }
-
- chararacters = borderStyle;
- }
-
- return chararacters;
- };
-
- const isHex = color => color.match(/^#[0-f]{3}(?:[0-f]{3})?$/i);
- const isColorValid = color => typeof color === 'string' && ((chalk[color]) || isHex(color));
- const getColorFn = color => isHex(color) ? chalk.hex(color) : chalk[color];
- const getBGColorFn = color => isHex(color) ? chalk.bgHex(color) : chalk[camelCase(['bg', color])];
-
- module.exports = (text, options) => {
- options = {
- padding: 0,
- borderStyle: 'single',
- dimBorder: false,
- align: 'left',
- float: 'left',
- ...options
- };
-
- if (options.borderColor && !isColorValid(options.borderColor)) {
- throw new Error(`${options.borderColor} is not a valid borderColor`);
- }
-
- if (options.backgroundColor && !isColorValid(options.backgroundColor)) {
- throw new Error(`${options.backgroundColor} is not a valid backgroundColor`);
- }
-
- const chars = getBorderChars(options.borderStyle);
- const padding = getObject(options.padding);
- const margin = getObject(options.margin);
-
- const colorizeBorder = border => {
- const newBorder = options.borderColor ? getColorFn(options.borderColor)(border) : border;
- return options.dimBorder ? chalk.dim(newBorder) : newBorder;
- };
-
- const colorizeContent = content => options.backgroundColor ? getBGColorFn(options.backgroundColor)(content) : content;
-
- text = ansiAlign(text, {align: options.align});
-
- const NL = '\n';
- const PAD = ' ';
-
- let lines = text.split(NL);
-
- if (padding.top > 0) {
- lines = new Array(padding.top).fill('').concat(lines);
- }
-
- if (padding.bottom > 0) {
- lines = lines.concat(new Array(padding.bottom).fill(''));
- }
-
- const contentWidth = widestLine(text) + padding.left + padding.right;
- const paddingLeft = PAD.repeat(padding.left);
- const {columns} = termSize();
- let marginLeft = PAD.repeat(margin.left);
-
- if (options.float === 'center') {
- const padWidth = Math.max((columns - contentWidth) / 2, 0);
- marginLeft = PAD.repeat(padWidth);
- } else if (options.float === 'right') {
- const padWidth = Math.max(columns - contentWidth - margin.right - 2, 0);
- marginLeft = PAD.repeat(padWidth);
- }
-
- const horizontal = chars.horizontal.repeat(contentWidth);
- const top = colorizeBorder(NL.repeat(margin.top) + marginLeft + chars.topLeft + horizontal + chars.topRight);
- const bottom = colorizeBorder(marginLeft + chars.bottomLeft + horizontal + chars.bottomRight + NL.repeat(margin.bottom));
- const side = colorizeBorder(chars.vertical);
-
- const middle = lines.map(line => {
- const paddingRight = PAD.repeat(contentWidth - stringWidth(line) - padding.left);
- return marginLeft + side + colorizeContent(paddingLeft + line + paddingRight) + side;
- }).join(NL);
-
- return top + NL + middle + NL + bottom;
- };
-
- module.exports._borderStyles = cliBoxes;
|