123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268 |
- 'use strict';
- const toString = Object.prototype.toString;
- const colors = require('ansi-colors');
- let called = false;
- let fns = [];
- const complements = {
- 'yellow': 'blue',
- 'cyan': 'red',
- 'green': 'magenta',
- 'black': 'white',
- 'blue': 'yellow',
- 'red': 'cyan',
- 'magenta': 'green',
- 'white': 'black'
- };
- exports.longest = (arr, prop) => {
- return arr.reduce((a, v) => Math.max(a, prop ? v[prop].length : v.length), 0);
- };
- exports.hasColor = str => !!str && colors.hasColor(str);
- const isObject = exports.isObject = val => {
- return val !== null && typeof val === 'object' && !Array.isArray(val);
- };
- exports.nativeType = val => {
- return toString.call(val).slice(8, -1).toLowerCase().replace(/\s/g, '');
- };
- exports.isAsyncFn = val => {
- return exports.nativeType(val) === 'asyncfunction';
- };
- exports.isPrimitive = val => {
- return val != null && typeof val !== 'object' && typeof val !== 'function';
- };
- exports.resolve = (context, value, ...rest) => {
- if (typeof value === 'function') {
- return value.call(context, ...rest);
- }
- return value;
- };
- exports.scrollDown = (choices = []) => [...choices.slice(1), choices[0]];
- exports.scrollUp = (choices = []) => [choices.pop(), ...choices];
- exports.reorder = (arr = []) => {
- let res = arr.slice();
- res.sort((a, b) => {
- if (a.index > b.index) return 1;
- if (a.index < b.index) return -1;
- return 0;
- });
- return res;
- };
- exports.swap = (arr, index, pos) => {
- let len = arr.length;
- let idx = pos === len ? 0 : pos < 0 ? len - 1 : pos;
- let choice = arr[index];
- arr[index] = arr[idx];
- arr[idx] = choice;
- };
- exports.width = (stream, fallback = 80) => {
- let columns = (stream && stream.columns) ? stream.columns : fallback;
- if (stream && typeof stream.getWindowSize === 'function') {
- columns = stream.getWindowSize()[0];
- }
- if (process.platform === 'win32') {
- return columns - 1;
- }
- return columns;
- };
- exports.height = (stream, fallback = 20) => {
- let rows = (stream && stream.rows) ? stream.rows : fallback;
- if (stream && typeof stream.getWindowSize === 'function') {
- rows = stream.getWindowSize()[1];
- }
- return rows;
- };
- exports.wordWrap = (str, options = {}) => {
- if (!str) return str;
- if (typeof options === 'number') {
- options = { width: options };
- }
- let { indent = '', newline = ('\n' + indent), width = 80 } = options;
- let spaces = (newline + indent).match(/[^\S\n]/g) || [];
- width -= spaces.length;
- let source = `.{1,${width}}([\\s\\u200B]+|$)|[^\\s\\u200B]+?([\\s\\u200B]+|$)`;
- let output = str.trim();
- let regex = new RegExp(source, 'g');
- let lines = output.match(regex) || [];
- lines = lines.map(line => line.replace(/\n$/, ''));
- if (options.padEnd) lines = lines.map(line => line.padEnd(width, ' '));
- if (options.padStart) lines = lines.map(line => line.padStart(width, ' '));
- return indent + lines.join(newline);
- };
- exports.unmute = color => {
- let name = color.stack.find(n => colors.keys.color.includes(n));
- if (name) {
- return colors[name];
- }
- let bg = color.stack.find(n => n.slice(2) === 'bg');
- if (bg) {
- return colors[name.slice(2)];
- }
- return str => str;
- };
- exports.pascal = str => str ? str[0].toUpperCase() + str.slice(1) : '';
- exports.inverse = color => {
- if (!color || !color.stack) return color;
- let name = color.stack.find(n => colors.keys.color.includes(n));
- if (name) {
- let col = colors['bg' + exports.pascal(name)];
- return col ? col.black : color;
- }
- let bg = color.stack.find(n => n.slice(0, 2) === 'bg');
- if (bg) {
- return colors[bg.slice(2).toLowerCase()] || color;
- }
- return colors.none;
- };
- exports.complement = color => {
- if (!color || !color.stack) return color;
- let name = color.stack.find(n => colors.keys.color.includes(n));
- let bg = color.stack.find(n => n.slice(0, 2) === 'bg');
- if (name && !bg) {
- return colors[complements[name] || name];
- }
- if (bg) {
- let lower = bg.slice(2).toLowerCase();
- let comp = complements[lower];
- if (!comp) return color;
- return colors['bg' + exports.pascal(comp)] || color;
- }
- return colors.none;
- };
- exports.meridiem = date => {
- let hours = date.getHours();
- let minutes = date.getMinutes();
- let ampm = hours >= 12 ? 'pm' : 'am';
- hours = hours % 12;
- let hrs = hours === 0 ? 12 : hours;
- let min = minutes < 10 ? '0' + minutes : minutes;
- return hrs + ':' + min + ' ' + ampm;
- };
- /**
- * Set a value on the given object.
- * @param {Object} obj
- * @param {String} prop
- * @param {any} value
- */
- exports.set = (obj = {}, prop = '', val) => {
- return prop.split('.').reduce((acc, k, i, arr) => {
- let value = arr.length - 1 > i ? (acc[k] || {}) : val;
- if (!exports.isObject(value) && i < arr.length - 1) value = {};
- return (acc[k] = value);
- }, obj);
- };
- /**
- * Get a value from the given object.
- * @param {Object} obj
- * @param {String} prop
- */
- exports.get = (obj = {}, prop = '', fallback) => {
- let value = obj[prop] == null
- ? prop.split('.').reduce((acc, k) => acc && acc[k], obj)
- : obj[prop];
- return value == null ? fallback : value;
- };
- exports.mixin = (target, b) => {
- if (!isObject(target)) return b;
- if (!isObject(b)) return target;
- for (let key of Object.keys(b)) {
- let desc = Object.getOwnPropertyDescriptor(b, key);
- if (desc.hasOwnProperty('value')) {
- if (target.hasOwnProperty(key) && isObject(desc.value)) {
- let existing = Object.getOwnPropertyDescriptor(target, key);
- if (isObject(existing.value)) {
- target[key] = exports.merge({}, target[key], b[key]);
- } else {
- Reflect.defineProperty(target, key, desc);
- }
- } else {
- Reflect.defineProperty(target, key, desc);
- }
- } else {
- Reflect.defineProperty(target, key, desc);
- }
- }
- return target;
- };
- exports.merge = (...args) => {
- let target = {};
- for (let ele of args) exports.mixin(target, ele);
- return target;
- };
- exports.mixinEmitter = (obj, emitter) => {
- let proto = emitter.constructor.prototype;
- for (let key of Object.keys(proto)) {
- let val = proto[key];
- if (typeof val === 'function') {
- exports.define(obj, key, val.bind(emitter));
- } else {
- exports.define(obj, key, val);
- }
- }
- };
- exports.onExit = callback => {
- const onExit = (quit, code) => {
- if (called) return;
- called = true;
- fns.forEach(fn => fn());
- if (quit === true) {
- process.exit(128 + code);
- }
- };
- if (fns.length === 0) {
- process.once('SIGTERM', onExit.bind(null, true, 15));
- process.once('SIGINT', onExit.bind(null, true, 2));
- process.once('exit', onExit);
- }
- fns.push(callback);
- };
- exports.define = (obj, key, value) => {
- Reflect.defineProperty(obj, key, { value });
- };
- exports.defineExport = (obj, key, fn) => {
- let custom;
- Reflect.defineProperty(obj, key, {
- enumerable: true,
- configurable: true,
- set(val) {
- custom = val;
- },
- get() {
- return custom ? custom() : fn();
- }
- });
- };
|