aboutsummaryrefslogtreecommitdiff
path: root/js/lib/xml.js
blob: e7c71b0738a8b180170ba675c6bba6e3f2c22be5 (plain)
  1. "use strict";
  2. // Helper function to produce an XML tag.
  3. var tag = function(name, attrs, selfclosing) {
  4. var result = '<' + name;
  5. if (attrs && attrs.length > 0) {
  6. var i = 0;
  7. var attrib;
  8. while ((attrib = attrs[i]) !== undefined) {
  9. result = result.concat(' ', attrib[0], '="', attrib[1], '"');
  10. i++;
  11. }
  12. }
  13. if (selfclosing) {
  14. result += ' /';
  15. }
  16. result += '>';
  17. return result;
  18. };
  19. var reXMLTag = /\<[^>]*\>/;
  20. var renderNodes = function(block) {
  21. var attrs;
  22. var info_words;
  23. var tagname;
  24. var walker = block.walker();
  25. var event, node, entering;
  26. var buffer = "";
  27. var lastOut = "\n";
  28. var disableTags = 0;
  29. var indentLevel = 0;
  30. var indent = ' ';
  31. var grandparent;
  32. var unescapedContents;
  33. var container;
  34. var selfClosing;
  35. var out = function(s) {
  36. if (disableTags > 0) {
  37. buffer += s.replace(reXMLTag, '');
  38. } else {
  39. buffer += s;
  40. }
  41. lastOut = s;
  42. };
  43. var esc = this.escape;
  44. var cr = function() {
  45. if (lastOut !== '\n') {
  46. buffer += '\n';
  47. lastOut = '\n';
  48. for (var i = indentLevel; i--;) {
  49. buffer += indent;
  50. }
  51. }
  52. };
  53. var options = this.options;
  54. if (options.time) { console.time("rendering"); }
  55. while ((event = walker.next())) {
  56. entering = event.entering;
  57. node = event.node;
  58. container = node.isContainer();
  59. selfClosing = node.t === 'HorizontalRule' || node.t === 'Hardbreak' ||
  60. node.t === 'Softbreak' || node.t === 'Image';
  61. unescapedContents = node.t === 'Html' || node.t === 'HtmlInline';
  62. tagname = node.t.toLowerCase();
  63. attrs = [];
  64. if (options.sourcepos) {
  65. var pos = node.sourcepos;
  66. if (pos) {
  67. attrs.push(['data-sourcepos', String(pos[0][0]) + ':' +
  68. String(pos[0][1]) + '-' + String(pos[1][0]) + ':' +
  69. String(pos[1][1])]);
  70. }
  71. }
  72. if (entering) {
  73. cr();
  74. out(tag(tagname, attrs, selfClosing));
  75. if (container) {
  76. indentLevel += 1;
  77. } else if (!container && !selfClosing) {
  78. if (node.literal) {
  79. out(unescapedContents ? node.literal : esc(node.literal));
  80. }
  81. out(tag('/' + tagname));
  82. }
  83. } else {
  84. indentLevel -= 1;
  85. cr();
  86. out(tag('/' + tagname));
  87. }
  88. }
  89. if (options.time) { console.timeEnd("rendering"); }
  90. return buffer;
  91. };
  92. var replaceUnsafeChar = function(s) {
  93. switch (s) {
  94. case '&':
  95. return '&amp;';
  96. case '<':
  97. return '&lt;';
  98. case '>':
  99. return '&gt;';
  100. case '"':
  101. return '&quot;';
  102. default:
  103. return s;
  104. }
  105. };
  106. var reNeedsEscaping = /[&<>"]/;
  107. // The XMLRenderer object.
  108. function XMLRenderer(options){
  109. return {
  110. // default options:
  111. softbreak: '\n', // by default, soft breaks are rendered as newlines in HTML
  112. // set to "<br />" to make them hard breaks
  113. // set to " " if you want to ignore line wrapping in source
  114. escape: function(s, preserve_entities) {
  115. if (reNeedsEscaping.test(s)) {
  116. if (preserve_entities) {
  117. return s.replace(/[&](?:[#](x[a-f0-9]{1,8}|[0-9]{1,8});|[a-z][a-z0-9]{1,31};)|[&<>"]/gi, replaceUnsafeChar);
  118. } else {
  119. return s.replace(/[&<>"]/g, replaceUnsafeChar);
  120. }
  121. } else {
  122. return s;
  123. }
  124. },
  125. options: options || {},
  126. render: renderNodes
  127. };
  128. }
  129. module.exports = XMLRenderer;