123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156 |
- <?php
- /**
- * @copyright Copyright (c) 2014 Carsten Brandt
- * @license https://github.com/cebe/markdown/blob/master/LICENSE
- * @link https://github.com/cebe/markdown#readme
- */
- namespace cebe\markdown\block;
- /**
- * Adds the table blocks
- */
- trait TableTrait
- {
- /**
- * identify a line as the beginning of a table block.
- */
- protected function identifyTable($line, $lines, $current)
- {
- return strpos($line, '|') !== false && isset($lines[$current + 1])
- && preg_match('~^\\s*\\|?(\\s*:?-[\\-\\s]*:?\\s*\\|?)*\\s*$~', $lines[$current + 1])
- && strpos($lines[$current + 1], '|') !== false
- && isset($lines[$current + 2]) && trim($lines[$current + 1]) !== '';
- }
- /**
- * Consume lines for a table
- */
- protected function consumeTable($lines, $current)
- {
- // consume until newline
- $block = [
- 'table',
- 'cols' => [],
- 'rows' => [],
- ];
- for ($i = $current, $count = count($lines); $i < $count; $i++) {
- $line = trim($lines[$i]);
- // extract alignment from second line
- if ($i == $current+1) {
- $cols = explode('|', trim($line, ' |'));
- foreach($cols as $col) {
- $col = trim($col);
- if (empty($col)) {
- $block['cols'][] = '';
- continue;
- }
- $l = ($col[0] === ':');
- $r = (substr($col, -1, 1) === ':');
- if ($l && $r) {
- $block['cols'][] = 'center';
- } elseif ($l) {
- $block['cols'][] = 'left';
- } elseif ($r) {
- $block['cols'][] = 'right';
- } else {
- $block['cols'][] = '';
- }
- }
- continue;
- }
- if ($line === '' || substr($lines[$i], 0, 4) === ' ') {
- break;
- }
- if ($line[0] === '|') {
- $line = substr($line, 1);
- }
- if (substr($line, -1, 1) === '|' && (substr($line, -2, 2) !== '\\|' || substr($line, -3, 3) === '\\\\|')) {
- $line = substr($line, 0, -1);
- }
- array_unshift($this->context, 'table');
- $row = $this->parseInline($line);
- array_shift($this->context);
- $r = count($block['rows']);
- $c = 0;
- $block['rows'][] = [];
- foreach ($row as $absy) {
- if (!isset($block['rows'][$r][$c])) {
- $block['rows'][$r][] = [];
- }
- if ($absy[0] === 'tableBoundary') {
- $c++;
- } else {
- $block['rows'][$r][$c][] = $absy;
- }
- }
- }
- return [$block, --$i];
- }
- /**
- * render a table block
- */
- protected function renderTable($block)
- {
- $head = '';
- $body = '';
- $cols = $block['cols'];
- $first = true;
- foreach($block['rows'] as $row) {
- $cellTag = $first ? 'th' : 'td';
- $tds = '';
- foreach ($row as $c => $cell) {
- $align = empty($cols[$c]) ? '' : ' align="' . $cols[$c] . '"';
- $tds .= "<$cellTag$align>" . trim($this->renderAbsy($cell)) . "</$cellTag>";
- }
- if ($first) {
- $head .= "<tr>$tds</tr>\n";
- } else {
- $body .= "<tr>$tds</tr>\n";
- }
- $first = false;
- }
- return $this->composeTable($head, $body);
- }
- /**
- * This method composes a table from parsed body and head HTML.
- *
- * You may override this method to customize the table rendering, for example by
- * adding a `class` to the table tag:
- *
- * ```php
- * return "<table class="table table-striped">\n<thead>\n$head</thead>\n<tbody>\n$body</tbody>\n</table>\n"
- * ```
- *
- * @param string $head table head HTML.
- * @param string $body table body HTML.
- * @return string the complete table HTML.
- * @since 1.2.0
- */
- protected function composeTable($head, $body)
- {
- return "<table>\n<thead>\n$head</thead>\n<tbody>\n$body</tbody>\n</table>\n";
- }
- /**
- * @marker |
- */
- protected function parseTd($markdown)
- {
- if (isset($this->context[1]) && $this->context[1] === 'table') {
- return [['tableBoundary'], isset($markdown[1]) && $markdown[1] === ' ' ? 2 : 1];
- }
- return [['text', $markdown[0]], 1];
- }
- abstract protected function parseInline($text);
- abstract protected function renderAbsy($absy);
- }
|