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:----Z/lcBOmBkem+g1IcD7Eg5pOpkv/PdWXhfg8xFdsV5xjai9+tM+wnoJjqNLkouV/0vYT8PVnUM0OnFtPQVQ4kO7G9URdNlvK5Gavy+ZIDHnB8LaNcn2KJaqHyi17dDXSVEyfLNzO0ID+a300meWNiujB4F2prDkbSRs/1VIU1K1QEKqBd3SLeFLqWeNXAw+4FoJ0FhI/egf+ct6Yldf8zoCV9xgPafP2q3tyXeJeZx7Nfp7EWAx4znBjPstlBPogbZYNutqwY2ukGpDWCl0l+iDdGSinbrYcWX1T3RL1nq0Vxy9XhUsNTWdsqHrOppuSRu0gHDFsATt5OrTkAM+i0W8xuD+LpDNfAUuL97qCH3+lFKT/qNxt9ImTRW7UBCfm9+calTAzWLGhsYod+pTm6gkSGaVVNf7MKLV/7wT6MF3ysDvK2K7b8U3Lctksx2wzHjc9Pa3liPLCasru6c5hdv0WAXfq5I2EWM8AF/+LLJpks6eHLyMY4V5JDsbQcX5LrESUg5EUI0SeYmvjmLqvofkae1K3cEsjWMVX4DFOSzvuaPbZKEurUx7L6VMeXokqkE8C0wFw3qW14VNbsndlLnkcvF3UE5KIah9pnAzE3kV5cKqk7gTRe0Raau7KrmzJ+BrNzxlvrJLkE/g0rwnBIah9y0dtSwmsy5OgaDH7hL6c=----ATTACHMENT:----NzQ1NDk1ODg4NDc4MzAyNCA2Mjg1NTI2NjYxMTQxMjg1IDEwMjQyMzM2NTQ1OTY4MQ==