2012-05-26 17:30:32 +00:00
|
|
|
/*\
|
|
|
|
title: $:/core/modules/parsers/newwikitextparser/newwikitextparser.js
|
|
|
|
type: application/javascript
|
|
|
|
module-type: parser
|
|
|
|
|
|
|
|
A new-school wikitext parser
|
|
|
|
|
|
|
|
\*/
|
|
|
|
(function(){
|
|
|
|
|
|
|
|
/*jslint node: true, browser: true */
|
|
|
|
/*global $tw: false */
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
/*
|
|
|
|
Define the wikitext renderer constructor
|
|
|
|
*/
|
|
|
|
var WikiTextRenderer = function(text,options) {
|
|
|
|
this.source = text || "";
|
|
|
|
this.sourceLength = this.source.length;
|
|
|
|
this.pos = 0;
|
|
|
|
this.wiki = options.wiki;
|
|
|
|
this.parser = options.parser;
|
|
|
|
this.tree = [];
|
|
|
|
this.dependencies = new $tw.Dependencies();
|
|
|
|
// Parse the text into blocks
|
|
|
|
while(this.pos < this.sourceLength) {
|
|
|
|
this.tree.push.apply(this.tree,this.parseBlock());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
Now make WikiTextRenderer inherit from the default Renderer class
|
|
|
|
*/
|
|
|
|
var Renderer = require("$:/core/modules/renderer.js").Renderer;
|
|
|
|
WikiTextRenderer.prototype = new Renderer();
|
|
|
|
WikiTextRenderer.constructor = WikiTextRenderer;
|
|
|
|
|
|
|
|
/*
|
|
|
|
Parse a block of text at the current position
|
|
|
|
*/
|
|
|
|
WikiTextRenderer.prototype.parseBlock = function() {
|
|
|
|
this.skipWhitespace();
|
2012-05-26 22:42:48 +00:00
|
|
|
if(this.pos >= this.sourceLength) {
|
|
|
|
return [];
|
|
|
|
}
|
2012-05-26 17:30:32 +00:00
|
|
|
// Look for a block rule
|
2012-05-26 23:22:58 +00:00
|
|
|
this.parser.blockRegExp.lastIndex = this.pos;
|
|
|
|
var match = this.parser.blockRegExp.exec(this.source);
|
|
|
|
if(this.parser.blockRules.length && match && match.index === this.pos) {
|
2012-05-26 17:30:32 +00:00
|
|
|
var rule;
|
2012-05-26 23:22:58 +00:00
|
|
|
for(var t=0; t<this.parser.blockRules.length; t++) {
|
2012-05-26 17:30:32 +00:00
|
|
|
if(match[t+1]) {
|
2012-05-26 23:22:58 +00:00
|
|
|
rule = this.parser.blockRules[t];
|
2012-05-26 17:30:32 +00:00
|
|
|
}
|
|
|
|
}
|
2012-05-28 14:51:52 +00:00
|
|
|
return rule ? rule.parse.call(this,match,true) : [];
|
2012-05-26 17:30:32 +00:00
|
|
|
} else {
|
|
|
|
// Treat it as a paragraph if we didn't find a block rule
|
|
|
|
return [$tw.Tree.Element("p",{},this.parseRun())];
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
WikiTextRenderer.prototype.skipWhitespace = function() {
|
|
|
|
var whitespaceRegExp = /(\s+)/mg;
|
|
|
|
whitespaceRegExp.lastIndex = this.pos;
|
|
|
|
var whitespaceMatch = whitespaceRegExp.exec(this.source);
|
|
|
|
if(whitespaceMatch && whitespaceMatch.index === this.pos) {
|
|
|
|
this.pos = whitespaceRegExp.lastIndex;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
Parse a run of text at the current position
|
|
|
|
terminatorRegExp: a regexp at which to stop the run
|
|
|
|
Returns an array of tree nodes
|
|
|
|
*/
|
|
|
|
WikiTextRenderer.prototype.parseRun = function(terminatorRegExp) {
|
|
|
|
var tree = [];
|
|
|
|
// Find the next occurrence of the terminator
|
|
|
|
terminatorRegExp = terminatorRegExp || /(\r?\n\r?\n)/mg;
|
|
|
|
terminatorRegExp.lastIndex = this.pos;
|
|
|
|
var terminatorMatch = terminatorRegExp.exec(this.source);
|
|
|
|
// Find the next occurrence of a runrule
|
2012-05-26 23:22:58 +00:00
|
|
|
this.parser.runRegExp.lastIndex = this.pos;
|
|
|
|
var runRuleMatch = this.parser.runRegExp.exec(this.source);
|
2012-05-26 17:30:32 +00:00
|
|
|
// Loop around until we've reached the end of the text
|
|
|
|
while(this.pos < this.sourceLength && (terminatorMatch || runRuleMatch)) {
|
|
|
|
// Return if we've found the terminator, and it precedes any run rule match
|
|
|
|
if(terminatorMatch) {
|
|
|
|
if(!runRuleMatch || runRuleMatch.index > terminatorMatch.index) {
|
|
|
|
if(terminatorMatch.index > this.pos) {
|
|
|
|
tree.push($tw.Tree.Text(this.source.substring(this.pos,terminatorMatch.index)));
|
|
|
|
}
|
2012-06-02 08:41:03 +00:00
|
|
|
this.pos = terminatorMatch.index + terminatorMatch[0].length;
|
2012-05-26 17:30:32 +00:00
|
|
|
return tree;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Process any run rule, along with the text preceding it
|
|
|
|
if(runRuleMatch) {
|
|
|
|
// Preceding text
|
|
|
|
if(runRuleMatch.index > this.pos) {
|
|
|
|
tree.push($tw.Tree.Text(this.source.substring(this.pos,runRuleMatch.index)));
|
|
|
|
this.pos = runRuleMatch.index;
|
|
|
|
}
|
|
|
|
// Process the run rule
|
|
|
|
var rule;
|
2012-05-26 23:22:58 +00:00
|
|
|
for(var t=0; t<this.parser.runRules.length; t++) {
|
2012-05-26 17:30:32 +00:00
|
|
|
if(runRuleMatch[t+1]) {
|
2012-05-26 23:22:58 +00:00
|
|
|
rule = this.parser.runRules[t];
|
2012-05-26 17:30:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if(rule) {
|
2012-05-28 14:51:52 +00:00
|
|
|
tree.push.apply(tree,rule.parse.call(this,runRuleMatch,false));
|
2012-05-26 17:30:32 +00:00
|
|
|
}
|
|
|
|
// Look for the next run rule
|
2012-05-26 23:22:58 +00:00
|
|
|
this.parser.runRegExp.lastIndex = this.pos;
|
|
|
|
runRuleMatch = this.parser.runRegExp.exec(this.source);
|
2012-05-26 17:30:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// Process the remaining text
|
|
|
|
if(this.pos < this.sourceLength) {
|
2012-06-02 08:41:03 +00:00
|
|
|
tree.push($tw.Tree.Text(this.source.substr(this.pos)));
|
2012-05-26 17:30:32 +00:00
|
|
|
}
|
|
|
|
this.pos = this.sourceLength;
|
|
|
|
return tree;
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
Parse a run of text preceded by an optional class specifier `{{class}}`
|
|
|
|
*/
|
|
|
|
WikiTextRenderer.prototype.parseClassedRun = function(terminatorRegExp) {
|
|
|
|
var classRegExp = /\{\{([^\}]*)\}\}/mg,
|
|
|
|
className;
|
|
|
|
classRegExp.lastIndex = this.pos;
|
|
|
|
var match = classRegExp.exec(this.source);
|
|
|
|
if(match && match.index === this.pos) {
|
|
|
|
className = match[1];
|
|
|
|
this.pos = match.index + match[0].length;
|
|
|
|
}
|
|
|
|
var tree = this.parseRun(terminatorRegExp);
|
|
|
|
return {
|
|
|
|
"class": className,
|
|
|
|
tree: tree
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
The wikitext parser assembles the rules and uses the wikitext renderer to do the parsing
|
|
|
|
*/
|
|
|
|
var WikiTextParser = function(options) {
|
2012-05-26 23:22:58 +00:00
|
|
|
this.wiki = options.wiki;
|
|
|
|
// Assemble the rule regexps
|
|
|
|
this.blockRules = [];
|
|
|
|
this.runRules = [];
|
|
|
|
var blockRegExpStrings = [],
|
|
|
|
runRegExpStrings = [],
|
2012-05-27 08:19:37 +00:00
|
|
|
rules = ($tw.plugins.moduleTypes.wikitextrule || []).slice(0);
|
2012-05-26 23:22:58 +00:00
|
|
|
for(var t=0; t<rules.length; t++) {
|
|
|
|
if(rules[t].blockParser) {
|
|
|
|
this.blockRules.push(rules[t]);
|
|
|
|
blockRegExpStrings.push("(" + rules[t].regExpString + ")");
|
|
|
|
}
|
|
|
|
if(rules[t].runParser) {
|
|
|
|
this.runRules.push(rules[t]);
|
|
|
|
runRegExpStrings.push("(" + rules[t].regExpString + ")");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.blockRegExp = new RegExp(blockRegExpStrings.join("|"),"mg");
|
|
|
|
this.runRegExp = new RegExp(runRegExpStrings.join("|"),"mg");
|
2012-05-26 17:30:32 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
The wikitext parser constructs a wikitext renderer to do the work
|
|
|
|
*/
|
|
|
|
WikiTextParser.prototype.parse = function(type,text) {
|
|
|
|
return new WikiTextRenderer(text,{
|
|
|
|
wiki: this.wiki,
|
|
|
|
parser: this
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
exports["text/x-tiddlywiki-new"] = WikiTextParser;
|
|
|
|
|
|
|
|
})();
|