|
@@ -0,0 +1,99 @@
|
|
|
+<?php
|
|
|
+
|
|
|
+namespace App\Markdown\Parser;
|
|
|
+
|
|
|
+use App\Markdown\Block\IntroBlock;
|
|
|
+use League\CommonMark\Node\Block\AbstractBlock;
|
|
|
+use League\CommonMark\Node\Block\Paragraph;
|
|
|
+use League\CommonMark\Node\Inline\Text;
|
|
|
+use League\CommonMark\Parser\Block\AbstractBlockContinueParser;
|
|
|
+use League\CommonMark\Parser\Cursor;
|
|
|
+use League\CommonMark\Parser\Block\BlockContinueParserInterface;
|
|
|
+use League\CommonMark\Parser\Block\BlockContinue;
|
|
|
+use League\CommonMark\Parser\Block\BlockContinueParserWithInlinesInterface;
|
|
|
+use League\CommonMark\Parser\Block\BlockStart;
|
|
|
+use League\CommonMark\Parser\Block\BlockStartParserInterface;
|
|
|
+use League\CommonMark\Parser\InlineParserEngineInterface;
|
|
|
+use League\CommonMark\Parser\MarkdownParserStateInterface;
|
|
|
+use Twig\Node\TextNode;
|
|
|
+
|
|
|
+class IntroParser extends AbstractBlockContinueParser implements BlockContinueParserWithInlinesInterface
|
|
|
+{
|
|
|
+ protected IntroBlock $introBlock;
|
|
|
+
|
|
|
+
|
|
|
+ public function __construct(string $className ='')
|
|
|
+ {
|
|
|
+ $this->introBlock = new IntroBlock($className);
|
|
|
+
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ public function getBlock(): AbstractBlock
|
|
|
+ {
|
|
|
+ return $this->introBlock;
|
|
|
+ }
|
|
|
+
|
|
|
+ public function tryContinue(Cursor $cursor, BlockContinueParserInterface $activeBlockParser): ?BlockContinue
|
|
|
+ {
|
|
|
+ if (trim($cursor->getLine()) === ':::') {
|
|
|
+ return BlockContinue::finished();
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ $line = $cursor->getRemainder();
|
|
|
+ $paragraph =new Paragraph();
|
|
|
+ $paragraph->appendChild(new Text($line));
|
|
|
+ $this->introBlock->appendChild($paragraph);
|
|
|
+ return BlockContinue::at($cursor);
|
|
|
+ }
|
|
|
+
|
|
|
+ public function parseInlines(InlineParserEngineInterface $inlineParser): void
|
|
|
+ {
|
|
|
+ foreach($this->introBlock->children() as $child) {
|
|
|
+ if ($child instanceof Paragraph) {
|
|
|
+ $textContent = '';
|
|
|
+
|
|
|
+ foreach($child->children() as $textNode) {
|
|
|
+ if ($textNode instanceof Text) {
|
|
|
+ $textContent .= $textNode->getLiteral();
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ foreach($child->children() as $textNode) {
|
|
|
+ $textNode->detach();
|
|
|
+ }
|
|
|
+
|
|
|
+ $inlineParser->parse($textContent, $child);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ public static function createBlockStartParser(): BlockStartParserInterface
|
|
|
+ {
|
|
|
+ return new class implements BlockStartParserInterface
|
|
|
+ {
|
|
|
+ public function tryStart(Cursor $cursor, MarkdownParserStateInterface $parserState): ?BlockStart
|
|
|
+ {
|
|
|
+ if ($match = $cursor->match('/^:::\s+([^\s]+)/')) {
|
|
|
+ $className = explode(' ', $match)[1] ?? '';
|
|
|
+ return BlockStart::of(new IntroParser($className))->at($cursor);
|
|
|
+ }
|
|
|
+ return BlockStart::none();
|
|
|
+ }
|
|
|
+ };
|
|
|
+ }
|
|
|
+}
|