/*\ title: $:/core/modules/parsers/wikiparser/rules/html.js type: application/javascript module-type: wikirule Wiki rule for HTML elements and widgets. For example: {{{ <_slider target="MyTiddler"> This is a widget invocation }}} \*/ (function(){ /*jslint node: true, browser: true */ /*global $tw: false */ "use strict"; exports.name = "html"; exports.types = {inline: true}; var voidElements = "area,base,br,col,command,embed,hr,img,input,keygen,link,meta,param,source,track,wbr".split(","); exports.init = function(parser) { this.parser = parser; // Regexp to match this.matchRegExp = /<(_)?([A-Za-z]+)(\s*[^>]*?)(\/)?>/mg; }; /* Parse the most recent match */ exports.parse = function() { // Get all the details of the match in case this parser is called recursively var isWidget = !!this.match[1], tagName = this.match[2], attributeString = this.match[3], isSelfClosing = !!this.match[4]; // Move past the tag name and parameters this.parser.pos = this.matchRegExp.lastIndex; var reLineBreak = /(\r?\n)/mg, reAttr = /\s*([A-Za-z\-_]+)(?:\s*=\s*(?:("[^"]*")|('[^']*')|(\{\{[^\}]*\}\})|([^"'\s]+)))?/mg, isBlock; // Process the attributes var attrMatch = reAttr.exec(attributeString), attributes = {}; while(attrMatch) { var name = attrMatch[1], value; if(attrMatch[2]) { // Double quoted value = {type: "string", value: attrMatch[2].substring(1,attrMatch[2].length-1)}; } else if(attrMatch[3]) { // Single quoted value = {type: "string", value: attrMatch[3].substring(1,attrMatch[3].length-1)}; } else if(attrMatch[4]) { // Double curly brace quoted value = {type: "indirect", textReference: attrMatch[4].substr(2,attrMatch[4].length-4)}; } else if(attrMatch[5]) { // Unquoted value = {type: "string", value: attrMatch[5]}; } else { // Valueless value = {type: "string", value: "true"}; // TODO: We should have a way of indicating we want an attribute without a value } attributes[name] = value; attrMatch = reAttr.exec(attributeString); } // Check for a line break immediate after the opening tag reLineBreak.lastIndex = this.parser.pos; var lineBreakMatch = reLineBreak.exec(this.parser.source); if(lineBreakMatch && lineBreakMatch.index === this.parser.pos) { this.parser.pos = lineBreakMatch.index + lineBreakMatch[0].length; isBlock = true; } else { isBlock = false; } if(!isSelfClosing && (isWidget || voidElements.indexOf(tagName) === -1)) { var reEndString = "()", reEnd = new RegExp(reEndString,"mg"), content; if(isBlock) { content = this.parser.parseBlocks(reEndString); } else { content = this.parser.parseInlineRun(reEnd); } reEnd.lastIndex = this.parser.pos; var endMatch = reEnd.exec(this.parser.source); if(endMatch && endMatch.index === this.parser.pos) { this.parser.pos = endMatch.index + endMatch[0].length; } } else { content = []; } var element = {type: isWidget ? "widget" : "element", tag: tagName, isBlock: isBlock, attributes: attributes, children: content}; return [element]; }; })();