2011-05-22 12:44:39 -04:00
|
|
|
/** Handle the creation of HTML links to documented symbols.
|
|
|
|
@constructor
|
|
|
|
*/
|
|
|
|
function Link() {
|
|
|
|
this.alias = "";
|
|
|
|
this.src = "";
|
|
|
|
this.file = "";
|
|
|
|
this.text = "";
|
|
|
|
this.innerName = "";
|
|
|
|
this.classLink = false;
|
|
|
|
this.targetName = "";
|
|
|
|
|
|
|
|
this.target = function(targetName) {
|
|
|
|
if (defined(targetName)) this.targetName = targetName;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
this.inner = function(inner) {
|
|
|
|
if (defined(inner)) this.innerName = inner;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
this.withText = function(text) {
|
|
|
|
if (defined(text)) this.text = text;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
this.toSrc = function(filename) {
|
|
|
|
if (defined(filename)) this.src = filename;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
this.toSymbol = function(alias) {
|
2011-06-06 19:00:30 -04:00
|
|
|
// 'Class[]' to 'Array of Class objects'
|
2011-05-26 14:13:07 -04:00
|
|
|
if(/\[\]$/.test(alias)) {
|
|
|
|
alias = alias.replace(/\[\]$/, '');
|
2011-06-06 19:00:30 -04:00
|
|
|
this.prefix = 'Array of ';
|
|
|
|
this.text = alias;
|
|
|
|
this.suffix = ' objects';
|
2011-05-26 14:13:07 -04:00
|
|
|
}
|
2011-05-22 12:44:39 -04:00
|
|
|
if (defined(alias)) this.alias = new String(alias);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
this.toClass = function(alias) {
|
|
|
|
this.classLink = true;
|
|
|
|
return this.toSymbol(alias);
|
|
|
|
}
|
|
|
|
this.toFile = function(file) {
|
|
|
|
if (defined(file)) this.file = file;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.toString = function() {
|
|
|
|
var linkString;
|
|
|
|
var thisLink = this;
|
|
|
|
|
|
|
|
if (this.alias) {
|
2011-05-27 13:43:45 -04:00
|
|
|
linkString = this.alias.replace(/(^|[^a-z$0-9_#.:^-])([|a-z$0-9_#.:^-]+)(\([^)]+\))*($|[^a-z$0-9_#.:^-])/i,
|
|
|
|
function(match, prematch, symbolName, parameters, postmatch) {
|
2011-05-22 12:44:39 -04:00
|
|
|
var symbolNames = symbolName.split("|");
|
|
|
|
var links = [];
|
|
|
|
for (var i = 0, l = symbolNames.length; i < l; i++) {
|
|
|
|
thisLink.alias = symbolNames[i];
|
2011-05-27 13:43:45 -04:00
|
|
|
links.push(thisLink._makeSymbolLink(symbolNames[i], parameters));
|
2011-05-22 12:44:39 -04:00
|
|
|
}
|
2011-06-06 19:01:07 -04:00
|
|
|
return prematch+links.join(" / ")+postmatch;
|
2011-05-22 12:44:39 -04:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else if (this.src) {
|
|
|
|
linkString = thisLink._makeSrcLink(this.src);
|
|
|
|
}
|
|
|
|
else if (this.file) {
|
|
|
|
linkString = thisLink._makeFileLink(this.file);
|
|
|
|
}
|
|
|
|
|
|
|
|
return linkString;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** prefixed for hashes */
|
|
|
|
Link.hashPrefix = "";
|
|
|
|
|
|
|
|
/** Appended to the front of relative link paths. */
|
|
|
|
Link.base = "";
|
|
|
|
|
|
|
|
Link.symbolNameToLinkName = function(symbol) {
|
|
|
|
var linker = "",
|
|
|
|
ns = "";
|
|
|
|
|
2011-05-28 13:31:41 -04:00
|
|
|
if (symbol.isStatic) linker = "-";
|
2011-05-22 12:44:39 -04:00
|
|
|
else if (symbol.isInner) linker = "-";
|
|
|
|
|
|
|
|
if (symbol.isEvent && !/^event:/.test(symbol.name)) {
|
|
|
|
ns = "event:";
|
|
|
|
}
|
|
|
|
return Link.hashPrefix+linker+ns+symbol.name;
|
|
|
|
}
|
|
|
|
|
|
|
|
Link.getSymbol= function(alias) {
|
|
|
|
var symbol= Link.symbolSet.getSymbol(alias);
|
|
|
|
|
|
|
|
if (symbol)
|
|
|
|
return symbol;
|
|
|
|
|
|
|
|
if ('#'!==alias.charAt(0) || !Link.currentSymbol)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
// resolve relative name
|
|
|
|
var container= Link.currentSymbol;
|
|
|
|
|
|
|
|
while (container)
|
|
|
|
{
|
|
|
|
symbol= Link.symbolSet.getSymbol(container.alias + alias);
|
|
|
|
if (symbol)
|
|
|
|
return symbol;
|
|
|
|
|
|
|
|
// No superclass
|
|
|
|
if (!container.augments.length)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
container= Link.symbolSet.getSymbol(container.augments[0].desc);
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Create a link to another symbol. */
|
2011-05-27 13:43:45 -04:00
|
|
|
Link.prototype._makeSymbolLink = function(alias, parameters) {
|
2011-06-07 05:17:33 -04:00
|
|
|
var linkBase = Link.base + publish.conf.symbolsDir;
|
2011-05-22 12:44:39 -04:00
|
|
|
var linkTo = Link.getSymbol(alias);
|
|
|
|
var linkPath;
|
|
|
|
var target = (this.targetName)? " target=\""+this.targetName+"\"" : "";
|
2011-05-27 21:05:23 -04:00
|
|
|
// Expand (parameter,parameter) to (parameter, parameter)
|
|
|
|
if (parameters) {
|
|
|
|
parameters = parameters.replace(/\,([^\s])/, ', $1')
|
|
|
|
}
|
|
|
|
|
2011-05-22 12:44:39 -04:00
|
|
|
// if there is no symbol by that name just return the name unaltered
|
2011-05-27 13:43:45 -04:00
|
|
|
if (!linkTo) {
|
2011-05-27 13:58:31 -04:00
|
|
|
return this.text || alias + (parameters || '');
|
2011-05-27 13:43:45 -04:00
|
|
|
} else {
|
|
|
|
// it's a symbol in another file
|
2011-05-22 12:44:39 -04:00
|
|
|
if (!linkTo.is("CONSTRUCTOR") && !linkTo.isNamespace) { // it's a method or property
|
|
|
|
linkPath= (Link.filemap) ? Link.filemap[linkTo.memberOf] :
|
2011-05-28 13:31:41 -04:00
|
|
|
escape(linkTo.memberOf) || "global";
|
2011-05-27 13:58:31 -04:00
|
|
|
linkPath += publish.conf.ext + "#" + Link.symbolNameToLinkName(linkTo).toLowerCase();
|
2011-05-27 13:43:45 -04:00
|
|
|
if (parameters) {
|
|
|
|
linkPath += '-' + parameters.replace(/[()]+/g, '').split(', ').join('-').toLowerCase();
|
|
|
|
}
|
|
|
|
} else {
|
2011-05-22 12:44:39 -04:00
|
|
|
linkPath = (Link.filemap)? Link.filemap[linkTo.alias] : escape(linkTo.alias);
|
|
|
|
linkPath += publish.conf.ext;// + (this.classLink? "":"#" + Link.hashPrefix + "constructor");
|
|
|
|
}
|
|
|
|
linkPath = linkBase + linkPath
|
|
|
|
}
|
|
|
|
|
2011-05-27 13:43:45 -04:00
|
|
|
var linkText= (this.text || alias) + (parameters || '');
|
2011-05-22 12:44:39 -04:00
|
|
|
|
2011-05-27 13:43:45 -04:00
|
|
|
var link = {
|
|
|
|
linkPath: linkPath,
|
|
|
|
linkText: linkText.replace(/^#/, ''),
|
|
|
|
linkInner: (this.innerName? "#" + this.innerName : "")
|
|
|
|
};
|
2011-05-22 12:44:39 -04:00
|
|
|
|
|
|
|
if (typeof JSDOC.PluginManager != "undefined") {
|
|
|
|
JSDOC.PluginManager.run("onSymbolLink", link);
|
|
|
|
}
|
2011-05-26 14:13:07 -04:00
|
|
|
if (/^[A-Z].+\#/.test(link.linkText)) {
|
|
|
|
link.linkText = link.linkText.charAt(0).toLowerCase() + link.linkText.slice(1);
|
|
|
|
link.linkText = link.linkText.replace(/#/g, '.');
|
|
|
|
}
|
2011-05-29 11:23:45 -04:00
|
|
|
var onClick = '';
|
|
|
|
// If linking to a member of the same class, we also need to toggle it:
|
|
|
|
if (alias.charAt(0) == '#') {
|
|
|
|
var linkName = link.linkPath.replace(/^[^#]+#/, '');
|
|
|
|
onClick = " onclick=\"return toggleMember('" + linkName + "', true);\"";
|
|
|
|
}
|
2011-06-06 19:00:30 -04:00
|
|
|
var html = "<a href=\""+link.linkPath+link.linkInner+"\""+target+onClick+"><tt>"+link.linkText+"</tt></a>";
|
|
|
|
if (this.prefix || this.suffix)
|
|
|
|
html = (this.prefix || '') + html + (this.suffix || '');
|
|
|
|
return html;
|
2011-05-22 12:44:39 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Create a link to a source file. */
|
|
|
|
Link.prototype._makeSrcLink = function(srcFilePath) {
|
|
|
|
var target = (this.targetName)? " target=\""+this.targetName+"\"" : "";
|
|
|
|
|
|
|
|
// transform filepath into a filename
|
|
|
|
var srcFile = srcFilePath.replace(/\.\.?[\\\/]/g, "").replace(/[:\\\/]/g, "_");
|
|
|
|
var outFilePath = Link.base + publish.conf.srcDir + srcFile + publish.conf.ext;
|
|
|
|
|
|
|
|
if (!this.text) this.text = FilePath.fileName(srcFilePath);
|
|
|
|
return "<a href=\""+outFilePath+"\""+target+">"+this.text+"</a>";
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Create a link to a source file. */
|
|
|
|
Link.prototype._makeFileLink = function(filePath) {
|
|
|
|
var target = (this.targetName)? " target=\""+this.targetName+"\"" : "";
|
|
|
|
|
|
|
|
var outFilePath = Link.base + filePath;
|
|
|
|
|
|
|
|
if (!this.text) this.text = filePath;
|
|
|
|
return "<a href=\""+outFilePath+"\""+target+">"+this.text+"</a>";
|
|
|
|
}
|