1
0
cms11/app/Services/Markdown/Formatter.php

161 lines
5.3 KiB
PHP
Raw Normal View History

<?php
namespace App\Services\Markdown;
use App\Classes\Bundle;
use App\Services\Markdown\Renderers\LinkRenderer;
2024-04-22 17:04:46 +02:00
use DOMDocument;
use Illuminate\Support\Facades\Blade;
use Illuminate\Support\Facades\Cache;
use League\CommonMark\Environment\Environment;
use League\CommonMark\Extension\CommonMark\CommonMarkCoreExtension;
use League\CommonMark\Extension\CommonMark\Node\Inline\Link;
use League\CommonMark\Extension\DisallowedRawHtml\DisallowedRawHtmlExtension;
use League\CommonMark\Extension\ExternalLink\ExternalLinkExtension;
use League\CommonMark\Extension\Footnote\FootnoteExtension;
use League\CommonMark\Extension\FrontMatter\FrontMatterExtension;
use League\CommonMark\Extension\HeadingPermalink\HeadingPermalinkExtension;
use League\CommonMark\Extension\Strikethrough\StrikethroughExtension;
use League\CommonMark\Extension\Table\TableExtension;
use League\CommonMark\Extension\TableOfContents\TableOfContentsExtension;
use League\CommonMark\Extension\TaskList\TaskListExtension;
use League\CommonMark\MarkdownConverter;
use Spatie\CommonMarkShikiHighlighter\HighlightCodeExtension;
/**
* Formats markdown content using a variety of CommonMark extensions
* and custom renderers, also incorporating Blade template rendering.
*/
class Formatter
{
/**
* Constructor.
*
* @param string $source The markdown source content.
*/
public function __construct(protected string $source, protected ?Bundle $bundle = null)
{
}
/**
* Renders the markdown content with all enabled extensions and custom processing.
*
* @return string The rendered HTML content.
*/
public function render(): string
{
2024-04-22 21:51:50 +02:00
if (empty($this->source)) {
return $this->source;
}
$hash = md5($this->source);
$cacheKey = sprintf('markdown_formatted_%s', $hash);
if (Cache::has($cacheKey)) {
2024-04-24 00:47:46 +02:00
return Cache::get($cacheKey);
}
// First, process the source with Blade to handle any dynamic content
$result = Blade::render($this->source);
// Convert markdown to HTML using the configured environment and extensions
$converter = new MarkdownConverter($this->prepareEnvironment());
$converted = $converter->convert($result);
// Perform final adjustments like inserting non-breaking spaces
$result = $this->insertNonBreakingSpaces($converted->getContent());
2024-04-22 17:04:46 +02:00
$result = $this->removeEmptyCodeLines($result);
2024-04-24 00:47:46 +02:00
Cache::put($cacheKey, $result, now()->addMonth());
return $result;
}
/**
* Prepares the markdown parser environment with extensions and custom renderers.
*
* @return Environment The configured markdown environment.
*/
protected function prepareEnvironment(): Environment
{
// Load markdown configuration and initialize the environment
$environment = new Environment(config('markdown'));
// Define all the extensions to be used
$extensions = [
new CommonMarkCoreExtension(),
new HighlightCodeExtension(theme: 'aurora-x'),
new FrontMatterExtension(),
new FootnoteExtension(),
new ExternalLinkExtension(),
new HeadingPermalinkExtension(),
new StrikethroughExtension(),
new TableExtension(),
new TableOfContentsExtension(),
new DisallowedRawHtmlExtension(),
new TaskListExtension(),
];
// Add each extension to the environment
foreach ($extensions as $extension) {
$environment->addExtension($extension);
}
// Use a custom renderer for links
$environment->addRenderer(Link::class, new LinkRenderer($this->bundle));
return $environment;
}
/**
* Inserts non-breaking spaces before certain punctuation marks in French typography.
*
* @param string $html The HTML content to process.
* @return string The processed HTML content with non-breaking spaces added.
*/
protected function insertNonBreakingSpaces(string $html): string
{
// Define patterns for spaces that should be replaced with non-breaking spaces
$patterns = [
'/ (\;)/',
'/ (\:)/',
'/ (\!)/',
'/ (\?)/',
];
// Corresponding replacements for each pattern
$replacements = array_fill(0, count($patterns), '&nbsp;$1');
// Perform the replacements and return the modified HTML
return preg_replace($patterns, $replacements, $html);
}
2024-04-22 17:04:46 +02:00
/**
* For some reason, to date (04-2024), HighlightCodeExtension appends empty
* lines at the end of a code block. We need to get rid of them.
*/
protected function removeEmptyCodeLines(string $html): string
{
$body = mb_convert_encoding($html, 'HTML-ENTITIES', 'UTF-8');
libxml_use_internal_errors(true);
$dom = new DOMDocument('1.0', 'UTF-8');
$dom->loadHtml($body);
libxml_clear_errors();
$xpath = new \DOMXPath($dom);
$spans = $xpath->query('//pre/code/span[@class="line" and not(node())]');
foreach ($spans as $span) {
if ($span->isSameNode($span->parentNode->lastChild)) {
$span->parentNode->removeChild($span);
}
}
return $dom->saveHTML();
}
}