2012-12-13 21:34:31 +00:00
|
|
|
/*\
|
|
|
|
title: $:/core/modules/widgets/transclude.js
|
|
|
|
type: application/javascript
|
|
|
|
module-type: widget
|
|
|
|
|
|
|
|
The transclude widget includes another tiddler into the tiddler being rendered.
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
target: the title of the tiddler to transclude
|
|
|
|
template: the title of the tiddler to use as a template for the transcluded tiddler
|
|
|
|
|
|
|
|
The simplest case is to just supply a target tiddler:
|
|
|
|
|
|
|
|
{{{
|
2013-01-03 16:27:55 +00:00
|
|
|
<$transclude target="Foo"/>
|
2012-12-13 21:34:31 +00:00
|
|
|
}}}
|
|
|
|
|
|
|
|
This will render the tiddler Foo within the current tiddler. If the tiddler Foo includes
|
|
|
|
the view widget (or other widget that reference the fields of the current tiddler), then the
|
|
|
|
fields of the tiddler Foo will be accessed.
|
|
|
|
|
|
|
|
If you want to transclude the tiddler as a template, so that the fields referenced by the view
|
|
|
|
widget are those of the tiddler doing the transcluding, then you can instead specify the tiddler
|
|
|
|
as a template:
|
|
|
|
|
|
|
|
{{{
|
2013-01-03 16:27:55 +00:00
|
|
|
<$transclude template="Foo"/>
|
2012-12-13 21:34:31 +00:00
|
|
|
}}}
|
|
|
|
|
|
|
|
The effect is the same as the previous example: the text of the tiddler Foo is rendered. The
|
|
|
|
difference is that the view widget will access the fields of the tiddler doing the transcluding.
|
|
|
|
|
|
|
|
The `target` and `template` attributes may be combined:
|
|
|
|
|
|
|
|
{{{
|
2013-01-03 16:27:55 +00:00
|
|
|
<$transclude template="Bar" target="Foo"/>
|
2012-12-13 21:34:31 +00:00
|
|
|
}}}
|
|
|
|
|
|
|
|
Here, the text of the tiddler `Bar` will be transcluded, with the widgets within it accessing the fields
|
|
|
|
of the tiddler `Foo`.
|
|
|
|
|
|
|
|
\*/
|
|
|
|
(function(){
|
|
|
|
|
|
|
|
/*jslint node: true, browser: true */
|
|
|
|
/*global $tw: false */
|
|
|
|
"use strict";
|
|
|
|
|
2013-01-01 17:51:02 +00:00
|
|
|
var TranscludeWidget = function(renderer) {
|
2012-12-13 21:34:31 +00:00
|
|
|
// Save state
|
|
|
|
this.renderer = renderer;
|
|
|
|
// Generate child nodes
|
2013-01-03 16:27:55 +00:00
|
|
|
this.generate();
|
2012-12-13 21:34:31 +00:00
|
|
|
};
|
|
|
|
|
2013-01-03 16:27:55 +00:00
|
|
|
TranscludeWidget.prototype.generate = function() {
|
2012-12-13 21:34:31 +00:00
|
|
|
var tr, templateParseTree, templateTiddler;
|
|
|
|
// Get the render target details
|
|
|
|
this.targetTitle = this.renderer.getAttribute("target",this.renderer.getContextTiddlerTitle());
|
2013-01-03 16:27:55 +00:00
|
|
|
this.targetField = this.renderer.getAttribute("field","text");
|
2012-12-13 21:34:31 +00:00
|
|
|
// Get the render tree for the template
|
|
|
|
this.templateTitle = undefined;
|
|
|
|
if(this.renderer.parseTreeNode.children && this.renderer.parseTreeNode.children.length > 0) {
|
|
|
|
// Use the child nodes as the template if we've got them
|
|
|
|
templateParseTree = this.renderer.parseTreeNode.children;
|
|
|
|
} else {
|
|
|
|
this.templateTitle = this.renderer.getAttribute("template",this.targetTitle);
|
|
|
|
// Check for recursion
|
|
|
|
if(this.renderer.checkContextRecursion({
|
|
|
|
tiddlerTitle: this.targetTitle,
|
|
|
|
templateTitle: this.templateTitle
|
|
|
|
})) {
|
|
|
|
templateParseTree = [{type: "text", text: "Tiddler recursion error in transclude widget"}];
|
|
|
|
} else {
|
2013-01-03 16:27:55 +00:00
|
|
|
var parser;
|
|
|
|
if(this.targetField === "text") {
|
|
|
|
parser = this.renderer.renderTree.wiki.parseTiddler(this.templateTitle,{parseAsInline: !this.renderer.parseTreeNode.isBlock})
|
|
|
|
} else {
|
|
|
|
var tiddler = this.renderer.renderTree.wiki.getTiddler(this.targetTitle),
|
|
|
|
text = tiddler ? tiddler.fields[this.targetField] : "";
|
|
|
|
if(text === undefined) {
|
|
|
|
text = ""
|
|
|
|
}
|
|
|
|
parser = this.renderer.renderTree.wiki.parseText("text/vnd.tiddlywiki",text,{parseAsInline: !this.renderer.parseTreeNode.isBlock});
|
|
|
|
}
|
2012-12-13 21:34:31 +00:00
|
|
|
templateParseTree = parser ? parser.tree : [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Set up the attributes for the wrapper element
|
2013-01-03 16:27:55 +00:00
|
|
|
var classes = ["tw-transclude"];
|
2012-12-20 17:19:28 +00:00
|
|
|
if(this.renderer.hasAttribute("class")) {
|
|
|
|
$tw.utils.pushTop(classes,this.renderer.getAttribute("class").split(" "));
|
|
|
|
}
|
2012-12-13 21:34:31 +00:00
|
|
|
if(!this.renderer.renderTree.wiki.tiddlerExists(this.targetTitle)) {
|
|
|
|
$tw.utils.pushTop(classes,"tw-tiddler-missing");
|
|
|
|
}
|
|
|
|
// Create the renderers for the wrapper and the children
|
|
|
|
var newRenderContext = {
|
|
|
|
tiddlerTitle: this.targetTitle,
|
|
|
|
templateTitle: this.templateTitle,
|
|
|
|
parentContext: this.renderer.renderContext
|
|
|
|
};
|
2013-01-03 16:27:55 +00:00
|
|
|
// Set the element
|
|
|
|
this.tag = this.renderer.parseTreeNode.isBlock ? "div" : "span";
|
|
|
|
this.attributes = {};
|
|
|
|
if(classes.length > 0) {
|
|
|
|
this.attributes["class"] = classes.join(" ");
|
|
|
|
}
|
|
|
|
if(this.renderer.hasAttribute("style")) {
|
|
|
|
this.attributes.style = this.renderer.getAttribute("style");
|
|
|
|
}
|
|
|
|
if(this.renderer.hasAttribute("tooltip")) {
|
|
|
|
this.attributes.title = this.renderer.getAttribute("tooltip");
|
|
|
|
}
|
|
|
|
this.children = this.renderer.renderTree.createRenderers(newRenderContext,templateParseTree);
|
2012-12-13 21:34:31 +00:00
|
|
|
};
|
|
|
|
|
2013-01-01 17:51:02 +00:00
|
|
|
TranscludeWidget.prototype.refreshInDom = function(changedAttributes,changedTiddlers) {
|
2012-12-13 21:34:31 +00:00
|
|
|
// Set the class for missing tiddlers
|
2012-12-14 19:31:37 +00:00
|
|
|
if(this.targetTitle && changedTiddlers[this.targetTitle]) {
|
2013-01-03 20:54:34 +00:00
|
|
|
$tw.utils.toggleClass(this.renderer.domNode,"tw-tiddler-missing",!this.renderer.renderTree.wiki.tiddlerExists(this.targetTitle));
|
2012-12-13 21:34:31 +00:00
|
|
|
}
|
|
|
|
// Check if any of our attributes have changed, or if a tiddler we're interested in has changed
|
|
|
|
if(changedAttributes.target || changedAttributes.template || (this.targetTitle && changedTiddlers[this.targetTitle]) || (this.templateTitle && changedTiddlers[this.templateTitle])) {
|
2013-01-03 16:27:55 +00:00
|
|
|
// Regenerate and rerender the widget and replace the existing DOM node
|
|
|
|
this.generate();
|
|
|
|
var oldDomNode = this.renderer.domNode,
|
|
|
|
newDomNode = this.renderer.renderInDom();
|
|
|
|
oldDomNode.parentNode.replaceChild(newDomNode,oldDomNode);
|
2012-12-13 21:34:31 +00:00
|
|
|
} else {
|
|
|
|
// We don't need to refresh ourselves, so just refresh any child nodes
|
|
|
|
$tw.utils.each(this.children,function(node) {
|
|
|
|
if(node.refreshInDom) {
|
|
|
|
node.refreshInDom(changedTiddlers);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-01-01 17:51:02 +00:00
|
|
|
exports.transclude = TranscludeWidget;
|
|
|
|
|
2012-12-13 21:34:31 +00:00
|
|
|
})();
|