init
This commit is contained in:
197
vendor/cebe/markdown/block/ListTrait.php
vendored
Normal file
197
vendor/cebe/markdown/block/ListTrait.php
vendored
Normal file
@@ -0,0 +1,197 @@
|
||||
<?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 list blocks
|
||||
*/
|
||||
trait ListTrait
|
||||
{
|
||||
/**
|
||||
* @var bool enable support `start` attribute of ordered lists. This means that lists
|
||||
* will start with the number you actually type in markdown and not the HTML generated one.
|
||||
* Defaults to `false` which means that numeration of all ordered lists(<ol>) starts with 1.
|
||||
*/
|
||||
public $keepListStartNumber = false;
|
||||
|
||||
/**
|
||||
* identify a line as the beginning of an ordered list.
|
||||
*/
|
||||
protected function identifyOl($line)
|
||||
{
|
||||
return (($l = $line[0]) > '0' && $l <= '9' || $l === ' ') && preg_match('/^ {0,3}\d+\.[ \t]/', $line);
|
||||
}
|
||||
|
||||
/**
|
||||
* identify a line as the beginning of an unordered list.
|
||||
*/
|
||||
protected function identifyUl($line)
|
||||
{
|
||||
$l = $line[0];
|
||||
return ($l === '-' || $l === '+' || $l === '*') && (isset($line[1]) && (($l1 = $line[1]) === ' ' || $l1 === "\t")) ||
|
||||
($l === ' ' && preg_match('/^ {0,3}[\-\+\*][ \t]/', $line));
|
||||
}
|
||||
|
||||
/**
|
||||
* Consume lines for an ordered list
|
||||
*/
|
||||
protected function consumeOl($lines, $current)
|
||||
{
|
||||
// consume until newline
|
||||
|
||||
$block = [
|
||||
'list',
|
||||
'list' => 'ol',
|
||||
'attr' => [],
|
||||
'items' => [],
|
||||
];
|
||||
return $this->consumeList($lines, $current, $block, 'ol');
|
||||
}
|
||||
|
||||
/**
|
||||
* Consume lines for an unordered list
|
||||
*/
|
||||
protected function consumeUl($lines, $current)
|
||||
{
|
||||
// consume until newline
|
||||
|
||||
$block = [
|
||||
'list',
|
||||
'list' => 'ul',
|
||||
'items' => [],
|
||||
];
|
||||
return $this->consumeList($lines, $current, $block, 'ul');
|
||||
}
|
||||
|
||||
private function consumeList($lines, $current, $block, $type)
|
||||
{
|
||||
$item = 0;
|
||||
$indent = '';
|
||||
$len = 0;
|
||||
$lastLineEmpty = false;
|
||||
// track the indentation of list markers, if indented more than previous element
|
||||
// a list marker is considered to be long to a lower level
|
||||
$leadSpace = 3;
|
||||
$marker = $type === 'ul' ? ltrim($lines[$current])[0] : '';
|
||||
for ($i = $current, $count = count($lines); $i < $count; $i++) {
|
||||
$line = $lines[$i];
|
||||
// match list marker on the beginning of the line
|
||||
$pattern = ($type == 'ol') ? '/^( {0,'.$leadSpace.'})(\d+)\.[ \t]+/' : '/^( {0,'.$leadSpace.'})\\'.$marker.'[ \t]+/';
|
||||
if (preg_match($pattern, $line, $matches)) {
|
||||
if (($len = substr_count($matches[0], "\t")) > 0) {
|
||||
$indent = str_repeat("\t", $len);
|
||||
$line = substr($line, strlen($matches[0]));
|
||||
} else {
|
||||
$len = strlen($matches[0]);
|
||||
$indent = str_repeat(' ', $len);
|
||||
$line = substr($line, $len);
|
||||
}
|
||||
if ($i === $current) {
|
||||
$leadSpace = strlen($matches[1]) + 1;
|
||||
}
|
||||
|
||||
if ($type == 'ol' && $this->keepListStartNumber) {
|
||||
// attr `start` for ol
|
||||
if (!isset($block['attr']['start']) && isset($matches[2])) {
|
||||
$block['attr']['start'] = $matches[2];
|
||||
}
|
||||
}
|
||||
|
||||
$block['items'][++$item][] = $line;
|
||||
$block['lazyItems'][$item] = $lastLineEmpty;
|
||||
$lastLineEmpty = false;
|
||||
} elseif (ltrim($line) === '') {
|
||||
// line is empty, may be a lazy list
|
||||
$lastLineEmpty = true;
|
||||
|
||||
// two empty lines will end the list
|
||||
if (!isset($lines[$i + 1][0])) {
|
||||
break;
|
||||
|
||||
// next item is the continuation of this list -> lazy list
|
||||
} elseif (preg_match($pattern, $lines[$i + 1])) {
|
||||
$block['items'][$item][] = $line;
|
||||
$block['lazyItems'][$item] = true;
|
||||
|
||||
// next item is indented as much as this list -> lazy list if it is not a reference
|
||||
} elseif (strncmp($lines[$i + 1], $indent, $len) === 0 || !empty($lines[$i + 1]) && $lines[$i + 1][0] == "\t") {
|
||||
$block['items'][$item][] = $line;
|
||||
$nextLine = $lines[$i + 1][0] === "\t" ? substr($lines[$i + 1], 1) : substr($lines[$i + 1], $len);
|
||||
$block['lazyItems'][$item] = empty($nextLine) || !method_exists($this, 'identifyReference') || !$this->identifyReference($nextLine);
|
||||
|
||||
// everything else ends the list
|
||||
} else {
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
if ($line[0] === "\t") {
|
||||
$line = substr($line, 1);
|
||||
} elseif (strncmp($line, $indent, $len) === 0) {
|
||||
$line = substr($line, $len);
|
||||
}
|
||||
$block['items'][$item][] = $line;
|
||||
$lastLineEmpty = false;
|
||||
}
|
||||
}
|
||||
|
||||
foreach($block['items'] as $itemId => $itemLines) {
|
||||
$content = [];
|
||||
if (!$block['lazyItems'][$itemId]) {
|
||||
$firstPar = [];
|
||||
while (!empty($itemLines) && rtrim($itemLines[0]) !== '' && $this->detectLineType($itemLines, 0) === 'paragraph') {
|
||||
$firstPar[] = array_shift($itemLines);
|
||||
}
|
||||
$content = $this->parseInline(implode("\n", $firstPar));
|
||||
}
|
||||
if (!empty($itemLines)) {
|
||||
$content = array_merge($content, $this->parseBlocks($itemLines));
|
||||
}
|
||||
$block['items'][$itemId] = $content;
|
||||
}
|
||||
|
||||
return [$block, $i];
|
||||
}
|
||||
|
||||
/**
|
||||
* Renders a list
|
||||
*/
|
||||
protected function renderList($block)
|
||||
{
|
||||
$type = $block['list'];
|
||||
|
||||
if (!empty($block['attr'])) {
|
||||
$output = "<$type " . $this->generateHtmlAttributes($block['attr']) . ">\n";
|
||||
} else {
|
||||
$output = "<$type>\n";
|
||||
}
|
||||
|
||||
foreach ($block['items'] as $item => $itemLines) {
|
||||
$output .= '<li>' . $this->renderAbsy($itemLines). "</li>\n";
|
||||
}
|
||||
return $output . "</$type>\n";
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Return html attributes string from [attrName => attrValue] list
|
||||
* @param array $attributes the attribute name-value pairs.
|
||||
* @return string
|
||||
*/
|
||||
private function generateHtmlAttributes($attributes)
|
||||
{
|
||||
foreach ($attributes as $name => $value) {
|
||||
$attributes[$name] = "$name=\"$value\"";
|
||||
}
|
||||
return implode(' ', $attributes);
|
||||
}
|
||||
|
||||
abstract protected function parseBlocks($lines);
|
||||
abstract protected function parseInline($text);
|
||||
abstract protected function renderAbsy($absy);
|
||||
abstract protected function detectLineType($lines, $current);
|
||||
}
|
||||
Reference in New Issue
Block a user