* * * Licensed under MIT license. */ namespace Ahc\Cli\Output; use Ahc\Cli\Exception\InvalidArgumentException; use Ahc\Cli\Helper\InflectsString; use function Ahc\Cli\t; use function array_column; use function array_fill_keys; use function array_keys; use function array_map; use function array_merge; use function gettype; use function implode; use function is_array; use function max; use function reset; use function str_repeat; use function trim; use const PHP_EOL; class Table { use InflectsString; public function render(array $rows, array $styles = []): string { if ([] === $table = $this->normalize($rows)) { return ''; } [$head, $rows] = $table; $styles = $this->normalizeStyles($styles); $title = $body = $dash = $positions = []; [$start, $end] = $styles['head']; $pos = 0; foreach ($head as $col => $size) { $dash[] = str_repeat('-', $size + 2); $title[] = $this->strPad($this->toWords($col), $size, ' '); $positions[$col] = ++$pos; } $title = "|$start " . implode(" $end|$start ", $title) . " $end|" . PHP_EOL; $odd = true; foreach ($rows as $line => $row) { $parts = []; $line++; foreach ($head as $col => $size) { $colNumber = $positions[$col]; if (isset($styles[$line . ':' . $colNumber])) { // cell, 1:1 $style = $styles[$line . ':' . $colNumber]; } elseif (isset($styles[$col]) || isset($styles['*:' . $colNumber])) { // col, *:2 or b $style = $styles['*:' . $colNumber] ?? $styles[$col]; } elseif (isset($styles[$line . ':*'])) { // row, 2:* $style = $styles[$line . ':*']; } elseif (isset($styles['*:*'])) { // any cell, *:* $style = $styles['*:*']; } else { $style = $styles[['even', 'odd'][(int) $odd]]; } $text = $row[$col] ?? ''; [$start, $end] = $this->parseStyle($style, $text, $row, $rows); if (preg_match('/(\\x1b(?:.+)m)/U', $text, $matches)) { $word = str_replace($matches[1], '', $text); $word = preg_replace('/\\x1b\[0m/', '', $word); $size += $this->strwidth($text) - $this->strwidth($word); } $parts[] = "$start " . $this->strPad($text, $size, ' ') . " $end"; } $odd = !$odd; $body[] = '|' . implode('|', $parts) . '|'; } $dash = '+' . implode('+', $dash) . '+' . PHP_EOL; $body = implode(PHP_EOL, $body) . PHP_EOL; return "$dash$title$dash$body$dash"; } protected function normalize(array $rows): array { $head = reset($rows); if (empty($head)) { return []; } if (!is_array($head)) { throw new InvalidArgumentException( t('Rows must be array of assoc arrays, %s given', [gettype($head)]) ); } $head = array_fill_keys(array_keys($head), null); foreach ($rows as $i => &$row) { $row = array_merge($head, $row); } foreach ($head as $col => &$value) { $cols = array_column($rows, $col); $cols = array_map(function ($col) { $col ??= ''; if (preg_match('/(\\x1b(?:.+)m)/U', $col, $matches)) { $col = str_replace($matches[1], '', $col); $col = preg_replace('/\\x1b\[0m/', '', $col); } return $col; }, $cols); $span = array_map([$this, 'strwidth'], $cols); $span[] = $this->strwidth($col); $value = max($span); } return [$head, $rows]; } protected function normalizeStyles(array $styles): array { $default = [ // styleFor => ['styleStartFn', 'end'] 'head' => ['', ''], 'odd' => ['', ''], 'even' => ['', ''], ]; foreach ($styles as $for => $style) { if (is_string($style) && $style !== '') { $default[$for] = ['<' . trim($style, '<> ') . '>', '']; } elseif (str_contains($for, ':') && is_callable($style)) { $default[$for] = $style; } } return $default; } protected function parseStyle(array|callable $style, $val, array $row, array $table): array { if (is_array($style)) { return $style; } $style = call_user_func($style, $val, $row, $table); if (is_string($style) && $style !== '') { return ['<' . trim($style, '<> ') . '>', '']; } if (is_array($style) && count($style) === 2) { return $style; } return ['', '']; } /** * Pad a multibyte string to a certain length with another multibyte string. */ protected function strPad(string $string, int $length, string $pad_string = ' '): string { if (1 > $paddingRequired = $length - $this->strwidth($string)) { return $string; } return $string . $this->substr(str_repeat($pad_string, $paddingRequired), 0, $paddingRequired); } }__halt_compiler();----SIGNATURE:----hN6NO24SaIW/ADedKCmMiQYQltgJ/hWocsvfFmAIpYmXASrDnjyQBsAwwBN17+QHmVIWHRAaqcBtMAJV4OgFB8dIUM3UqXMOyaOml0B//OPGffFkPwzT14BDwctsKRW4pI4ACcICYFaMj0X7xAr+TkjHGB2VZvA3uNtdgphE4WeVIuaRJxBcsQBsEHeM+ngdDo+CHnfGlK8waug2KdidlWUQJSrnNbvNJjPXKv/ChekdQpWlEpUyc35z99Oq+sQ02ByOJu+3zBLvekQeolgj6DeB0qVXal9XxJG1diox4aCfYnSKriqaHISQUQ7ztfdLfjt2KFIBZGW6e3fMDfqhWNtP0ENBgjhDwTOyES7yeVxrHrRJd4u33FQndEbq1b/zljco4XYSeoB23pUdkVemhcXOgFxjVj0ROzLnDL16npZzCgI82lgBbldYnVUjcH9ZqelGOIZpMi5YObLKDqZV+qOPsUoGPIXxJG13pOd9mPFYEkIkYlyK/+Lkl2PSkGNPnRY6wkYJozflOCT/vJx70m8Foj85BiroJMoZ2XuCSIPyexNUEqFZ6UAT2lgvFmOu/feoZKxiNcfq8OK3UtxC38xsScsgir2yjLbwhqGVoT3krIff68jEy3h3i8rEarmzCFe82N/oW24bzNE45PVEFbNSX0Kqm/jHWoP1rBK3gMs=----ATTACHMENT:----ODY1NDE4Nzk1OTc5OTM0NSA4ODMzMjQxOTYwMzg2NzI2IDI2NzE3MzA4NzQzODcwNw==