Table.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412
  1. <?php
  2. /**
  3. * @link https://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license https://www.yiiframework.com/license/
  6. */
  7. namespace yii\console\widgets;
  8. use Yii;
  9. use yii\base\Widget;
  10. use yii\helpers\ArrayHelper;
  11. use yii\helpers\Console;
  12. /**
  13. * Table class displays a table in console.
  14. *
  15. * For example,
  16. *
  17. * ```php
  18. * $table = new Table();
  19. *
  20. * echo $table
  21. * ->setHeaders(['test1', 'test2', 'test3'])
  22. * ->setRows([
  23. * ['col1', 'col2', 'col3'],
  24. * ['col1', 'col2', ['col3-0', 'col3-1', 'col3-2']],
  25. * ])
  26. * ->run();
  27. * ```
  28. *
  29. * or
  30. *
  31. * ```php
  32. * echo Table::widget([
  33. * 'headers' => ['test1', 'test2', 'test3'],
  34. * 'rows' => [
  35. * ['col1', 'col2', 'col3'],
  36. * ['col1', 'col2', ['col3-0', 'col3-1', 'col3-2']],
  37. * ],
  38. * ]);
  39. *
  40. * @property-write string $listPrefix List prefix.
  41. * @property-write int $screenWidth Screen width.
  42. *
  43. * @author Daniel Gomez Pan <pana_1990@hotmail.com>
  44. * @since 2.0.13
  45. */
  46. class Table extends Widget
  47. {
  48. const DEFAULT_CONSOLE_SCREEN_WIDTH = 120;
  49. const CONSOLE_SCROLLBAR_OFFSET = 3;
  50. const CHAR_TOP = 'top';
  51. const CHAR_TOP_MID = 'top-mid';
  52. const CHAR_TOP_LEFT = 'top-left';
  53. const CHAR_TOP_RIGHT = 'top-right';
  54. const CHAR_BOTTOM = 'bottom';
  55. const CHAR_BOTTOM_MID = 'bottom-mid';
  56. const CHAR_BOTTOM_LEFT = 'bottom-left';
  57. const CHAR_BOTTOM_RIGHT = 'bottom-right';
  58. const CHAR_LEFT = 'left';
  59. const CHAR_LEFT_MID = 'left-mid';
  60. const CHAR_MID = 'mid';
  61. const CHAR_MID_MID = 'mid-mid';
  62. const CHAR_RIGHT = 'right';
  63. const CHAR_RIGHT_MID = 'right-mid';
  64. const CHAR_MIDDLE = 'middle';
  65. /**
  66. * @var array table headers
  67. * @since 2.0.19
  68. */
  69. protected $headers = [];
  70. /**
  71. * @var array table rows
  72. * @since 2.0.19
  73. */
  74. protected $rows = [];
  75. /**
  76. * @var array table chars
  77. * @since 2.0.19
  78. */
  79. protected $chars = [
  80. self::CHAR_TOP => '═',
  81. self::CHAR_TOP_MID => '╤',
  82. self::CHAR_TOP_LEFT => '╔',
  83. self::CHAR_TOP_RIGHT => '╗',
  84. self::CHAR_BOTTOM => '═',
  85. self::CHAR_BOTTOM_MID => '╧',
  86. self::CHAR_BOTTOM_LEFT => '╚',
  87. self::CHAR_BOTTOM_RIGHT => '╝',
  88. self::CHAR_LEFT => '║',
  89. self::CHAR_LEFT_MID => '╟',
  90. self::CHAR_MID => '─',
  91. self::CHAR_MID_MID => '┼',
  92. self::CHAR_RIGHT => '║',
  93. self::CHAR_RIGHT_MID => '╢',
  94. self::CHAR_MIDDLE => '│',
  95. ];
  96. /**
  97. * @var array table column widths
  98. * @since 2.0.19
  99. */
  100. protected $columnWidths = [];
  101. /**
  102. * @var int screen width
  103. * @since 2.0.19
  104. */
  105. protected $screenWidth;
  106. /**
  107. * @var string list prefix
  108. * @since 2.0.19
  109. */
  110. protected $listPrefix = '• ';
  111. /**
  112. * Set table headers.
  113. *
  114. * @param array $headers table headers
  115. * @return $this
  116. */
  117. public function setHeaders(array $headers)
  118. {
  119. $this->headers = array_values($headers);
  120. return $this;
  121. }
  122. /**
  123. * Set table rows.
  124. *
  125. * @param array $rows table rows
  126. * @return $this
  127. */
  128. public function setRows(array $rows)
  129. {
  130. $this->rows = array_map(function($row) {
  131. return array_map(function($value) {
  132. return empty($value) && !is_numeric($value) ? ' ' : $value;
  133. }, array_values($row));
  134. }, $rows);
  135. return $this;
  136. }
  137. /**
  138. * Set table chars.
  139. *
  140. * @param array $chars table chars
  141. * @return $this
  142. */
  143. public function setChars(array $chars)
  144. {
  145. $this->chars = $chars;
  146. return $this;
  147. }
  148. /**
  149. * Set screen width.
  150. *
  151. * @param int $width screen width
  152. * @return $this
  153. */
  154. public function setScreenWidth($width)
  155. {
  156. $this->screenWidth = $width;
  157. return $this;
  158. }
  159. /**
  160. * Set list prefix.
  161. *
  162. * @param string $listPrefix list prefix
  163. * @return $this
  164. */
  165. public function setListPrefix($listPrefix)
  166. {
  167. $this->listPrefix = $listPrefix;
  168. return $this;
  169. }
  170. /**
  171. * @return string the rendered table
  172. */
  173. public function run()
  174. {
  175. $this->calculateRowsSize();
  176. $headerCount = count($this->headers);
  177. $buffer = $this->renderSeparator(
  178. $this->chars[self::CHAR_TOP_LEFT],
  179. $this->chars[self::CHAR_TOP_MID],
  180. $this->chars[self::CHAR_TOP],
  181. $this->chars[self::CHAR_TOP_RIGHT]
  182. );
  183. // Header
  184. if ($headerCount > 0) {
  185. $buffer .= $this->renderRow($this->headers,
  186. $this->chars[self::CHAR_LEFT],
  187. $this->chars[self::CHAR_MIDDLE],
  188. $this->chars[self::CHAR_RIGHT]
  189. );
  190. }
  191. // Content
  192. foreach ($this->rows as $i => $row) {
  193. if ($i > 0 || $headerCount > 0) {
  194. $buffer .= $this->renderSeparator(
  195. $this->chars[self::CHAR_LEFT_MID],
  196. $this->chars[self::CHAR_MID_MID],
  197. $this->chars[self::CHAR_MID],
  198. $this->chars[self::CHAR_RIGHT_MID]
  199. );
  200. }
  201. $buffer .= $this->renderRow($row,
  202. $this->chars[self::CHAR_LEFT],
  203. $this->chars[self::CHAR_MIDDLE],
  204. $this->chars[self::CHAR_RIGHT]);
  205. }
  206. $buffer .= $this->renderSeparator(
  207. $this->chars[self::CHAR_BOTTOM_LEFT],
  208. $this->chars[self::CHAR_BOTTOM_MID],
  209. $this->chars[self::CHAR_BOTTOM],
  210. $this->chars[self::CHAR_BOTTOM_RIGHT]
  211. );
  212. return $buffer;
  213. }
  214. /**
  215. * Renders a row of data into a string.
  216. *
  217. * @param array $row row of data
  218. * @param string $spanLeft character for left border
  219. * @param string $spanMiddle character for middle border
  220. * @param string $spanRight character for right border
  221. * @return string
  222. * @see \yii\console\widgets\Table::render()
  223. */
  224. protected function renderRow(array $row, $spanLeft, $spanMiddle, $spanRight)
  225. {
  226. $size = $this->columnWidths;
  227. $buffer = '';
  228. $arrayPointer = [];
  229. $renderedChunkTexts = [];
  230. for ($i = 0, ($max = $this->calculateRowHeight($row)) ?: $max = 1; $i < $max; $i++) {
  231. $buffer .= $spanLeft . ' ';
  232. foreach ($size as $index => $cellSize) {
  233. $cell = isset($row[$index]) ? $row[$index] : null;
  234. $prefix = '';
  235. if ($index !== 0) {
  236. $buffer .= $spanMiddle . ' ';
  237. }
  238. if (is_array($cell)) {
  239. if (empty($renderedChunkTexts[$index])) {
  240. $renderedChunkTexts[$index] = '';
  241. $start = 0;
  242. $prefix = $this->listPrefix;
  243. if (!isset($arrayPointer[$index])) {
  244. $arrayPointer[$index] = 0;
  245. }
  246. } else {
  247. $start = mb_strwidth($renderedChunkTexts[$index], Yii::$app->charset);
  248. }
  249. $chunk = Console::ansiColorizedSubstr($cell[$arrayPointer[$index]], $start, $cellSize - 4);
  250. $renderedChunkTexts[$index] .= Console::stripAnsiFormat($chunk);
  251. $fullChunkText = Console::stripAnsiFormat($cell[$arrayPointer[$index]]);
  252. if (isset($cell[$arrayPointer[$index] + 1]) && $renderedChunkTexts[$index] === $fullChunkText) {
  253. $arrayPointer[$index]++;
  254. $renderedChunkTexts[$index] = '';
  255. }
  256. } else {
  257. $chunk = Console::ansiColorizedSubstr($cell, ($cellSize * $i) - ($i * 2), $cellSize - 2);
  258. }
  259. $chunk = $prefix . $chunk;
  260. $repeat = $cellSize - Console::ansiStrwidth($chunk) - 1;
  261. $buffer .= $chunk;
  262. if ($repeat >= 0) {
  263. $buffer .= str_repeat(' ', $repeat);
  264. }
  265. }
  266. $buffer .= "$spanRight\n";
  267. }
  268. return $buffer;
  269. }
  270. /**
  271. * Renders separator.
  272. *
  273. * @param string $spanLeft character for left border
  274. * @param string $spanMid character for middle border
  275. * @param string $spanMidMid character for middle-middle border
  276. * @param string $spanRight character for right border
  277. * @return string the generated separator row
  278. * @see \yii\console\widgets\Table::render()
  279. */
  280. protected function renderSeparator($spanLeft, $spanMid, $spanMidMid, $spanRight)
  281. {
  282. $separator = $spanLeft;
  283. foreach ($this->columnWidths as $index => $rowSize) {
  284. if ($index !== 0) {
  285. $separator .= $spanMid;
  286. }
  287. $separator .= str_repeat($spanMidMid, $rowSize);
  288. }
  289. $separator .= $spanRight . "\n";
  290. return $separator;
  291. }
  292. /**
  293. * Calculate the size of rows to draw anchor of columns in console.
  294. *
  295. * @see \yii\console\widgets\Table::render()
  296. */
  297. protected function calculateRowsSize()
  298. {
  299. $this->columnWidths = $columns = [];
  300. $totalWidth = 0;
  301. $screenWidth = $this->getScreenWidth() - self::CONSOLE_SCROLLBAR_OFFSET;
  302. $headerCount = count($this->headers);
  303. if (empty($this->rows)) {
  304. $rowColCount = 0;
  305. } else {
  306. $rowColCount = max(array_map('count', $this->rows));
  307. }
  308. $count = max($headerCount, $rowColCount);
  309. for ($i = 0; $i < $count; $i++) {
  310. $columns[] = ArrayHelper::getColumn($this->rows, $i);
  311. if ($i < $headerCount) {
  312. $columns[$i][] = $this->headers[$i];
  313. }
  314. }
  315. foreach ($columns as $column) {
  316. $columnWidth = max(array_map(function ($val) {
  317. if (is_array($val)) {
  318. return max(array_map('yii\helpers\Console::ansiStrwidth', $val)) + Console::ansiStrwidth($this->listPrefix);
  319. }
  320. return Console::ansiStrwidth($val);
  321. }, $column)) + 2;
  322. $this->columnWidths[] = $columnWidth;
  323. $totalWidth += $columnWidth;
  324. }
  325. if ($totalWidth > $screenWidth) {
  326. $minWidth = 3;
  327. $fixWidths = [];
  328. $relativeWidth = $screenWidth / $totalWidth;
  329. foreach ($this->columnWidths as $j => $width) {
  330. $scaledWidth = (int) ($width * $relativeWidth);
  331. if ($scaledWidth < $minWidth) {
  332. $fixWidths[$j] = 3;
  333. }
  334. }
  335. $totalFixWidth = array_sum($fixWidths);
  336. $relativeWidth = ($screenWidth - $totalFixWidth) / ($totalWidth - $totalFixWidth);
  337. foreach ($this->columnWidths as $j => $width) {
  338. if (!array_key_exists($j, $fixWidths)) {
  339. $this->columnWidths[$j] = (int) ($width * $relativeWidth);
  340. }
  341. }
  342. }
  343. }
  344. /**
  345. * Calculate the height of a row.
  346. *
  347. * @param array $row
  348. * @return int maximum row per cell
  349. * @see \yii\console\widgets\Table::render()
  350. */
  351. protected function calculateRowHeight($row)
  352. {
  353. $rowsPerCell = array_map(function ($size, $columnWidth) {
  354. if (is_array($columnWidth)) {
  355. $rows = 0;
  356. foreach ($columnWidth as $width) {
  357. $rows += $size == 2 ? 0 : ceil($width / ($size - 2));
  358. }
  359. return $rows;
  360. }
  361. return $size == 2 || $columnWidth == 0 ? 0 : ceil($columnWidth / ($size - 2));
  362. }, $this->columnWidths, array_map(function ($val) {
  363. if (is_array($val)) {
  364. return array_map('yii\helpers\Console::ansiStrwidth', $val);
  365. }
  366. return Console::ansiStrwidth($val);
  367. }, $row));
  368. return max($rowsPerCell);
  369. }
  370. /**
  371. * Getting screen width.
  372. * If it is not able to determine screen width, default value `123` will be set.
  373. *
  374. * @return int screen width
  375. */
  376. protected function getScreenWidth()
  377. {
  378. if (!$this->screenWidth) {
  379. $size = Console::getScreenSize();
  380. $this->screenWidth = isset($size[0])
  381. ? $size[0]
  382. : self::DEFAULT_CONSOLE_SCREEN_WIDTH + self::CONSOLE_SCROLLBAR_OFFSET;
  383. }
  384. return $this->screenWidth;
  385. }
  386. }