用javascript打印XML

Dar*_*rov 123 javascript xml xslt pretty-print

我有一个字符串,表示我想要打印的非缩进XML.例如:

<root><node/></root>
Run Code Online (Sandbox Code Playgroud)

应成为:

<root>
  <node/>
</root>
Run Code Online (Sandbox Code Playgroud)

语法突出显示不是必需的.为了解决这个问题,我首先转换XML以添加回车符和空格,然后使用pre标签输出XML.为了添加新行和空格,我编写了以下函数:

function formatXml(xml) {
    var formatted = '';
    var reg = /(>)(<)(\/*)/g;
    xml = xml.replace(reg, '$1\r\n$2$3');
    var pad = 0;
    jQuery.each(xml.split('\r\n'), function(index, node) {
        var indent = 0;
        if (node.match( /.+<\/\w[^>]*>$/ )) {
            indent = 0;
        } else if (node.match( /^<\/\w/ )) {
            if (pad != 0) {
                pad -= 1;
            }
        } else if (node.match( /^<\w[^>]*[^\/]>.*$/ )) {
            indent = 1;
        } else {
            indent = 0;
        }

        var padding = '';
        for (var i = 0; i < pad; i++) {
            padding += '  ';
        }

        formatted += padding + node + '\r\n';
        pad += indent;
    });

    return formatted;
}
Run Code Online (Sandbox Code Playgroud)

然后我调用这个函数:

jQuery('pre.formatted-xml').text(formatXml('<root><node1/></root>'));
Run Code Online (Sandbox Code Playgroud)

这对我来说非常好,但在我写前一个函数时,我认为必须有更好的方法.所以我的问题是,你知道有什么更好的方法给XML字符串在html页面中漂亮打印吗?任何可以完成这项工作的javascript框架和/或插件都是受欢迎的.我唯一的要求就是在客户端完成.

Dim*_*hev 56

从问题的文本中我得到的结果是预期字符串结果,而不是HTML格式的结果.

如果是这样,最简单的方法是使用标识转换<xsl:output indent="yes"/>指令处理XML文档:

<xsl:stylesheet version="1.0"
 xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
 <xsl:output omit-xml-declaration="yes" indent="yes"/>

    <xsl:template match="node()|@*">
      <xsl:copy>
        <xsl:apply-templates select="node()|@*"/>
      </xsl:copy>
    </xsl:template>
</xsl:stylesheet>

在提供的XML文档上应用此转换时:

<root><node/></root>

大多数XSLT处理器(.NET XslCompiledTransform,Saxon 6.5.4和Saxon 9.0.0.2,AltovaXML)产生想要的结果:

<root>
  <node />
</root>

  • @ablmf:什么"不起作用"?什么是"Chrome"?我从来没有听说过这样的XSLT处理器.此外,如果您查看答案的日期,那么Chrome浏览器当时就不存在了. (5认同)
  • 它看起来像一个很好的解决方案 是否有任何跨浏览器方式在javascript中应用此转换?我没有可依赖的服务器端脚本. (3认同)
  • @ablmf:还要注意这个问题(以及我对它的回答)是将漂亮的XML作为字符串(文本)而不是HTML.难怪这样的字符串不会在浏览器中显示.对于精美的HTML输出(ala IE XML显示),请参阅XPath Visualizer中使用的XSLT转换.您可以在http://huttar.net/dimitre/XPV/TopXML-XPV.html下载XPath Visualizer.您可能需要稍微调整一下代码(例如删除用于折叠/展开节点的javascript扩展函数),否则生成的HTML应该显示正常. (3认同)
  • 是.看看Sarissa:http://dev.abiss.gr/sarissa/,在这里:http://www.xml.com/pub/a/2005/02/23/sarissa.html (2认同)
  • 最初的问题要求使用 javascript 的方法。如何获得这个答案以使用 javascript? (2认同)
  • JohnK,2008 年,当这个问题得到解答时,人们正在 IE 中从 JavaScript 启动 XSLT 转换——调用 MSXML3。现在他们仍然可以这样做,尽管 IE11 附带的 XSLT 处理器是 MSXML6。所有其他浏览器都具有类似的功能,尽管它们具有不同的内置 XSLT 处理器。这就是为什么最初的提问者从未提出过这样的问题。 (2认同)

小智 31

轻微修改efnx clckclcks的javascript函数.我将格式从空格更改为制表符,但最重要的是我允许文本保留在一行:

var formatXml = this.formatXml = function (xml) {
        var reg = /(>)\s*(<)(\/*)/g; // updated Mar 30, 2015
        var wsexp = / *(.*) +\n/g;
        var contexp = /(<.+>)(.+\n)/g;
        xml = xml.replace(reg, '$1\n$2$3').replace(wsexp, '$1\n').replace(contexp, '$1\n$2');
        var pad = 0;
        var formatted = '';
        var lines = xml.split('\n');
        var indent = 0;
        var lastType = 'other';
        // 4 types of tags - single, closing, opening, other (text, doctype, comment) - 4*4 = 16 transitions 
        var transitions = {
            'single->single': 0,
            'single->closing': -1,
            'single->opening': 0,
            'single->other': 0,
            'closing->single': 0,
            'closing->closing': -1,
            'closing->opening': 0,
            'closing->other': 0,
            'opening->single': 1,
            'opening->closing': 0,
            'opening->opening': 1,
            'opening->other': 1,
            'other->single': 0,
            'other->closing': -1,
            'other->opening': 0,
            'other->other': 0
        };

        for (var i = 0; i < lines.length; i++) {
            var ln = lines[i];

            // Luca Viggiani 2017-07-03: handle optional <?xml ... ?> declaration
            if (ln.match(/\s*<\?xml/)) {
                formatted += ln + "\n";
                continue;
            }
            // ---

            var single = Boolean(ln.match(/<.+\/>/)); // is this line a single tag? ex. <br />
            var closing = Boolean(ln.match(/<\/.+>/)); // is this a closing tag? ex. </a>
            var opening = Boolean(ln.match(/<[^!].*>/)); // is this even a tag (that's not <!something>)
            var type = single ? 'single' : closing ? 'closing' : opening ? 'opening' : 'other';
            var fromTo = lastType + '->' + type;
            lastType = type;
            var padding = '';

            indent += transitions[fromTo];
            for (var j = 0; j < indent; j++) {
                padding += '\t';
            }
            if (fromTo == 'opening->closing')
                formatted = formatted.substr(0, formatted.length - 1) + ln + '\n'; // substr removes line break (\n) from prev loop
            else
                formatted += padding + ln + '\n';
        }

        return formatted;
    };
Run Code Online (Sandbox Code Playgroud)


Art*_*sun 25

这可以使用原生的javascript工具完成,没有第三方库,扩展了@Dimitre Novatchev的答案:

var prettifyXml = function(sourceXml)
{
    var xmlDoc = new DOMParser().parseFromString(sourceXml, 'application/xml');
    var xsltDoc = new DOMParser().parseFromString([
        // describes how we want to modify the XML - indent everything
        '<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform">',
        '  <xsl:strip-space elements="*"/>',
        '  <xsl:template match="para[content-style][not(text())]">', // change to just text() to strip space in text nodes
        '    <xsl:value-of select="normalize-space(.)"/>',
        '  </xsl:template>',
        '  <xsl:template match="node()|@*">',
        '    <xsl:copy><xsl:apply-templates select="node()|@*"/></xsl:copy>',
        '  </xsl:template>',
        '  <xsl:output indent="yes"/>',
        '</xsl:stylesheet>',
    ].join('\n'), 'application/xml');

    var xsltProcessor = new XSLTProcessor();    
    xsltProcessor.importStylesheet(xsltDoc);
    var resultDoc = xsltProcessor.transformToDocument(xmlDoc);
    var resultXml = new XMLSerializer().serializeToString(resultDoc);
    return resultXml;
};

console.log(prettifyXml('<root><node/></root>'));
Run Code Online (Sandbox Code Playgroud)

输出:

<root>
  <node/>
</root>
Run Code Online (Sandbox Code Playgroud)

的jsfiddle

  • 这是在以下位置讨论的:/sf/ask/3639290511/ 显然,Firefox 需要 xsl 的版本规范,但这并不重要因为 Mozilla 实现不尊重任何“xsl:output”标记,因此无论如何您都不会获得良好的格式。 (5认同)
  • 我收到一个错误,但是该错误没有消息。使用Firefox,它也发生在小提琴中。 (3认同)
  • 这对我来说也不起作用,在 Firefox 中出现空白错误 (2认同)

Tou*_*ouv 19

Personnaly,我使​​用google-code-prettify这个函数:

prettyPrintOne('<root><node1><root>', 'xml')
Run Code Online (Sandbox Code Playgroud)

  • Oups,你需要缩进XML和google-code-prettify只对代码着色.抱歉. (3认同)
  • 这与http://code.google.com/p/vkbeautify/结合使用了一个很好的组合缩进. (3认同)

arc*_*rus 13

当我有类似的要求时发现这个线程但我简化了OP的代码如下:

function formatXml(xml, tab) { // tab = optional indent value, default is tab (\t)
    var formatted = '', indent= '';
    tab = tab || '\t';
    xml.split(/>\s*</).forEach(function(node) {
        if (node.match( /^\/\w/ )) indent = indent.substring(tab.length); // decrease indent by one 'tab'
        formatted += indent + '<' + node + '>\r\n';
        if (node.match( /^<?\w[^>]*[^\/]$/ )) indent += tab;              // increase indent
    });
    return formatted.substring(1, formatted.length-3);
}
Run Code Online (Sandbox Code Playgroud)

适合我!

  • 建议[this](https://jsfiddle.net/fbn5j7ya/)编辑,这几乎快了两倍。功能循环和正则表达式[慢](https://jsbench.me/zhkm1virni/3) (3认同)

sch*_*san 8

或者如果你只是喜欢另一个js函数来做,我已经修改了Darin(很多):

var formatXml = this.formatXml = function (xml) {
    var reg = /(>)(<)(\/*)/g;
    var wsexp = / *(.*) +\n/g;
    var contexp = /(<.+>)(.+\n)/g;
    xml = xml.replace(reg, '$1\n$2$3').replace(wsexp, '$1\n').replace(contexp, '$1\n$2');
    var pad = 0;
    var formatted = '';
    var lines = xml.split('\n');
    var indent = 0;
    var lastType = 'other';
    // 4 types of tags - single, closing, opening, other (text, doctype, comment) - 4*4 = 16 transitions 
    var transitions = {
        'single->single'    : 0,
        'single->closing'   : -1,
        'single->opening'   : 0,
        'single->other'     : 0,
        'closing->single'   : 0,
        'closing->closing'  : -1,
        'closing->opening'  : 0,
        'closing->other'    : 0,
        'opening->single'   : 1,
        'opening->closing'  : 0, 
        'opening->opening'  : 1,
        'opening->other'    : 1,
        'other->single'     : 0,
        'other->closing'    : -1,
        'other->opening'    : 0,
        'other->other'      : 0
    };

    for (var i=0; i < lines.length; i++) {
        var ln = lines[i];
        var single = Boolean(ln.match(/<.+\/>/)); // is this line a single tag? ex. <br />
        var closing = Boolean(ln.match(/<\/.+>/)); // is this a closing tag? ex. </a>
        var opening = Boolean(ln.match(/<[^!].*>/)); // is this even a tag (that's not <!something>)
        var type = single ? 'single' : closing ? 'closing' : opening ? 'opening' : 'other';
        var fromTo = lastType + '->' + type;
        lastType = type;
        var padding = '';

        indent += transitions[fromTo];
        for (var j = 0; j < indent; j++) {
            padding += '    ';
        }

        formatted += padding + ln + '\n';
    }

    return formatted;
};
Run Code Online (Sandbox Code Playgroud)


Chu*_* Ma 6

此处给出的所有javascript函数都不适用于在结束标记">"和开始标记"<"之间具有未指定空格的xml文档.要修复它们,您只需要替换函数中的第一行

var reg = /(>)(<)(\/*)/g;
Run Code Online (Sandbox Code Playgroud)

通过

var reg = /(>)\s*(<)(\/*)/g;
Run Code Online (Sandbox Code Playgroud)


Hue*_*fel 5

对于当前的项目,我需要在不使用额外库的情况下对 XML 进行美化和着色。下面的自包含代码运行得很好。

function formatXml(xml,colorize,indent) { 
  function esc(s){return s.replace(/[-\/&<> ]/g,function(c){         // Escape special chars
    return c==' '?'&nbsp;':'&#'+c.charCodeAt(0)+';';});}            
  var sm='<div class="xmt">',se='<div class="xel">',sd='<div class="xdt">',
      sa='<div class="xat">',tb='<div class="xtb">',tc='<div class="xtc">',
      ind=indent||'  ',sz='</div>',tz='</div>',re='',is='',ib,ob,at,i;
  if (!colorize) sm=se=sd=sa=sz='';   
  xml.match(/(?<=<).*(?=>)|$/s)[0].split(/>\s*</).forEach(function(nd){
    ob=('<'+nd+'>').match(/^(<[!?\/]?)(.*?)([?\/]?>)$/s);             // Split outer brackets
    ib=ob[2].match(/^(.*?)>(.*)<\/(.*)$/s)||['',ob[2],''];            // Split inner brackets 
    at=ib[1].match(/^--.*--$|=|('|").*?\1|[^\t\n\f \/>"'=]+/g)||['']; // Split attributes
    if (ob[1]=='</') is=is.substring(ind.length);                     // Decrease indent
    re+=tb+tc+esc(is)+tz+tc+sm+esc(ob[1])+sz+se+esc(at[0])+sz;
    for (i=1;i<at.length;i++) re+=(at[i]=="="?sm+"="+sz+sd+esc(at[++i]):sa+' '+at[i])+sz;
    re+=ib[2]?sm+esc('>')+sz+sd+esc(ib[2])+sz+sm+esc('</')+sz+se+ib[3]+sz:'';
    re+=sm+esc(ob[3])+sz+tz+tz;
    if (ob[1]+ob[3]+ib[2]=='<>') is+=ind;                             // Increase indent
  });
  return re;
}
Run Code Online (Sandbox Code Playgroud)

参见https://jsfiddle.net/dkb0La16/