/*\ title: $:/core/modules/widgets/text.js type: application/javascript module-type: widget An override of the core text widget that automatically linkifies the text \*/ (function(){ /*jslint node: true, browser: true */ /*global $tw: false */ "use strict"; var Widget = require("$:/core/modules/widgets/widget.js").widget; var TextNodeWidget = function(parseTreeNode,options) { this.initialise(parseTreeNode,options); }; /* Inherit from the base widget class */ TextNodeWidget.prototype = new Widget(); /* Render this widget into the DOM */ TextNodeWidget.prototype.render = function(parent,nextSibling) { this.parentDomNode = parent; this.computeAttributes(); this.execute(); this.renderChildren(parent,nextSibling); }; /* Compute the internal state of the widget */ TextNodeWidget.prototype.execute = function() { var self = this; // Get our parameters var childParseTree = [{ type: "plain-text", text: this.getAttribute("text",this.parseTreeNode.text || "") }]; // Only process links if not disabled if(this.getVariable("tv-wikilinks",{defaultValue:"yes"}).trim() !== "no" && this.getVariable("tv-freelinks",{defaultValue:"no"}).trim() === "yes") { // Get the information about the current tiddler titles, and construct a regexp this.tiddlerTitleInfo = this.wiki.getGlobalCache("tiddler-title-info",function() { var titles = [], reparts = [], sortedTitles = self.wiki.allTitles().sort(function(a,b) { var lenA = a.length, lenB = b.length; // First sort by length, so longer titles are first if(lenA !== lenB) { if(lenA < lenB) { return +1; } else { return -1; } } else { // Then sort alphabetically within titles of the same length if(a < b) { return -1; } else if(a > b) { return +1; } else { return 0; } } }); $tw.utils.each(sortedTitles,function(title) { if(title.substring(0,3) !== "$:/") { titles.push(title); reparts.push("(\\b" + $tw.utils.escapeRegExp(title) + "\\b)"); } }); return { titles: titles, regexp: new RegExp(reparts.join("|"),"") }; }); // Repeatedly linkify if(this.tiddlerTitleInfo.titles.length > 0) { var index,text,match,matchEnd; do { index = childParseTree.length - 1; text = childParseTree[index].text; match = this.tiddlerTitleInfo.regexp.exec(text); if(match) { // Make a text node for any text before the match if(match.index > 0) { childParseTree[index].text = text.substring(0,match.index); index += 1; } // Make a link node for the match childParseTree[index] = { type: "link", attributes: { to: {type: "string", value: match[0]} }, children: [{ type: "plain-text", text: match[0] }] }; index += 1; // Make a text node for any text after the match matchEnd = match.index + match[0].length; if(matchEnd < text.length) { childParseTree[index] = { type: "plain-text", text: text.substring(matchEnd) }; } } } while(match && childParseTree[childParseTree.length - 1].type === "plain-text"); } } // Make the child widgets this.makeChildWidgets(childParseTree); }; /* Selectively refreshes the widget if needed. Returns true if the widget or any of its children needed re-rendering */ TextNodeWidget.prototype.refresh = function(changedTiddlers) { var self = this, changedAttributes = this.computeAttributes(), titlesHaveChanged = false; $tw.utils.each(changedTiddlers,function(change,title) { if(change.isDeleted) { titlesHaveChanged = true } else { titlesHaveChanged = titlesHaveChanged || !self.tiddlerTitleInfo || self.tiddlerTitleInfo.titles.indexOf(title) === -1; } }); if(changedAttributes.text || titlesHaveChanged) { this.refreshSelf(); return true; } else { return false; } }; exports.text = TextNodeWidget; })();