1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
|
<?php
/**
* SPDX-FileCopyrightText: 2016-2024 Nextcloud GmbH and Nextcloud contributors
* SPDX-FileCopyrightText: 2016 ownCloud, Inc.
* SPDX-License-Identifier: AGPL-3.0-only
*/
namespace OC\Preview;
use OCP\Files\File;
use OCP\IImage;
class MarkDown extends TXT {
/**
* {@inheritDoc}
*/
public function getMimeType(): string {
return '/text\/(x-)?markdown/';
}
public function getThumbnail(File $file, int $maxX, int $maxY): ?IImage {
$content = $file->fopen('r');
if ($content === false) {
return null;
}
$content = stream_get_contents($content, 3000);
//don't create previews of empty text files
if (trim($content) === '') {
return null;
}
// Merge text paragraph lines that might belong together
$content = preg_replace('/^(\s*)\*\s/mU', '$1- ', $content);
$content = preg_replace('/((?!^(\s*-|#)).*)(\w|\\|\.)(\r\n|\n|\r)(\w|\*)/mU', '$1 $3', $content);
// Remove markdown symbols that we cannot easily represent in rendered text in the preview
$content = preg_replace('/\*\*(.*)\*\*/U', '$1', $content);
$content = preg_replace('/\*(.*)\*/U', '$1', $content);
$content = preg_replace('/\_\_(.*)\_\_/U', '$1', $content);
$content = preg_replace('/\_(.*)\_/U', '$1', $content);
$content = preg_replace('/\~\~(.*)\~\~/U', '$1', $content);
$content = preg_replace('/\!?\[((.|\n)*)\]\((.*)\)/mU', '$1 ($3)', $content);
$content = preg_replace('/\n\n+/', "\n", $content);
$content = preg_replace('/[\x{10000}-\x{10FFFF}]/u', '', $content);
$lines = preg_split("/\r\n|\n|\r/", $content);
// Define text size of text file preview
$fontSize = $maxX ? (int)((1 / ($maxX >= 512 ? 60 : 40) * $maxX)) : 10;
$image = imagecreate($maxX, $maxY);
imagecolorallocate($image, 255, 255, 255);
$textColor = imagecolorallocate($image, 0, 0, 0);
$fontFile = __DIR__ . '/../../../core/fonts/NotoSans-Regular.ttf';
$fontFileBold = __DIR__ . '/../../../core/fonts/NotoSans-Bold.ttf';
$canUseTTF = function_exists('imagettftext');
$textOffset = (int)min($maxX * 0.05, $maxY * 0.05);
$nextLineStart = 0;
$y = $textOffset;
foreach ($lines as $line) {
$actualFontSize = $fontSize;
if (mb_strpos($line, '# ') === 0) {
$actualFontSize *= 2;
}
if (mb_strpos($line, '## ') === 0) {
$actualFontSize *= 1.8;
}
if (mb_strpos($line, '### ') === 0) {
$actualFontSize *= 1.6;
}
if (mb_strpos($line, '#### ') === 0) {
$actualFontSize *= 1.4;
}
if (mb_strpos($line, '##### ') === 0) {
$actualFontSize *= 1.2;
}
if (mb_strpos($line, '###### ') === 0) {
$actualFontSize *= 1.1;
}
// Add spacing before headlines
if ($actualFontSize !== $fontSize && $y !== $textOffset) {
$y += (int)($actualFontSize * 2);
}
$x = $textOffset;
$y += (int)($nextLineStart + $actualFontSize);
if ($canUseTTF === true) {
$wordWrap = (int)((1 / $actualFontSize * 1.3) * $maxX);
// Get rid of markdown symbols that we still needed for the font size
$line = preg_replace('/^#*\s/', '', $line);
$wrappedText = wordwrap($line, $wordWrap, "\n");
$linesWrapped = count(explode("\n", $wrappedText));
imagettftext($image, $actualFontSize, 0, $x, $y, $textColor, $actualFontSize === $fontSize ? $fontFile : $fontFileBold, $wrappedText);
$nextLineStart = (int)($linesWrapped * ceil($actualFontSize * 2));
if ($actualFontSize !== $fontSize && $y !== $textOffset) {
$nextLineStart -= $actualFontSize;
}
} else {
$y -= $fontSize;
imagestring($image, 1, $x, $y, $line, $textColor);
$nextLineStart = $fontSize;
}
if ($y >= $maxY) {
break;
}
}
$imageObject = new \OCP\Image();
$imageObject->setResource($image);
return $imageObject->valid() ? $imageObject : null;
}
}
|