/src/3rdparty/webkit/Source/WebCore/xml/XMLViewer.js

https://bitbucket.org/ultra_iter/qt-vtl · JavaScript · 434 lines · 331 code · 67 blank · 36 comment · 29 complexity · 9961ec218c5b40fb08dc424844b5482b MD5 · raw file

  1. /*
  2. * Copyright (C) 2011 Google Inc. All rights reserved.
  3. *
  4. * Redistribution and use in source and binary forms, with or without
  5. * modification, are permitted provided that the following conditions are
  6. * met:
  7. *
  8. * 1. Redistributions of source code must retain the above copyright
  9. * notice, this list of conditions and the following disclaimer.
  10. *
  11. * 2. Redistributions in binary form must reproduce the above
  12. * copyright notice, this list of conditions and the following disclaimer
  13. * in the documentation and/or other materials provided with the
  14. * distribution.
  15. *
  16. * THIS SOFTWARE IS PROVIDED BY GOOGLE INC. AND ITS CONTRIBUTORS
  17. * “AS IS” AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  18. * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  19. * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL GOOGLE INC.
  20. * OR ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  21. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  22. * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  23. * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  24. * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  25. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  26. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  27. */
  28. var nodeParentPairs = [];
  29. // Script entry point.
  30. function prepareWebKitXMLViewer(noStyleMessage)
  31. {
  32. var html = createHTMLElement('html');
  33. var head = createHTMLElement('head');
  34. html.appendChild(head);
  35. var style = createHTMLElement('style');
  36. style.id = 'xml-viewer-style';
  37. head.appendChild(style);
  38. var body = createHTMLElement('body');
  39. html.appendChild(body);
  40. var sourceXML = createHTMLElement('div');
  41. sourceXML.id = 'webkit-xml-viewer-source-xml';
  42. body.appendChild(sourceXML);
  43. var child;
  44. while (child = document.firstChild) {
  45. document.removeChild(child);
  46. if (child.nodeType != Node.DOCUMENT_TYPE_NODE)
  47. sourceXML.appendChild(child);
  48. }
  49. document.appendChild(html);
  50. var header = createHTMLElement('div');
  51. body.appendChild(header);
  52. header.classList.add('header');
  53. var headerSpan = createHTMLElement('span');
  54. header.appendChild(headerSpan);
  55. headerSpan.textContent = noStyleMessage;
  56. header.appendChild(createHTMLElement('br'));
  57. var tree = createHTMLElement('div');
  58. body.appendChild(tree);
  59. tree.classList.add('pretty-print');
  60. tree.id = 'tree';
  61. window.onload = sourceXMLLoaded;
  62. }
  63. function sourceXMLLoaded()
  64. {
  65. var sourceXML = document.getElementById('webkit-xml-viewer-source-xml');
  66. if (!sourceXML)
  67. return; // Stop if some XML tree extension is already processing this document
  68. //var style = document.head.firstChild;
  69. //document.head.removeChild(style);
  70. //document.head.appendChild(style);
  71. var root = document.getElementById('tree');
  72. for (var child = sourceXML.firstChild; child; child = child.nextSibling)
  73. nodeParentPairs.push({parentElement: root, node: child});
  74. for (var i = 0; i < nodeParentPairs.length; i++)
  75. processNode(nodeParentPairs[i].parentElement, nodeParentPairs[i].node);
  76. drawArrows();
  77. initButtons();
  78. if (typeof(onAfterWebkitXMLViewerLoaded) == 'function')
  79. onAfterWebkitXMLViewerLoaded();
  80. }
  81. // Tree processing.
  82. function processNode(parentElement, node)
  83. {
  84. if (!processNode.processorsMap) {
  85. processNode.processorsMap = {};
  86. processNode.processorsMap[Node.PROCESSING_INSTRUCTION_NODE] = processProcessingInstruction;
  87. processNode.processorsMap[Node.ELEMENT_NODE] = processElement;
  88. processNode.processorsMap[Node.COMMENT_NODE] = processComment;
  89. processNode.processorsMap[Node.TEXT_NODE] = processText;
  90. processNode.processorsMap[Node.CDATA_SECTION_NODE] = processCDATA;
  91. }
  92. if (processNode.processorsMap[node.nodeType])
  93. processNode.processorsMap[node.nodeType].call(this, parentElement, node);
  94. }
  95. function processElement(parentElement, node)
  96. {
  97. if (!node.firstChild)
  98. processEmptyElement(parentElement, node);
  99. else {
  100. var child = node.firstChild;
  101. if (child.nodeType == Node.TEXT_NODE && isShort(child.nodeValue) && !child.nextSibling)
  102. processShortTextOnlyElement(parentElement, node);
  103. else
  104. processComplexElement(parentElement, node);
  105. }
  106. }
  107. function processEmptyElement(parentElement, node)
  108. {
  109. var line = createLine();
  110. line.appendChild(createTag(node, false, true));
  111. parentElement.appendChild(line);
  112. }
  113. function processShortTextOnlyElement(parentElement, node)
  114. {
  115. var line = createLine();
  116. line.appendChild(createTag(node, false, false));
  117. for (var child = node.firstChild; child; child = child.nextSibling)
  118. line.appendChild(createText(child.nodeValue));
  119. line.appendChild(createTag(node, true, false));
  120. parentElement.appendChild(line);
  121. }
  122. function processComplexElement(parentElement, node)
  123. {
  124. var collapsible = createCollapsible();
  125. collapsible.expanded.start.appendChild(createTag(node, false, false));
  126. for (var child = node.firstChild; child; child = child.nextSibling)
  127. nodeParentPairs.push({parentElement: collapsible.expanded.content, node: child});
  128. collapsible.expanded.end.appendChild(createTag(node, true, false));
  129. collapsible.collapsed.content.appendChild(createTag(node, false, false));
  130. collapsible.collapsed.content.appendChild(createText('...'));
  131. collapsible.collapsed.content.appendChild(createTag(node, true, false));
  132. parentElement.appendChild(collapsible);
  133. }
  134. function processComment(parentElement, node)
  135. {
  136. if (isShort(node.nodeValue)) {
  137. var line = createLine();
  138. line.appendChild(createComment('<!-- ' + node.nodeValue + ' -->'));
  139. parentElement.appendChild(line);
  140. } else {
  141. var collapsible = createCollapsible();
  142. collapsible.expanded.start.appendChild(createComment('<!--'));
  143. collapsible.expanded.content.appendChild(createComment(node.nodeValue));
  144. collapsible.expanded.end.appendChild(createComment('-->'));
  145. collapsible.collapsed.content.appendChild(createComment('<!--'));
  146. collapsible.collapsed.content.appendChild(createComment('...'));
  147. collapsible.collapsed.content.appendChild(createComment('-->'));
  148. parentElement.appendChild(collapsible);
  149. }
  150. }
  151. function processCDATA(parentElement, node)
  152. {
  153. if (isShort(node.nodeValue)) {
  154. var line = createLine();
  155. line.appendChild(createText('<![CDATA[ ' + node.nodeValue + ' ]]>'));
  156. parentElement.appendChild(line);
  157. } else {
  158. var collapsible = createCollapsible();
  159. collapsible.expanded.start.appendChild(createText('<![CDATA['));
  160. collapsible.expanded.content.appendChild(createText(node.nodeValue));
  161. collapsible.expanded.end.appendChild(createText(']]>'));
  162. collapsible.collapsed.content.appendChild(createText('<![CDATA['));
  163. collapsible.collapsed.content.appendChild(createText('...'));
  164. collapsible.collapsed.content.appendChild(createText(']]>'));
  165. parentElement.appendChild(collapsible);
  166. }
  167. }
  168. function processProcessingInstruction(parentElement, node)
  169. {
  170. if (isShort(node.nodeValue)) {
  171. var line = createLine();
  172. line.appendChild(createComment('<?' + node.nodeName + ' ' + node.nodeValue + '?>'));
  173. parentElement.appendChild(line);
  174. } else {
  175. var collapsible = createCollapsible();
  176. collapsible.expanded.start.appendChild(createComment('<?' + node.nodeName));
  177. collapsible.expanded.content.appendChild(createComment(node.nodeValue));
  178. collapsible.expanded.end.appendChild(createComment('?>'));
  179. collapsible.collapsed.content.appendChild(createComment('<?' + node.nodeName));
  180. collapsible.collapsed.content.appendChild(createComment('...'));
  181. collapsible.collapsed.content.appendChild(createComment('?>'));
  182. parentElement.appendChild(collapsible);
  183. }
  184. }
  185. function processText(parentElement, node)
  186. {
  187. parentElement.appendChild(createText(node.nodeValue));
  188. }
  189. // Processing utils.
  190. function trim(value)
  191. {
  192. return value.replace(/^\s\s*/, '').replace(/\s\s*$/, '');
  193. }
  194. function isShort(value)
  195. {
  196. return trim(value).length <= 50;
  197. }
  198. // Tree rendering.
  199. function createHTMLElement(elementName)
  200. {
  201. return document.createElementNS('http://www.w3.org/1999/xhtml', elementName)
  202. }
  203. function createCollapsible()
  204. {
  205. var collapsible = createHTMLElement('div');
  206. collapsible.classList.add('collapsible');
  207. collapsible.expanded = createHTMLElement('div');
  208. collapsible.expanded.classList.add('expanded');
  209. collapsible.appendChild(collapsible.expanded);
  210. collapsible.expanded.start = createLine();
  211. collapsible.expanded.start.appendChild(createCollapseButton());
  212. collapsible.expanded.appendChild(collapsible.expanded.start);
  213. collapsible.expanded.content = createHTMLElement('div');
  214. collapsible.expanded.content.classList.add('collapsible-content');
  215. collapsible.expanded.appendChild(collapsible.expanded.content);
  216. collapsible.expanded.end = createLine();
  217. collapsible.expanded.appendChild(collapsible.expanded.end);
  218. collapsible.collapsed = createHTMLElement('div');
  219. collapsible.collapsed.classList.add('collapsed');
  220. collapsible.collapsed.classList.add('hidden');
  221. collapsible.appendChild(collapsible.collapsed);
  222. collapsible.collapsed.content = createLine();
  223. collapsible.collapsed.content.appendChild(createExpandButton());
  224. collapsible.collapsed.appendChild(collapsible.collapsed.content);
  225. return collapsible;
  226. }
  227. function createButton()
  228. {
  229. var button = createHTMLElement('span');
  230. button.classList.add('button');
  231. return button;
  232. }
  233. function createCollapseButton(str)
  234. {
  235. var button = createButton();
  236. button.classList.add('collapse-button');
  237. return button;
  238. }
  239. function createExpandButton(str)
  240. {
  241. var button = createButton();
  242. button.classList.add('expand-button');
  243. return button;
  244. }
  245. function createComment(commentString)
  246. {
  247. var comment = createHTMLElement('span');
  248. comment.classList.add('webkit-html-comment');
  249. comment.textContent = commentString;
  250. return comment;
  251. }
  252. function createText(value)
  253. {
  254. var text = createHTMLElement('span');
  255. text.textContent = trim(value);
  256. text.classList.add('text');
  257. return text;
  258. }
  259. function createLine()
  260. {
  261. var line = createHTMLElement('div');
  262. line.classList.add('line');
  263. return line;
  264. }
  265. function createTag(node, isClosing, isEmpty)
  266. {
  267. var tag = createHTMLElement('span');
  268. tag.classList.add('webkit-html-tag');
  269. var stringBeforeAttrs = '<';
  270. if (isClosing)
  271. stringBeforeAttrs += '/';
  272. stringBeforeAttrs += node.nodeName;
  273. var textBeforeAttrs = document.createTextNode(stringBeforeAttrs);
  274. tag.appendChild(textBeforeAttrs);
  275. if (!isClosing) {
  276. for (var i = 0; i < node.attributes.length; i++)
  277. tag.appendChild(createAttribute(node.attributes[i]));
  278. }
  279. var stringAfterAttrs = '';
  280. if (isEmpty)
  281. stringAfterAttrs += '/';
  282. stringAfterAttrs += '>';
  283. var textAfterAttrs = document.createTextNode(stringAfterAttrs);
  284. tag.appendChild(textAfterAttrs);
  285. return tag;
  286. }
  287. function createAttribute(attributeNode)
  288. {
  289. var attribute = createHTMLElement('span');
  290. attribute.classList.add('webkit-html-attribute');
  291. var attributeName = createHTMLElement('span');
  292. attributeName.classList.add('webkit-html-attribute-name');
  293. attributeName.textContent = attributeNode.name;
  294. var textBefore = document.createTextNode(' ');
  295. var textBetween = document.createTextNode('="');
  296. var attributeValue = createHTMLElement('span');
  297. attributeValue.classList.add('webkit-html-attribute-value');
  298. attributeValue.textContent = attributeNode.value;
  299. var textAfter = document.createTextNode('"');
  300. attribute.appendChild(textBefore);
  301. attribute.appendChild(attributeName);
  302. attribute.appendChild(textBetween);
  303. attribute.appendChild(attributeValue);
  304. attribute.appendChild(textAfter);
  305. return attribute;
  306. }
  307. // Tree behaviour.
  308. function drawArrows()
  309. {
  310. var ctx = document.getCSSCanvasContext("2d", "arrowRight", 10, 11);
  311. ctx.fillStyle = "rgb(90,90,90)";
  312. ctx.beginPath();
  313. ctx.moveTo(0, 0);
  314. ctx.lineTo(0, 8);
  315. ctx.lineTo(7, 4);
  316. ctx.lineTo(0, 0);
  317. ctx.fill();
  318. ctx.closePath();
  319. var ctx = document.getCSSCanvasContext("2d", "arrowDown", 10, 10);
  320. ctx.fillStyle = "rgb(90,90,90)";
  321. ctx.beginPath();
  322. ctx.moveTo(0, 0);
  323. ctx.lineTo(8, 0);
  324. ctx.lineTo(4, 7);
  325. ctx.lineTo(0, 0);
  326. ctx.fill();
  327. ctx.closePath();
  328. }
  329. function expandFunction(sectionId)
  330. {
  331. return function()
  332. {
  333. document.querySelector('#' + sectionId + ' > .expanded').className = 'expanded';
  334. document.querySelector('#' + sectionId + ' > .collapsed').className = 'collapsed hidden';
  335. };
  336. }
  337. function collapseFunction(sectionId)
  338. {
  339. return function()
  340. {
  341. document.querySelector('#' + sectionId + ' > .expanded').className = 'expanded hidden';
  342. document.querySelector('#' + sectionId + ' > .collapsed').className = 'collapsed';
  343. };
  344. }
  345. function initButtons()
  346. {
  347. var sections = document.querySelectorAll('.collapsible');
  348. for (var i = 0; i < sections.length; i++) {
  349. var sectionId = 'collapsible' + i;
  350. sections[i].id = sectionId;
  351. var expandedPart = sections[i].querySelector('#' + sectionId + ' > .expanded');
  352. var collapseButton = expandedPart.querySelector('.collapse-button');
  353. collapseButton.onclick = collapseFunction(sectionId);
  354. collapseButton.onmousedown = handleButtonMouseDown;
  355. var collapsedPart = sections[i].querySelector('#' + sectionId + ' > .collapsed');
  356. var expandButton = collapsedPart.querySelector('.expand-button');
  357. expandButton.onclick = expandFunction(sectionId);
  358. expandButton.onmousedown = handleButtonMouseDown;
  359. }
  360. }
  361. function handleButtonMouseDown(e)
  362. {
  363. // To prevent selection on double click
  364. e.preventDefault();
  365. }