| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451 | <?php/* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */namespace Symfony\Component\VarDumper\Dumper;use Symfony\Component\VarDumper\Cloner\Cursor;use Symfony\Component\VarDumper\Cloner\Data;/** * HtmlDumper dumps variables as HTML. * * @author Nicolas Grekas <p@tchwork.com> */class HtmlDumper extends CliDumper{    public static $defaultOutput = 'php://output';    protected $dumpHeader;    protected $dumpPrefix = '<pre class=sf-dump id=%s data-indent-pad="%s">';    protected $dumpSuffix = '</pre><script>Sfdump("%s")</script>';    protected $dumpId = 'sf-dump';    protected $colors = true;    protected $headerIsDumped = false;    protected $lastDepth = -1;    protected $styles = array(        'default' => 'background-color:#18171B; color:#FF8400; line-height:1.2em; font:12px Menlo, Monaco, Consolas, monospace; word-wrap: break-word; white-space: pre-wrap; position:relative; z-index:99999; word-break: normal',        'num' => 'font-weight:bold; color:#1299DA',        'const' => 'font-weight:bold',        'str' => 'font-weight:bold; color:#56DB3A',        'note' => 'color:#1299DA',        'ref' => 'color:#A0A0A0',        'public' => 'color:#FFFFFF',        'protected' => 'color:#FFFFFF',        'private' => 'color:#FFFFFF',        'meta' => 'color:#B729D9',        'key' => 'color:#56DB3A',        'index' => 'color:#1299DA',    );    /**     * {@inheritdoc}     */    public function __construct($output = null, $charset = null)    {        AbstractDumper::__construct($output, $charset);        $this->dumpId = 'sf-dump-'.mt_rand();    }    /**     * {@inheritdoc}     */    public function setStyles(array $styles)    {        $this->headerIsDumped = false;        $this->styles = $styles + $this->styles;    }    /**     * Sets an HTML header that will be dumped once in the output stream.     *     * @param string $header An HTML string     */    public function setDumpHeader($header)    {        $this->dumpHeader = $header;    }    /**     * Sets an HTML prefix and suffix that will encapse every single dump.     *     * @param string $prefix The prepended HTML string     * @param string $suffix The appended HTML string     */    public function setDumpBoundaries($prefix, $suffix)    {        $this->dumpPrefix = $prefix;        $this->dumpSuffix = $suffix;    }    /**     * {@inheritdoc}     */    public function dump(Data $data, $output = null)    {        parent::dump($data, $output);        $this->dumpId = 'sf-dump-'.mt_rand();    }    /**     * Dumps the HTML header.     */    protected function getDumpHeader()    {        $this->headerIsDumped = null !== $this->outputStream ? $this->outputStream : $this->lineDumper;        if (null !== $this->dumpHeader) {            return $this->dumpHeader;        }        $line = <<<'EOHTML'<script>Sfdump = window.Sfdump || (function (doc) {var refStyle = doc.createElement('style'),    rxEsc = /([.*+?^${}()|\[\]\/\\])/g,    idRx = /\bsf-dump-\d+-ref[012]\w+\b/,    keyHint = 0 <= navigator.platform.toUpperCase().indexOf('MAC') ? 'Cmd' : 'Ctrl',    addEventListener = function (e, n, cb) {        e.addEventListener(n, cb, false);    };(doc.documentElement.firstElementChild || doc.documentElement.children[0]).appendChild(refStyle);if (!doc.addEventListener) {    addEventListener = function (element, eventName, callback) {        element.attachEvent('on' + eventName, function (e) {            e.preventDefault = function () {e.returnValue = false;};            e.target = e.srcElement;            callback(e);        });    };}function toggle(a, recursive) {    var s = a.nextSibling || {}, oldClass = s.className, arrow, newClass;    if (/\bsf-dump-compact\b/.test(oldClass)) {        arrow = '▼';        newClass = 'sf-dump-expanded';    } else if (/\bsf-dump-expanded\b/.test(oldClass)) {        arrow = '▶';        newClass = 'sf-dump-compact';    } else {        return false;    }    if (doc.createEvent && s.dispatchEvent) {        var event = doc.createEvent('Event');        event.initEvent('sf-dump-expanded' === newClass ? 'sfbeforedumpexpand' : 'sfbeforedumpcollapse', true, false);        s.dispatchEvent(event);    }    a.lastChild.innerHTML = arrow;    s.className = s.className.replace(/\bsf-dump-(compact|expanded)\b/, newClass);    if (recursive) {        try {            a = s.querySelectorAll('.'+oldClass);            for (s = 0; s < a.length; ++s) {                if (-1 == a[s].className.indexOf(newClass)) {                    a[s].className = newClass;                    a[s].previousSibling.lastChild.innerHTML = arrow;                }            }        } catch (e) {        }    }    return true;};return function (root) {    root = doc.getElementById(root);    function a(e, f) {        addEventListener(root, e, function (e) {            if ('A' == e.target.tagName) {                f(e.target, e);            } else if ('A' == e.target.parentNode.tagName) {                f(e.target.parentNode, e);            }        });    };    function isCtrlKey(e) {        return e.ctrlKey || e.metaKey;    }    addEventListener(root, 'mouseover', function (e) {        if ('' != refStyle.innerHTML) {            refStyle.innerHTML = '';        }    });    a('mouseover', function (a) {        if (a = idRx.exec(a.className)) {            try {                refStyle.innerHTML = 'pre.sf-dump .'+a[0]+'{background-color: #B729D9; color: #FFF !important; border-radius: 2px}';            } catch (e) {            }        }    });    a('click', function (a, e) {        if (/\bsf-dump-toggle\b/.test(a.className)) {            e.preventDefault();            if (!toggle(a, isCtrlKey(e))) {                var r = doc.getElementById(a.getAttribute('href').substr(1)),                    s = r.previousSibling,                    f = r.parentNode,                    t = a.parentNode;                t.replaceChild(r, a);                f.replaceChild(a, s);                t.insertBefore(s, r);                f = f.firstChild.nodeValue.match(indentRx);                t = t.firstChild.nodeValue.match(indentRx);                if (f && t && f[0] !== t[0]) {                    r.innerHTML = r.innerHTML.replace(new RegExp('^'+f[0].replace(rxEsc, '\\$1'), 'mg'), t[0]);                }                if (/\bsf-dump-compact\b/.test(r.className)) {                    toggle(s, isCtrlKey(e));                }            }            if (doc.getSelection) {                try {                    doc.getSelection().removeAllRanges();                } catch (e) {                    doc.getSelection().empty();                }            } else {                doc.selection.empty();            }        }    });    var indentRx = new RegExp('^('+(root.getAttribute('data-indent-pad') || '  ').replace(rxEsc, '\\$1')+')+', 'm'),        elt = root.getElementsByTagName('A'),        len = elt.length,        i = 0,        t = [];    while (i < len) t.push(elt[i++]);    elt = root.getElementsByTagName('SAMP');    len = elt.length;    i = 0;    while (i < len) t.push(elt[i++]);    root = t;    len = t.length;    i = t = 0;    while (i < len) {        elt = root[i];        if ("SAMP" == elt.tagName) {            elt.className = "sf-dump-expanded";            a = elt.previousSibling || {};            if ('A' != a.tagName) {                a = doc.createElement('A');                a.className = 'sf-dump-ref';                elt.parentNode.insertBefore(a, elt);            } else {                a.innerHTML += ' ';            }            a.title = (a.title ? a.title+'\n[' : '[')+keyHint+'+click] Expand all children';            a.innerHTML += '<span>▼</span>';            a.className += ' sf-dump-toggle';            if (!/\bsf-dump\b/.test(elt.parentNode.className)) {                toggle(a);            }        } else if (/\bsf-dump-ref\b/.test(elt.className) && (a = elt.getAttribute('href'))) {            a = a.substr(1);            elt.className += ' '+a;            if (/[\[{]$/.test(elt.previousSibling.nodeValue)) {                a = a != elt.nextSibling.id && doc.getElementById(a);                try {                    t = a.nextSibling;                    elt.appendChild(a);                    t.parentNode.insertBefore(a, t);                    if (/^[@#]/.test(elt.innerHTML)) {                        elt.innerHTML += ' <span>▶</span>';                    } else {                        elt.innerHTML = '<span>▶</span>';                        elt.className = 'sf-dump-ref';                    }                    elt.className += ' sf-dump-toggle';                } catch (e) {                    if ('&' == elt.innerHTML.charAt(0)) {                        elt.innerHTML = '…';                        elt.className = 'sf-dump-ref';                    }                }            }        }        ++i;    }};})(document);</script><style>pre.sf-dump {    display: block;    white-space: pre;    padding: 5px;}pre.sf-dump span {    display: inline;}pre.sf-dump .sf-dump-compact {    display: none;}pre.sf-dump abbr {    text-decoration: none;    border: none;    cursor: help;}pre.sf-dump a {    text-decoration: none;    cursor: pointer;    border: 0;    outline: none;}EOHTML;        foreach ($this->styles as $class => $style) {            $line .= 'pre.sf-dump'.('default' !== $class ? ' .sf-dump-'.$class : '').'{'.$style.'}';        }        return $this->dumpHeader = preg_replace('/\s+/', ' ', $line).'</style>'.$this->dumpHeader;    }    /**     * {@inheritdoc}     */    public function enterHash(Cursor $cursor, $type, $class, $hasChild)    {        parent::enterHash($cursor, $type, $class, false);        if ($hasChild) {            if ($cursor->refIndex) {                $r = Cursor::HASH_OBJECT !== $type ? 1 - (Cursor::HASH_RESOURCE !== $type) : 2;                $r .= $r && 0 < $cursor->softRefHandle ? $cursor->softRefHandle : $cursor->refIndex;                $this->line .= sprintf('<samp id=%s-ref%s>', $this->dumpId, $r);            } else {                $this->line .= '<samp>';            }            $this->dumpLine($cursor->depth);        }    }    /**     * {@inheritdoc}     */    public function leaveHash(Cursor $cursor, $type, $class, $hasChild, $cut)    {        $this->dumpEllipsis($cursor, $hasChild, $cut);        if ($hasChild) {            $this->line .= '</samp>';        }        parent::leaveHash($cursor, $type, $class, $hasChild, 0);    }    /**     * {@inheritdoc}     */    protected function style($style, $value, $attr = array())    {        if ('' === $value) {            return '';        }        $v = esc($value);        if ('ref' === $style) {            if (empty($attr['count'])) {                return sprintf('<a class=sf-dump-ref>%s</a>', $v);            }            $r = ('#' !== $v[0] ? 1 - ('@' !== $v[0]) : 2).substr($value, 1);            return sprintf('<a class=sf-dump-ref href=#%s-ref%s title="%d occurrences">%s</a>', $this->dumpId, $r, 1 + $attr['count'], $v);        }        if ('const' === $style && isset($attr['value'])) {            $style .= sprintf(' title="%s"', esc(is_scalar($attr['value']) ? $attr['value'] : json_encode($attr['value'])));        } elseif ('public' === $style) {            $style .= sprintf(' title="%s"', empty($attr['dynamic']) ? 'Public property' : 'Runtime added dynamic property');        } elseif ('str' === $style && 1 < $attr['length']) {            $style .= sprintf(' title="%d%s characters"', $attr['length'], $attr['binary'] ? ' binary or non-UTF-8' : '');        } elseif ('note' === $style && false !== $c = strrpos($v, '\\')) {            return sprintf('<abbr title="%s" class=sf-dump-%s>%s</abbr>', $v, $style, substr($v, $c + 1));        } elseif ('protected' === $style) {            $style .= ' title="Protected property"';        } elseif ('private' === $style) {            $style .= sprintf(' title="Private property defined in class:
`%s`"', esc($attr['class']));        }        $map = static::$controlCharsMap;        $style = "<span class=sf-dump-{$style}>";        $v = preg_replace_callback(static::$controlCharsRx, function ($c) use ($map, $style) {            $s = '</span>';            $c = $c[$i = 0];            do {                $s .= isset($map[$c[$i]]) ? $map[$c[$i]] : sprintf('\x%02X', ord($c[$i]));            } while (isset($c[++$i]));            return $s.$style;        }, $v, -1, $cchrCount);        if ($cchrCount && '<' === $v[0]) {            $v = substr($v, 7);        } else {            $v = $style.$v;        }        if ($cchrCount && '>' === substr($v, -1)) {            $v = substr($v, 0, -strlen($style));        } else {            $v .= '</span>';        }        return $v;    }    /**     * {@inheritdoc}     */    protected function dumpLine($depth, $endOfValue = false)    {        if (-1 === $this->lastDepth) {            $this->line = sprintf($this->dumpPrefix, $this->dumpId, $this->indentPad).$this->line;        }        if ($this->headerIsDumped !== (null !== $this->outputStream ? $this->outputStream : $this->lineDumper)) {            $this->line = $this->getDumpHeader().$this->line;        }        if (-1 === $depth) {            $this->line .= sprintf($this->dumpSuffix, $this->dumpId);        }        $this->lastDepth = $depth;        $this->line = mb_convert_encoding($this->line, 'HTML-ENTITIES', 'UTF-8');        if (-1 === $depth) {            AbstractDumper::dumpLine(0);        }        AbstractDumper::dumpLine($depth);    }}function esc($str){    return htmlspecialchars($str, ENT_QUOTES, 'UTF-8');}
 |