[TASK] Update php-cs-fixer to 2.5.0
[Packages/TYPO3.CMS.git] / typo3 / sysext / extbase / Classes / Utility / DebuggerUtility.php
1 <?php
2 namespace TYPO3\CMS\Extbase\Utility;
3
4 /* *
5 * This script belongs to the Extbase framework *
6 * *
7 * It is free software; you can redistribute it and/or modify it under *
8 * the terms of the GNU Lesser General Public License as published by the *
9 * Free Software Foundation, either version 3 of the License, or (at your *
10 * option) any later version. *
11 * *
12 * This script is distributed in the hope that it will be useful, but *
13 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHAN- *
14 * TABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser *
15 * General Public License for more details. *
16 * *
17 * You should have received a copy of the GNU Lesser General Public *
18 * License along with the script. *
19 * If not, see http://www.gnu.org/licenses/lgpl.html *
20 * *
21 * The TYPO3 project - inspiring people to share! *
22 * */
23 /**
24 * This class is a backport of the corresponding class of TYPO3 Flow.
25 * All credits go to the TYPO3 Flow team.
26 */
27 /**
28 * A debugging utility class
29 *
30 * @license http://www.gnu.org/licenses/lgpl.html GNU Lesser General Public License, version 3 or later
31 * @api
32 */
33 class DebuggerUtility
34 {
35 const PLAINTEXT_INDENT = ' ';
36 const HTML_INDENT = '&nbsp;&nbsp;&nbsp;';
37
38 /**
39 * @var \TYPO3\CMS\Extbase\Persistence\ObjectStorage
40 */
41 protected static $renderedObjects;
42
43 /**
44 * Hardcoded list of Extbase class names (regex) which should not be displayed during debugging
45 *
46 * @var array
47 */
48 protected static $blacklistedClassNames = [
49 'PHPUnit_Framework_MockObject_InvocationMocker',
50 \TYPO3\CMS\Extbase\Reflection\ReflectionService::class,
51 \TYPO3\CMS\Extbase\Object\ObjectManager::class,
52 \TYPO3\CMS\Extbase\Persistence\Generic\Mapper\DataMapper::class,
53 \TYPO3\CMS\Extbase\Persistence\Generic\PersistenceManager::class,
54 \TYPO3\CMS\Extbase\Persistence\Generic\Qom\QueryObjectModelFactory::class,
55 \TYPO3\CMS\Frontend\ContentObject\ContentObjectRenderer::class
56 ];
57
58 /**
59 * Hardcoded list of property names (regex) which should not be displayed during debugging
60 *
61 * @var array
62 */
63 protected static $blacklistedPropertyNames = ['warning'];
64
65 /**
66 * Is set to TRUE once the CSS file is included in the current page to prevent double inclusions of the CSS file.
67 *
68 * @var bool
69 */
70 protected static $stylesheetEchoed = false;
71
72 /**
73 * Defines the max recursion depth of the dump, set to 8 due to common memory limits
74 *
75 * @var int
76 */
77 protected static $maxDepth = 8;
78
79 /**
80 * Clear the state of the debugger
81 */
82 protected static function clearState()
83 {
84 self::$renderedObjects = new \TYPO3\CMS\Extbase\Persistence\ObjectStorage();
85 }
86
87 /**
88 * Renders a dump of the given value
89 *
90 * @param mixed $value
91 * @param int $level
92 * @param bool $plainText
93 * @param bool $ansiColors
94 * @return string
95 */
96 protected static function renderDump($value, $level, $plainText, $ansiColors)
97 {
98 $dump = '';
99 if (is_string($value)) {
100 $croppedValue = strlen($value) > 2000 ? substr($value, 0, 2000) . '...' : $value;
101 if ($plainText) {
102 $dump = self::ansiEscapeWrap(('"' . implode((PHP_EOL . str_repeat(self::PLAINTEXT_INDENT, ($level + 1))), str_split($croppedValue, 76)) . '"'), '33', $ansiColors) . ' (' . strlen($value) . ' chars)';
103 } else {
104 $lines = str_split($croppedValue, 76);
105 $lines = array_map('htmlspecialchars', $lines);
106 $dump = sprintf('\'<span class="extbase-debug-string">%s</span>\' (%s chars)', implode('<br />' . str_repeat(self::HTML_INDENT, $level + 1), $lines), strlen($value));
107 }
108 } elseif (is_numeric($value)) {
109 $dump = sprintf('%s (%s)', self::ansiEscapeWrap($value, '35', $ansiColors), gettype($value));
110 } elseif (is_bool($value)) {
111 $dump = $value ? self::ansiEscapeWrap('TRUE', '32', $ansiColors) : self::ansiEscapeWrap('FALSE', '32', $ansiColors);
112 } elseif (is_null($value) || is_resource($value)) {
113 $dump = gettype($value);
114 } elseif (is_array($value)) {
115 $dump = self::renderArray($value, $level + 1, $plainText, $ansiColors);
116 } elseif (is_object($value)) {
117 if ($value instanceof \Closure) {
118 $dump = self::renderClosure($value, $level + 1, $plainText, $ansiColors);
119 } else {
120 $dump = self::renderObject($value, $level + 1, $plainText, $ansiColors);
121 }
122 }
123 return $dump;
124 }
125
126 /**
127 * Renders a dump of the given array
128 *
129 * @param array|\Traversable $array
130 * @param int $level
131 * @param bool $plainText
132 * @param bool $ansiColors
133 * @return string
134 */
135 protected static function renderArray($array, $level, $plainText = false, $ansiColors = false)
136 {
137 $content = '';
138 $count = count($array);
139
140 if ($plainText) {
141 $header = self::ansiEscapeWrap('array', '36', $ansiColors);
142 } else {
143 $header = '<span class="extbase-debug-type">array</span>';
144 }
145 $header .= $count > 0 ? '(' . $count . ' item' . ($count > 1 ? 's' : '') . ')' : '(empty)';
146 if ($level >= self::$maxDepth) {
147 if ($plainText) {
148 $header .= ' ' . self::ansiEscapeWrap('max depth', '47;30', $ansiColors);
149 } else {
150 $header .= '<span class="extbase-debug-filtered">max depth</span>';
151 }
152 } else {
153 $content = self::renderCollection($array, $level, $plainText, $ansiColors);
154 if (!$plainText) {
155 $header = ($level > 1 && $count > 0 ? '<input type="checkbox" /><span class="extbase-debug-header" >' : '<span>') . $header . '</span >';
156 }
157 }
158 if ($level > 1 && $count > 0 && !$plainText) {
159 $dump = '<span class="extbase-debugger-tree">' . $header . '<span class="extbase-debug-content">' . $content . '</span></span>';
160 } else {
161 $dump = $header . $content;
162 }
163 return $dump;
164 }
165
166 /**
167 * Renders a dump of the given object
168 *
169 * @param object $object
170 * @param int $level
171 * @param bool $plainText
172 * @param bool $ansiColors
173 * @return string
174 */
175 protected static function renderObject($object, $level, $plainText = false, $ansiColors = false)
176 {
177 if ($object instanceof \TYPO3\CMS\Extbase\Persistence\Generic\LazyLoadingProxy) {
178 $object = $object->_loadRealInstance();
179 if (!$object) {
180 return gettype($object);
181 }
182 }
183 $header = self::renderHeader($object, $level, $plainText, $ansiColors);
184 if ($level < self::$maxDepth && !self::isBlacklisted($object) && !(self::isAlreadyRendered($object) && $plainText !== true)) {
185 $content = self::renderContent($object, $level, $plainText, $ansiColors);
186 } else {
187 $content = '';
188 }
189 if ($plainText) {
190 return $header . $content;
191 }
192 return '<span class="extbase-debugger-tree">' . $header . '<span class="extbase-debug-content">' . $content . '</span></span>';
193 }
194
195 /**
196 * Renders a dump of the given closure
197 *
198 * @param \Closure $object
199 * @param int $level
200 * @param bool $plainText
201 * @param bool $ansiColors
202 * @return string
203 */
204 protected static function renderClosure($object, $level, $plainText = false, $ansiColors = false)
205 {
206 $header = self::renderHeader($object, $level, $plainText, $ansiColors);
207 if ($level < self::$maxDepth && (!self::isAlreadyRendered($object) || $plainText)) {
208 $content = self::renderContent($object, $level, $plainText, $ansiColors);
209 } else {
210 $content = '';
211 }
212 if ($plainText) {
213 return $header . $content;
214 }
215 return '<span class="extbase-debugger-tree"><input type="checkbox" /><span class="extbase-debug-header">' . $header . '</span><span class="extbase-debug-content">' . $content . '</span></span>';
216 }
217
218 /**
219 * Checks if a given object or property should be excluded/filtered
220 *
221 * @param object $value An ReflectionProperty or other Object
222 * @return bool TRUE if the given object should be filtered
223 */
224 protected static function isBlacklisted($value)
225 {
226 $result = false;
227 if ($value instanceof \ReflectionProperty) {
228 $result = in_array($value->getName(), self::$blacklistedPropertyNames, true);
229 } elseif (is_object($value)) {
230 $result = in_array(get_class($value), self::$blacklistedClassNames, true);
231 }
232 return $result;
233 }
234
235 /**
236 * Checks if a given object was already rendered.
237 *
238 * @param object $object
239 * @return bool TRUE if the given object was already rendered
240 */
241 protected static function isAlreadyRendered($object)
242 {
243 return self::$renderedObjects->contains($object);
244 }
245
246 /**
247 * Renders the header of a given object/collection. It is usually the class name along with some flags.
248 *
249 * @param object $object
250 * @param int $level
251 * @param bool $plainText
252 * @param bool $ansiColors
253 * @return string The rendered header with tags
254 */
255 protected static function renderHeader($object, $level, $plainText, $ansiColors)
256 {
257 $dump = '';
258 $persistenceType = '';
259 $className = get_class($object);
260 $classReflection = new \ReflectionClass($className);
261 if ($plainText) {
262 $dump .= self::ansiEscapeWrap($className, '36', $ansiColors);
263 } else {
264 $dump .= '<span class="extbase-debug-type">' . $className . '</span>';
265 }
266 if (! $object instanceof \Closure) {
267 if ($object instanceof \TYPO3\CMS\Core\SingletonInterface) {
268 $scope = 'singleton';
269 } else {
270 $scope = 'prototype';
271 }
272 if ($plainText) {
273 $dump .= ' ' . self::ansiEscapeWrap($scope, '44;37', $ansiColors);
274 } else {
275 $dump .= $scope ? '<span class="extbase-debug-scope">' . $scope . '</span>' : '';
276 }
277 if ($object instanceof \TYPO3\CMS\Extbase\DomainObject\AbstractDomainObject) {
278 if ($object->_isDirty()) {
279 $persistenceType = 'modified';
280 } elseif ($object->_isNew()) {
281 $persistenceType = 'transient';
282 } else {
283 $persistenceType = 'persistent';
284 }
285 }
286 if ($object instanceof \TYPO3\CMS\Extbase\Persistence\ObjectStorage && $object->_isDirty()) {
287 $persistenceType = 'modified';
288 }
289 if ($object instanceof \TYPO3\CMS\Extbase\DomainObject\AbstractEntity) {
290 $domainObjectType = 'entity';
291 } elseif ($object instanceof \TYPO3\CMS\Extbase\DomainObject\AbstractValueObject) {
292 $domainObjectType = 'valueobject';
293 } else {
294 $domainObjectType = 'object';
295 }
296 if ($plainText) {
297 $dump .= ' ' . self::ansiEscapeWrap((($persistenceType ? $persistenceType . ' ' : '') . $domainObjectType), '42;30', $ansiColors);
298 } else {
299 $dump .= '<span class="extbase-debug-ptype">' . ($persistenceType ? $persistenceType . ' ' : '') . $domainObjectType . '</span>';
300 }
301 }
302 if (strpos(implode('|', self::$blacklistedClassNames), get_class($object)) > 0) {
303 if ($plainText) {
304 $dump .= ' ' . self::ansiEscapeWrap('filtered', '47;30', $ansiColors);
305 } else {
306 $dump .= '<span class="extbase-debug-filtered">filtered</span>';
307 }
308 } elseif (self::$renderedObjects->contains($object) && !$plainText) {
309 $dump = '<a href="javascript:;" onclick="document.location.hash=\'#' . spl_object_hash($object) . '\';" class="extbase-debug-seeabove">' . $dump . '<span class="extbase-debug-filtered">see above</span></a>';
310 } elseif ($level >= self::$maxDepth && !$object instanceof \DateTime) {
311 if ($plainText) {
312 $dump .= ' ' . self::ansiEscapeWrap('max depth', '47;30', $ansiColors);
313 } else {
314 $dump .= '<span class="extbase-debug-filtered">max depth</span>';
315 }
316 } elseif ($level > 1 && !$object instanceof \DateTime && !$plainText) {
317 if (($object instanceof \Countable && empty($object)) || empty($classReflection->getProperties())) {
318 $dump = '<span>' . $dump . '</span>';
319 } else {
320 $dump = '<input type="checkbox" id="' . spl_object_hash($object) . '" /><span class="extbase-debug-header">' . $dump . '</span>';
321 }
322 }
323 if ($object instanceof \Countable) {
324 $objectCount = count($object);
325 $dump .= $objectCount > 0 ? ' (' . $objectCount . ' items)' : ' (empty)';
326 }
327 if ($object instanceof \DateTime) {
328 $dump .= ' (' . $object->format(\DateTime::RFC3339) . ', ' . $object->getTimestamp() . ')';
329 }
330 if ($object instanceof \TYPO3\CMS\Extbase\DomainObject\DomainObjectInterface && !$object->_isNew()) {
331 $dump .= ' (uid=' . $object->getUid() . ', pid=' . $object->getPid() . ')';
332 }
333 return $dump;
334 }
335
336 /**
337 * @param object $object
338 * @param int $level
339 * @param bool $plainText
340 * @param bool $ansiColors
341 * @return string The rendered body content of the Object(Storage)
342 */
343 protected static function renderContent($object, $level, $plainText, $ansiColors)
344 {
345 $dump = '';
346 if ($object instanceof \TYPO3\CMS\Extbase\Persistence\ObjectStorage || $object instanceof \Iterator || $object instanceof \ArrayObject) {
347 $dump .= self::renderCollection($object, $level, $plainText, $ansiColors);
348 } else {
349 self::$renderedObjects->attach($object);
350 if (!$plainText) {
351 $dump .= '<a name="' . spl_object_hash($object) . '" id="' . spl_object_hash($object) . '"></a>';
352 }
353 if ($object instanceof \Closure) {
354 $dump .= PHP_EOL . str_repeat(self::PLAINTEXT_INDENT, $level)
355 . ($plainText ? '' : '<span class="extbase-debug-closure">')
356 . self::ansiEscapeWrap('function (', '33', $ansiColors) . ($plainText ? '' : '</span>');
357
358 $reflectionFunction = new \ReflectionFunction($object);
359 $params = [];
360 foreach ($reflectionFunction->getParameters() as $parameter) {
361 $parameterDump = '';
362 if ($parameter->isArray()) {
363 if ($plainText) {
364 $parameterDump .= self::ansiEscapeWrap('array ', '36', $ansiColors);
365 } else {
366 $parameterDump .= '<span class="extbase-debug-type">array </span>';
367 }
368 } elseif ($parameter->getClass()) {
369 if ($plainText) {
370 $parameterDump .= self::ansiEscapeWrap($parameter->getClass()->name . ' ', '36', $ansiColors);
371 } else {
372 $parameterDump .= '<span class="extbase-debug-type">'
373 . htmlspecialchars($parameter->getClass()->name) . '</span>';
374 }
375 }
376 if ($parameter->isPassedByReference()) {
377 $parameterDump .= '&';
378 }
379 if ($plainText) {
380 $parameterDump .= self::ansiEscapeWrap('$' . $parameter->name, '37', $ansiColors);
381 } else {
382 $parameterDump .= '<span class="extbase-debug-property">'
383 . htmlspecialchars('$' . $parameter->name) . '</span>';
384 }
385 if ($parameter->isOptional()) {
386 $parameterDump .= ' = ';
387 if ($plainText) {
388 $parameterDump .= self::ansiEscapeWrap(var_export($parameter->getDefaultValue(), true), '33', $ansiColors);
389 } else {
390 $parameterDump .= '<span class="extbase-debug-string">'
391 . htmlspecialchars(var_export($parameter->getDefaultValue(), true)) . '</span>';
392 }
393 }
394 $params[] = $parameterDump;
395 }
396 $dump .= implode(', ', $params);
397 if ($plainText) {
398 $dump .= self::ansiEscapeWrap(') {' . PHP_EOL, '33', $ansiColors);
399 } else {
400 $dump .= '<span class="extbase-debug-closure">) {' . PHP_EOL . '</span>';
401 }
402 $lines = file($reflectionFunction->getFileName());
403 for ($l = $reflectionFunction->getStartLine(); $l < $reflectionFunction->getEndLine() -1; ++$l) {
404 $dump .= $plainText ? $lines[$l] : htmlspecialchars($lines[$l]);
405 }
406 $dump .= str_repeat(self::PLAINTEXT_INDENT, $level);
407 if ($plainText) {
408 $dump .= self::ansiEscapeWrap('}' . PHP_EOL, '33', $ansiColors);
409 } else {
410 $dump .= '<span class="extbase-debug-closure">}</span>';
411 }
412 } else {
413 if (get_class($object) === 'stdClass') {
414 $objReflection = new \ReflectionObject($object);
415 $properties = $objReflection->getProperties();
416 } else {
417 $classReflection = new \ReflectionClass(get_class($object));
418 $properties = $classReflection->getProperties();
419 }
420 foreach ($properties as $property) {
421 if (self::isBlacklisted($property)) {
422 continue;
423 }
424 $dump .= PHP_EOL . str_repeat(self::PLAINTEXT_INDENT, $level);
425 if ($plainText) {
426 $dump .= self::ansiEscapeWrap($property->getName(), '37', $ansiColors);
427 } else {
428 $dump .= '<span class="extbase-debug-property">'
429 . htmlspecialchars($property->getName()) . '</span>';
430 }
431 $dump .= ' => ';
432 $property->setAccessible(true);
433 $visibility = ($property->isProtected() ? 'protected' : ($property->isPrivate() ? 'private' : 'public'));
434 if ($plainText) {
435 $dump .= self::ansiEscapeWrap($visibility, '42;30', $ansiColors) . ' ';
436 } else {
437 $dump .= '<span class="extbase-debug-visibility">' . $visibility . '</span>';
438 }
439 $dump .= self::renderDump($property->getValue($object), $level, $plainText, $ansiColors);
440 if ($object instanceof \TYPO3\CMS\Extbase\DomainObject\AbstractDomainObject && !$object->_isNew() && $object->_isDirty($property->getName())) {
441 if ($plainText) {
442 $dump .= ' ' . self::ansiEscapeWrap('modified', '43;30', $ansiColors);
443 } else {
444 $dump .= '<span class="extbase-debug-dirty">modified</span>';
445 }
446 }
447 }
448 }
449 }
450 return $dump;
451 }
452
453 /**
454 * @param mixed $collection
455 * @param int $level
456 * @param bool $plainText
457 * @param bool $ansiColors
458 * @return string
459 */
460 protected static function renderCollection($collection, $level, $plainText, $ansiColors)
461 {
462 $dump = '';
463 foreach ($collection as $key => $value) {
464 $dump .= PHP_EOL . str_repeat(self::PLAINTEXT_INDENT, $level) . ($plainText ? '' : '<span class="extbase-debug-property">') . self::ansiEscapeWrap($key, '37', $ansiColors) . ($plainText ? '' : '</span>') . ' => ';
465 $dump .= self::renderDump($value, $level, $plainText, $ansiColors);
466 }
467 if ($collection instanceof \Iterator) {
468 $collection->rewind();
469 }
470 return $dump;
471 }
472
473 /**
474 * Wrap a string with the ANSI escape sequence for colorful output
475 *
476 * @param string $string The string to wrap
477 * @param string $ansiColors The ansi color sequence (e.g. "1;37")
478 * @param bool $enable If FALSE, the raw string will be returned
479 * @return string The wrapped or raw string
480 */
481 protected static function ansiEscapeWrap($string, $ansiColors, $enable = true)
482 {
483 if ($enable) {
484 return '\e[' . $ansiColors . 'm' . $string . '\e[0m';
485 }
486 return $string;
487 }
488
489 /**
490 * A var_dump function optimized for Extbase's object structures
491 *
492 * @param mixed $variable The value to dump
493 * @param string $title optional custom title for the debug output
494 * @param int $maxDepth Sets the max recursion depth of the dump. De- or increase the number according to your needs and memory limit.
495 * @param bool $plainText If TRUE, the dump is in plain text, if FALSE the debug output is in HTML format.
496 * @param bool $ansiColors If TRUE (default), ANSI color codes is added to the output, if FALSE the debug output not colored.
497 * @param bool $return if TRUE, the dump is returned for custom post-processing (e.g. embed in custom HTML). If FALSE (default), the dump is directly displayed.
498 * @param array $blacklistedClassNames An array of class names (RegEx) to be filtered. Default is an array of some common class names.
499 * @param array $blacklistedPropertyNames An array of property names and/or array keys (RegEx) to be filtered. Default is an array of some common property names.
500 * @return string if $return is TRUE, the dump is returned. By default, the dump is directly displayed, and nothing is returned.
501 * @api
502 */
503 public static function var_dump($variable, $title = null, $maxDepth = 8, $plainText = false, $ansiColors = true, $return = false, $blacklistedClassNames = null, $blacklistedPropertyNames = null)
504 {
505 self::$maxDepth = $maxDepth;
506 if ($title === null) {
507 $title = 'Extbase Variable Dump';
508 }
509 $ansiColors = $plainText && $ansiColors;
510 if ($ansiColors === true) {
511 $title = '\e[1m' . $title . '\e[0m';
512 }
513 $backupBlacklistedClassNames = self::$blacklistedClassNames;
514 if (is_array($blacklistedClassNames)) {
515 self::$blacklistedClassNames = $blacklistedClassNames;
516 }
517 $backupBlacklistedPropertyNames = self::$blacklistedPropertyNames;
518 if (is_array($blacklistedPropertyNames)) {
519 self::$blacklistedPropertyNames = $blacklistedPropertyNames;
520 }
521 self::clearState();
522 $css = '';
523 if (!$plainText && self::$stylesheetEchoed === false) {
524 $css = '
525 <style type=\'text/css\'>
526 .extbase-debugger-tree{position:relative}
527 .extbase-debugger-tree input{position:absolute !important;float: none !important;top:0;left:0;height:14px;width:14px;margin:0 !important;cursor:pointer;opacity:0;z-index:2}
528 .extbase-debugger-tree input~.extbase-debug-content{display:none}
529 .extbase-debugger-tree .extbase-debug-header:before{position:relative;top:3px;content:"";padding:0;line-height:10px;height:12px;width:12px;text-align:center;margin:0 3px 0 0;background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz48c3ZnIHZlcnNpb249IjEuMSIgaWQ9IkViZW5lXzEiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiIHg9IjBweCIgeT0iMHB4IiB2aWV3Qm94PSIwIDAgMTIgMTIiIHN0eWxlPSJlbmFibGUtYmFja2dyb3VuZDpuZXcgMCAwIDEyIDEyOyIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PHN0eWxlIHR5cGU9InRleHQvY3NzIj4uc3Qwe2ZpbGw6Izg4ODg4ODt9PC9zdHlsZT48cGF0aCBpZD0iQm9yZGVyIiBjbGFzcz0ic3QwIiBkPSJNMTEsMTFIMFYwaDExVjExeiBNMTAsMUgxdjloOVYxeiIvPjxnIGlkPSJJbm5lciI+PHJlY3QgeD0iMiIgeT0iNSIgY2xhc3M9InN0MCIgd2lkdGg9IjciIGhlaWdodD0iMSIvPjxyZWN0IHg9IjUiIHk9IjIiIGNsYXNzPSJzdDAiIHdpZHRoPSIxIiBoZWlnaHQ9IjciLz48L2c+PC9zdmc+);display:inline-block}
530 .extbase-debugger-tree input:checked~.extbase-debug-content{display:inline}
531 .extbase-debugger-tree input:checked~.extbase-debug-header:before{background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz48c3ZnIHZlcnNpb249IjEuMSIgaWQ9IkViZW5lXzEiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiIHg9IjBweCIgeT0iMHB4IiB2aWV3Qm94PSIwIDAgMTIgMTIiIHN0eWxlPSJlbmFibGUtYmFja2dyb3VuZDpuZXcgMCAwIDEyIDEyOyIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PHN0eWxlIHR5cGU9InRleHQvY3NzIj4uc3Qwe2ZpbGw6Izg4ODg4ODt9PC9zdHlsZT48cGF0aCBpZD0iQm9yZGVyIiBjbGFzcz0ic3QwIiBkPSJNMTEsMTFIMFYwaDExVjExeiBNMTAsMUgxdjloOVYxeiIvPjxnIGlkPSJJbm5lciI+PHJlY3QgeD0iMiIgeT0iNSIgY2xhc3M9InN0MCIgd2lkdGg9IjciIGhlaWdodD0iMSIvPjwvZz48L3N2Zz4=)}
532 .extbase-debugger{display:block;text-align:left;background:#2a2a2a;border:1px solid #2a2a2a;box-shadow:0 3px 0 rgba(0,0,0,.5);color:#000;margin:20px;overflow:hidden;border-radius:4px}
533 .extbase-debugger-floating{position:relative;z-index:999}
534 .extbase-debugger-top{background:#444;font-size:12px;font-family:monospace;color:#f1f1f1;padding:6px 15px}
535 .extbase-debugger-center{padding:0 15px;margin:15px 0;background-image:repeating-linear-gradient(to bottom,transparent 0,transparent 20px,#252525 20px,#252525 40px)}
536 .extbase-debugger-center,.extbase-debugger-center .extbase-debug-string,.extbase-debugger-center a,.extbase-debugger-center p,.extbase-debugger-center pre,.extbase-debugger-center strong{font-size:12px;font-weight:400;font-family:monospace;line-height:20px;color:#f1f1f1}
537 .extbase-debugger-center pre{background-color:transparent;margin:0;padding:0;border:0;word-wrap:break-word;color:#999}
538 .extbase-debugger-center .extbase-debug-string{color:#ce9178;white-space:normal}
539 .extbase-debugger-center .extbase-debug-type{color:#569CD6;padding-right:4px}
540 .extbase-debugger-center .extbase-debug-unregistered{background-color:#dce1e8}
541 .extbase-debugger-center .extbase-debug-filtered,.extbase-debugger-center .extbase-debug-proxy,.extbase-debugger-center .extbase-debug-ptype,.extbase-debugger-center .extbase-debug-visibility,.extbase-debugger-center .extbase-debug-scope{color:#fff;font-size:10px;line-height:12px;padding:2px 4px;margin-right:2px;position:relative;top:-1px}
542 .extbase-debugger-center .extbase-debug-scope{background-color:#497AA2}
543 .extbase-debugger-center .extbase-debug-ptype{background-color:#698747}
544 .extbase-debugger-center .extbase-debug-visibility{background-color:#698747}
545 .extbase-debugger-center .extbase-debug-dirty{background-color:#FFFFB6}
546 .extbase-debugger-center .extbase-debug-filtered{background-color:#4F4F4F}
547 .extbase-debugger-center .extbase-debug-seeabove{text-decoration:none;font-style:italic}
548 .extbase-debugger-center .extbase-debug-property{color:#f1f1f1}
549 .extbase-debugger-center .extbase-debug-closure{color:#9BA223;}
550 </style>';
551 self::$stylesheetEchoed = true;
552 }
553 if ($plainText) {
554 $output = $title . PHP_EOL . self::renderDump($variable, 0, true, $ansiColors) . PHP_EOL . PHP_EOL;
555 } else {
556 $output = '
557 <div class="extbase-debugger ' . ($return ? 'extbase-debugger-inline' : 'extbase-debugger-floating') . '">
558 <div class="extbase-debugger-top">' . htmlspecialchars($title) . '</div>
559 <div class="extbase-debugger-center">
560 <pre dir="ltr">' . self::renderDump($variable, 0, false, false) . '</pre>
561 </div>
562 </div>
563 ';
564 }
565 self::$blacklistedClassNames = $backupBlacklistedClassNames;
566 self::$blacklistedPropertyNames = $backupBlacklistedPropertyNames;
567 if ($return === true) {
568 return $css . $output;
569 }
570 echo $css . $output;
571
572 return '';
573 }
574 }