publish.js 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966
  1. 'use strict';
  2. var doop = require('jsdoc/util/doop');
  3. var env = require('jsdoc/env');
  4. var fs = require('jsdoc/fs');
  5. var helper = require('jsdoc/util/templateHelper');
  6. var logger = require('jsdoc/util/logger');
  7. var path = require('jsdoc/path');
  8. var taffy = require('taffydb').taffy;
  9. var template = require('jsdoc/template');
  10. var tutorial = require('jsdoc/tutorial');
  11. var util = require('util');
  12. var cheerio = require('cheerio'); // for parse html to dom
  13. var _ = require('underscore');
  14. var htmlsafe = helper.htmlsafe;
  15. var linkto = helper.linkto;
  16. var resolveAuthorLinks = helper.resolveAuthorLinks;
  17. var scopeToPunc = helper.scopeToPunc;
  18. var hasOwnProp = Object.prototype.hasOwnProperty;
  19. var data;
  20. var view;
  21. var tutorialsName;
  22. var outdir = path.normalize(env.opts.destination);
  23. env.conf.templates = _.extend(
  24. {
  25. useCollapsibles: true
  26. },
  27. env.conf.templates
  28. );
  29. env.conf.templates.tabNames = _.extend(
  30. {
  31. api: 'API',
  32. tutorials: 'Examples'
  33. },
  34. env.conf.templates.tabNames
  35. );
  36. tutorialsName = env.conf.templates.tabNames.tutorials;
  37. // Set default useCollapsibles true
  38. env.conf.templates.useCollapsibles =
  39. env.conf.templates.useCollapsibles !== false;
  40. function find(spec) {
  41. return helper.find(data, spec);
  42. }
  43. function tutoriallink(tutorial) {
  44. return helper.toTutorial(tutorial, null, {
  45. tag: 'em',
  46. classname: 'disabled',
  47. prefix: 'Tutorial: '
  48. });
  49. }
  50. function getAncestorLinks(doclet) {
  51. return helper.getAncestorLinks(data, doclet);
  52. }
  53. function hashToLink(doclet, hash) {
  54. var url;
  55. if (!/^(#.+)/.test(hash)) {
  56. return hash;
  57. }
  58. url = helper.createLink(doclet);
  59. url = url.replace(/(#.+|$)/, hash);
  60. return '<a href="' + url + '">' + hash + '</a>';
  61. }
  62. function needsSignature(doclet) {
  63. var needsSig = false;
  64. // function and class definitions always get a signature
  65. if (doclet.kind === 'function' || doclet.kind === 'class') {
  66. needsSig = true;
  67. }
  68. // typedefs that contain functions get a signature, too
  69. else if (
  70. doclet.kind === 'typedef' &&
  71. doclet.type &&
  72. doclet.type.names &&
  73. doclet.type.names.length
  74. ) {
  75. for (var i = 0, l = doclet.type.names.length; i < l; i++) {
  76. if (doclet.type.names[i].toLowerCase() === 'function') {
  77. needsSig = true;
  78. break;
  79. }
  80. }
  81. }
  82. return needsSig;
  83. }
  84. function getSignatureAttributes(item) {
  85. var attributes = [];
  86. if (item.optional) {
  87. attributes.push('opt');
  88. }
  89. if (item.nullable === true) {
  90. attributes.push('nullable');
  91. } else if (item.nullable === false) {
  92. attributes.push('non-null');
  93. }
  94. return attributes;
  95. }
  96. function updateItemName(item) {
  97. var attributes = getSignatureAttributes(item);
  98. var itemName = item.name || '';
  99. if (item.variable) {
  100. itemName = '&hellip;' + itemName;
  101. }
  102. if (attributes && attributes.length) {
  103. itemName = util.format(
  104. '%s<span class="signature-attributes">%s</span>',
  105. itemName,
  106. attributes.join(', ')
  107. );
  108. }
  109. return itemName;
  110. }
  111. function addParamAttributes(params) {
  112. return params
  113. .filter(function(param) {
  114. return param.name && param.name.indexOf('.') === -1;
  115. })
  116. .map(updateItemName);
  117. }
  118. function buildItemTypeStrings(item) {
  119. var types = [];
  120. if (item && item.type && item.type.names) {
  121. item.type.names.forEach(function(name) {
  122. types.push(linkto(name, htmlsafe(name)));
  123. });
  124. }
  125. return types;
  126. }
  127. function buildAttribsString(attribs) {
  128. var attribsString = '';
  129. if (attribs && attribs.length) {
  130. attribsString = util.format(
  131. '<span class="icon green">%s</span> ',
  132. attribs.join('</span>, <span class="icon green">')
  133. );
  134. }
  135. return attribsString;
  136. }
  137. function addNonParamAttributes(items) {
  138. var types = [];
  139. items.forEach(function(item) {
  140. types = types.concat(buildItemTypeStrings(item));
  141. });
  142. return types;
  143. }
  144. function addSignatureParams(f) {
  145. var params = f.params ? addParamAttributes(f.params) : [];
  146. f.signature = util.format('%s(%s)', f.signature || '', params.join(', '));
  147. }
  148. function addSignatureReturns(f) {
  149. var attribs = [];
  150. var attribsString = '';
  151. var returnTypes = [];
  152. var returnTypesString = '';
  153. var source = f.yields || f.returns;
  154. // jam all the return-type attributes into an array. this could create odd results (for example,
  155. // if there are both nullable and non-nullable return types), but let's assume that most people
  156. // who use multiple @return tags aren't using Closure Compiler type annotations, and vice-versa.
  157. if (source) {
  158. source.forEach(function(item) {
  159. helper.getAttribs(item).forEach(function(attrib) {
  160. if (attribs.indexOf(attrib) === -1) {
  161. attribs.push(attrib);
  162. }
  163. });
  164. });
  165. attribsString = buildAttribsString(attribs);
  166. }
  167. if (source) {
  168. returnTypes = addNonParamAttributes(f.returns);
  169. }
  170. if (returnTypes.length) {
  171. returnTypesString = util.format(
  172. ' &rarr; %s{%s}',
  173. attribsString,
  174. returnTypes.join('|')
  175. );
  176. }
  177. f.signature =
  178. '<span class="signature">' +
  179. (f.signature || '') +
  180. '</span>' +
  181. '<span class="type-signature">' +
  182. returnTypesString +
  183. '</span> <br>';
  184. }
  185. function addSignatureTypes(f) {
  186. var types = f.type ? buildItemTypeStrings(f) : [];
  187. f.signature =
  188. (f.signature || '') +
  189. '<span class="type-signature">' +
  190. (types.length ? ' :' + types.join('|') : '') +
  191. '</span>';
  192. }
  193. function addAttribs(f) {
  194. var attribs = helper.getAttribs(f);
  195. var attribsString = buildAttribsString(attribs);
  196. f.attribs = util.format(
  197. '<span class="type-signature">%s</span>',
  198. attribsString
  199. );
  200. }
  201. function shortenPaths(files, commonPrefix) {
  202. Object.keys(files).forEach(function(file) {
  203. files[file].shortened = files[file].resolved
  204. .replace(commonPrefix, '')
  205. // always use forward slashes
  206. .replace(/\\/g, '/');
  207. });
  208. return files;
  209. }
  210. function getPathFromDoclet(doclet) {
  211. if (!doclet.meta) {
  212. return null;
  213. }
  214. return doclet.meta.path && doclet.meta.path !== 'null'
  215. ? path.join(doclet.meta.path, doclet.meta.filename)
  216. : doclet.meta.filename;
  217. }
  218. function generate(title, docs, filename, resolveLinks) {
  219. var docData;
  220. var html;
  221. var outpath;
  222. resolveLinks = resolveLinks === false ? false : true;
  223. docData = {
  224. env: env,
  225. isTutorial: false,
  226. title: title,
  227. docs: docs,
  228. package: find({ kind: 'package' })[0]
  229. };
  230. outpath = path.join(outdir, filename);
  231. html = view.render('container.tmpl', docData);
  232. if (resolveLinks) {
  233. html = helper.resolveLinks(html); // turn {@link foo} into <a href="foodoc.html">foo</a>
  234. }
  235. fs.writeFileSync(outpath, html, 'utf8');
  236. }
  237. function generateSourceFiles(sourceFiles, encoding) {
  238. encoding = encoding || 'utf8';
  239. Object.keys(sourceFiles).forEach(function(file) {
  240. var source;
  241. // links are keyed to the shortened path in each doclet's `meta.shortpath` property
  242. var sourceOutfile = helper.getUniqueFilename(sourceFiles[file].shortened);
  243. helper.registerLink(sourceFiles[file].shortened, sourceOutfile);
  244. try {
  245. source = {
  246. kind: 'source',
  247. code: helper.htmlsafe(
  248. fs.readFileSync(sourceFiles[file].resolved, encoding)
  249. )
  250. };
  251. } catch (e) {
  252. logger.error(
  253. 'Error while generating source file %s: %s',
  254. file,
  255. e.message
  256. );
  257. }
  258. generate(
  259. 'Source: ' + sourceFiles[file].shortened,
  260. [source],
  261. sourceOutfile,
  262. false
  263. );
  264. });
  265. }
  266. /**
  267. * Look for classes or functions with the same name as modules (which indicates that the module
  268. * exports only that class or function), then attach the classes or functions to the `module`
  269. * property of the appropriate module doclets. The name of each class or function is also updated
  270. * for display purposes. This function mutates the original arrays.
  271. *
  272. * @private
  273. * @param {Array.<module:jsdoc/doclet.Doclet>} doclets - The array of classes and functions to
  274. * check.
  275. * @param {Array.<module:jsdoc/doclet.Doclet>} modules - The array of module doclets to search.
  276. */
  277. function attachModuleSymbols(doclets, modules) {
  278. var symbols = {};
  279. // build a lookup table
  280. doclets.forEach(function(symbol) {
  281. symbols[symbol.longname] = symbols[symbol.longname] || [];
  282. symbols[symbol.longname].push(symbol);
  283. });
  284. return modules.map(function(module) {
  285. if (symbols[module.longname]) {
  286. module.modules = symbols[module.longname]
  287. // Only show symbols that have a description. Make an exception for classes, because
  288. // we want to show the constructor-signature heading no matter what.
  289. .filter(function(symbol) {
  290. return symbol.description || symbol.kind === 'class';
  291. })
  292. .map(function(symbol) {
  293. symbol = doop(symbol);
  294. if (symbol.kind === 'class' || symbol.kind === 'function') {
  295. symbol.name = symbol.name.replace('module:', '(require("') + '"))';
  296. }
  297. return symbol;
  298. });
  299. }
  300. });
  301. }
  302. /**
  303. * For lnb listing
  304. * -- 'Classes'
  305. * -- 'Namespaces'
  306. *
  307. * @param obj
  308. */
  309. function buildSubNav(obj) {
  310. var longname = obj.longname;
  311. var members = find({
  312. kind: 'member',
  313. memberof: longname
  314. });
  315. var methods = find({
  316. kind: 'function',
  317. memberof: longname
  318. });
  319. var events = find({
  320. kind: 'event',
  321. memberof: longname
  322. });
  323. var typedef = find({
  324. kind: 'typedef',
  325. memberof: longname
  326. });
  327. var html =
  328. '<div class="hidden" id="' + obj.longname.replace(/"/g, '_') + '_sub">';
  329. html += buildSubNavMembers(members, 'Members');
  330. html += buildSubNavMembers(methods, 'Methods');
  331. html += buildSubNavMembers(events, 'Events');
  332. html += buildSubNavMembers(typedef, 'Typedef');
  333. html += '</div>';
  334. return html;
  335. }
  336. function buildSubNavMembers(list, type) {
  337. var html = '';
  338. if (list.length) {
  339. html += '<div class="member-type">' + type + '</div>';
  340. html += '<ul class="inner">';
  341. list.forEach(function(item) {
  342. html += '<li>' + linkto(item.longname, item.name) + '</li>';
  343. });
  344. html += '</ul>';
  345. }
  346. return html;
  347. }
  348. function buildMemberNav(items, itemHeading, itemsSeen, linktoFn) {
  349. var nav = '';
  350. if (items.length) {
  351. var itemsNav = '';
  352. var className =
  353. itemHeading === tutorialsName ? 'lnb-examples hidden' : 'lnb-api hidden';
  354. var makeHtml = env.conf.templates.useCollapsibles
  355. ? makeCollapsibleItemHtmlInNav
  356. : makeItemHtmlInNav;
  357. items.forEach(function(item) {
  358. var linkHtml;
  359. if (!hasOwnProp.call(item, 'longname')) {
  360. itemsNav +=
  361. '<li>' + linktoFn('', item.name) + buildSubNav(item) + '</li>';
  362. } else if (!hasOwnProp.call(itemsSeen, item.longname)) {
  363. var displayName;
  364. if (
  365. env.conf.templates.default.useLongnameInNav ||
  366. item.kind === 'namespace'
  367. ) {
  368. displayName = item.longname;
  369. } else {
  370. displayName = item.name;
  371. }
  372. linkHtml = linktoFn(
  373. item.longname,
  374. displayName.replace(/\b(module|event):/g, '')
  375. );
  376. itemsNav += makeHtml(item, linkHtml);
  377. }
  378. itemsSeen[item.longname] = true;
  379. });
  380. if (itemsNav !== '') {
  381. nav +=
  382. '<div class="' +
  383. className +
  384. '"><h3>' +
  385. itemHeading +
  386. '</h3><ul>' +
  387. itemsNav +
  388. '</ul></div>';
  389. }
  390. }
  391. return nav;
  392. }
  393. function makeItemHtmlInNav(item, linkHtml) {
  394. return '<li>' + linkHtml + buildSubNav(item) + '</li>';
  395. }
  396. function makeCollapsibleItemHtmlInNav(item, linkHtml) {
  397. return (
  398. '<li>' +
  399. linkHtml +
  400. '<button type="button" class="hidden toggle-subnav btn btn-link">' +
  401. ' <span class="glyphicon glyphicon-plus"></span>' +
  402. '</button>' +
  403. buildSubNav(item) +
  404. '</li>'
  405. );
  406. }
  407. function linktoTutorial(longName, name) {
  408. return tutoriallink(name);
  409. }
  410. function linktoExternal(longName, name) {
  411. return linkto(longName, name.replace(/(^"|"$)/g, ''));
  412. }
  413. /**
  414. * Create the navigation sidebar.
  415. *
  416. * @param {object} members The members that will be used to create the sidebar.
  417. * @param {Array<object>} members.classes
  418. * @param {Array<object>} members.externals
  419. * @param {Array<object>} members.globals
  420. * @param {Array<object>} members.mixins
  421. * @param {Array<object>} members.modules
  422. * @param {Array<object>} members.namespaces
  423. * @param {Array<object>} members.tutorials
  424. * @param {Array<object>} members.events
  425. * @param {Array<object>} members.interfaces
  426. * @returns {string} The HTML for the navigation sidebar.
  427. */
  428. function buildNav(members) {
  429. var nav = '';
  430. var seen = {};
  431. var seenTutorials = {};
  432. nav += buildMemberNav(
  433. members.tutorials,
  434. tutorialsName,
  435. seenTutorials,
  436. linktoTutorial,
  437. true
  438. );
  439. nav += buildMemberNav(members.modules, 'Modules', {}, linkto);
  440. nav += buildMemberNav(members.externals, 'Externals', seen, linktoExternal);
  441. nav += buildMemberNav(members.classes, 'Classes', seen, linkto);
  442. nav += buildMemberNav(members.namespaces, 'Namespaces', seen, linkto);
  443. nav += buildMemberNav(members.mixins, 'Mixins', seen, linkto);
  444. nav += buildMemberNav(members.interfaces, 'Interfaces', seen, linkto);
  445. if (members.globals.length) {
  446. var globalNav = '';
  447. var useGlobalTitleLink = true;
  448. members.globals.forEach(function(g) {
  449. if (!hasOwnProp.call(seen, g.longname)) {
  450. // tuidoc
  451. // - Add global-typedef in hidden to search api.
  452. // - Default template did not add this.
  453. if (g.kind === 'typedef') {
  454. globalNav += '<li>' + linkto(g.longname, g.name) + '</li>';
  455. } else {
  456. globalNav += '<li>' + linkto(g.longname, g.name) + '</li>';
  457. useGlobalTitleLink = false;
  458. }
  459. }
  460. seen[g.longname] = true;
  461. });
  462. if (useGlobalTitleLink) {
  463. // turn the heading into a link so you can actually get to the global page
  464. nav +=
  465. '<div class="lnb-api hidden"><h3>' +
  466. linkto('global', 'Global') +
  467. '</h3></div>';
  468. } else {
  469. nav +=
  470. '<div class="lnb-api hidden"><h3>Global</h3><ul>' +
  471. globalNav +
  472. '</ul></div>';
  473. }
  474. }
  475. return nav;
  476. }
  477. /**
  478. * Look ma, it's cp -R.
  479. *
  480. * @param {string} src The path to the thing to copy.
  481. * @param {string} dest The path to the new copy.
  482. */
  483. var copyRecursiveSync = function(src, dest) {
  484. var contents,
  485. srcExists = fs.existsSync(src),
  486. destExists = fs.existsSync(dest),
  487. stats = srcExists && fs.statSync(src),
  488. isDirectory = srcExists && stats.isDirectory();
  489. if (srcExists) {
  490. if (isDirectory) {
  491. if (!destExists) {
  492. fs.mkdirSync(dest);
  493. }
  494. fs.readdirSync(src).forEach(function(childItemName) {
  495. copyRecursiveSync(
  496. path.join(src, childItemName),
  497. path.join(dest, childItemName)
  498. );
  499. });
  500. } else {
  501. contents = fs.readFileSync(src);
  502. fs.writeFileSync(dest, contents);
  503. }
  504. }
  505. };
  506. /**
  507. @param {TAFFY} taffyData See <http://taffydb.com/>.
  508. @param {object} opts
  509. @param {Tutorial} tutorials
  510. */
  511. exports.publish = function(taffyData, opts, tutorials) {
  512. data = taffyData;
  513. var conf = env.conf.templates || {};
  514. conf.default = conf.default || {};
  515. var templatePath = path.normalize(opts.template);
  516. view = new template.Template(path.join(templatePath, 'tmpl'));
  517. // claim some special filenames in advance, so the All-Powerful Overseer of Filename Uniqueness
  518. // doesn't try to hand them out later
  519. var indexUrl = helper.getUniqueFilename('index');
  520. // don't call registerLink() on this one! 'index' is also a valid longname
  521. var globalUrl = helper.getUniqueFilename('global');
  522. helper.registerLink('global', globalUrl);
  523. // set up templating
  524. view.layout = conf.default.layoutFile
  525. ? path.getResourcePath(
  526. path.dirname(conf.default.layoutFile),
  527. path.basename(conf.default.layoutFile)
  528. )
  529. : 'layout.tmpl';
  530. // set up tutorials for helper
  531. helper.setTutorials(tutorials);
  532. data = helper.prune(data);
  533. data.sort('longname, version, since');
  534. helper.addEventListeners(data);
  535. var sourceFiles = {};
  536. var sourceFilePaths = [];
  537. data().each(function(doclet) {
  538. doclet.attribs = '';
  539. if (doclet.examples) {
  540. doclet.examples = doclet.examples.map(function(example) {
  541. var caption, code;
  542. if (
  543. example.match(
  544. /^\s*<caption>([\s\S]+?)<\/caption>(\s*[\n\r])([\s\S]+)$/i
  545. )
  546. ) {
  547. caption = RegExp.$1;
  548. code = RegExp.$3;
  549. }
  550. return {
  551. caption: caption || '',
  552. code: code || example
  553. };
  554. });
  555. }
  556. if (doclet.see) {
  557. doclet.see.forEach(function(seeItem, i) {
  558. doclet.see[i] = hashToLink(doclet, seeItem);
  559. });
  560. }
  561. // build a list of source files
  562. var sourcePath;
  563. if (doclet.meta) {
  564. sourcePath = getPathFromDoclet(doclet);
  565. sourceFiles[sourcePath] = {
  566. resolved: sourcePath,
  567. shortened: null
  568. };
  569. if (sourceFilePaths.indexOf(sourcePath) === -1) {
  570. sourceFilePaths.push(sourcePath);
  571. }
  572. }
  573. });
  574. fs.mkPath(outdir);
  575. // copy the template's static files to outdir
  576. var fromDir = path.join(templatePath, 'static');
  577. var staticFiles = fs.ls(fromDir, 3);
  578. staticFiles.forEach(function(fileName) {
  579. var toDir = fs.toDir(fileName.replace(fromDir, outdir));
  580. fs.mkPath(toDir);
  581. fs.copyFileSync(fileName, toDir);
  582. });
  583. // copy user-specified static files to outdir
  584. var staticFilePaths;
  585. var staticFileFilter;
  586. var staticFileScanner;
  587. if (conf.default.staticFiles) {
  588. // The canonical property name is `include`. We accept `paths` for backwards compatibility
  589. // with a bug in JSDoc 3.2.x.
  590. staticFilePaths =
  591. conf.default.staticFiles.include || conf.default.staticFiles.paths || [];
  592. staticFileFilter = new (require('jsdoc/src/filter').Filter)(
  593. conf.default.staticFiles
  594. );
  595. staticFileScanner = new (require('jsdoc/src/scanner').Scanner)();
  596. staticFilePaths.forEach(function(filePath) {
  597. var extraStaticFiles;
  598. filePath = path.resolve(env.pwd, filePath);
  599. extraStaticFiles = staticFileScanner.scan(
  600. [filePath],
  601. 10,
  602. staticFileFilter
  603. );
  604. extraStaticFiles.forEach(function(fileName) {
  605. var sourcePath = fs.toDir(filePath);
  606. var toDir = fs.toDir(fileName.replace(sourcePath, outdir));
  607. fs.mkPath(toDir);
  608. fs.copyFileSync(fileName, toDir);
  609. });
  610. });
  611. }
  612. if (sourceFilePaths.length) {
  613. sourceFiles = shortenPaths(sourceFiles, path.commonPrefix(sourceFilePaths));
  614. }
  615. data().each(function(doclet) {
  616. var url = helper.createLink(doclet);
  617. helper.registerLink(doclet.longname, url);
  618. // add a shortened version of the full path
  619. var docletPath;
  620. if (doclet.meta) {
  621. docletPath = getPathFromDoclet(doclet);
  622. docletPath = sourceFiles[docletPath].shortened;
  623. if (docletPath) {
  624. doclet.meta.shortpath = docletPath;
  625. }
  626. }
  627. });
  628. data().each(function(doclet) {
  629. var url = helper.longnameToUrl[doclet.longname];
  630. if (url.indexOf('#') > -1) {
  631. doclet.id = helper.longnameToUrl[doclet.longname].split(/#/).pop();
  632. } else {
  633. doclet.id = doclet.name;
  634. }
  635. if (needsSignature(doclet)) {
  636. addSignatureParams(doclet);
  637. addSignatureReturns(doclet);
  638. addAttribs(doclet);
  639. }
  640. });
  641. // do this after the urls have all been generated
  642. data().each(function(doclet) {
  643. doclet.ancestors = getAncestorLinks(doclet);
  644. if (doclet.kind === 'member') {
  645. addSignatureTypes(doclet);
  646. addAttribs(doclet);
  647. }
  648. if (doclet.kind === 'constant') {
  649. addSignatureTypes(doclet);
  650. addAttribs(doclet);
  651. doclet.kind = 'member';
  652. }
  653. if (doclet.kind === 'typedef') {
  654. // console.log('doclet :>> ', doclet);
  655. if (doclet.scope === 'global') {
  656. doclet.scope = doclet.meta.filename;
  657. }
  658. addSignatureTypes(doclet);
  659. addAttribs(doclet);
  660. doclet.kind = 'member';
  661. }
  662. });
  663. var members = helper.getMembers(data);
  664. members.tutorials = tutorials.children;
  665. // output pretty-printed source files by default
  666. var outputSourceFiles =
  667. conf.default && conf.default.outputSourceFiles !== false ? true : false;
  668. // add template helpers
  669. view.find = find;
  670. view.linkto = linkto;
  671. view.resolveAuthorLinks = resolveAuthorLinks;
  672. view.tutoriallink = tutoriallink;
  673. view.htmlsafe = htmlsafe;
  674. view.outputSourceFiles = outputSourceFiles;
  675. // once for all
  676. view.nav = buildNav(members);
  677. attachModuleSymbols(find({ longname: { left: 'module:' } }), members.modules);
  678. // generate the pretty-printed source files first so other pages can link to them
  679. if (outputSourceFiles) {
  680. generateSourceFiles(sourceFiles, opts.encoding);
  681. }
  682. if (members.globals.length) {
  683. generate('Global', [{ kind: 'globalobj' }], globalUrl);
  684. }
  685. // index page displays information from package.json
  686. var packages = find({ kind: 'package' });
  687. generate(
  688. 'Home',
  689. packages.concat([
  690. {
  691. kind: 'mainpage',
  692. readme: opts.readme,
  693. longname: opts.mainpagetitle ? opts.mainpagetitle : 'Main Page'
  694. }
  695. ]),
  696. indexUrl
  697. );
  698. // set up the lists that we'll use to generate pages
  699. var classes = taffy(members.classes);
  700. var modules = taffy(members.modules);
  701. var namespaces = taffy(members.namespaces);
  702. var mixins = taffy(members.mixins);
  703. var externals = taffy(members.externals);
  704. var interfaces = taffy(members.interfaces);
  705. Object.keys(helper.longnameToUrl).forEach(function(longname) {
  706. var myModules = helper.find(modules, { longname: longname });
  707. if (myModules.length) {
  708. generate(
  709. 'Module: ' + myModules[0].name,
  710. myModules,
  711. helper.longnameToUrl[longname]
  712. );
  713. }
  714. var myClasses = helper.find(classes, { longname: longname });
  715. if (myClasses.length) {
  716. generate(
  717. 'Class: ' + myClasses[0].name,
  718. myClasses,
  719. helper.longnameToUrl[longname]
  720. );
  721. }
  722. var myNamespaces = helper.find(namespaces, { longname: longname });
  723. if (myNamespaces.length) {
  724. generate(
  725. 'Namespace: ' + myNamespaces[0].name,
  726. myNamespaces,
  727. helper.longnameToUrl[longname]
  728. );
  729. }
  730. var myMixins = helper.find(mixins, { longname: longname });
  731. if (myMixins.length) {
  732. generate(
  733. 'Mixin: ' + myMixins[0].name,
  734. myMixins,
  735. helper.longnameToUrl[longname]
  736. );
  737. }
  738. var myExternals = helper.find(externals, { longname: longname });
  739. if (myExternals.length) {
  740. generate(
  741. 'External: ' + myExternals[0].name,
  742. myExternals,
  743. helper.longnameToUrl[longname]
  744. );
  745. }
  746. var myInterfaces = helper.find(interfaces, { longname: longname });
  747. if (myInterfaces.length) {
  748. generate(
  749. 'Interface: ' + myInterfaces[0].name,
  750. myInterfaces,
  751. helper.longnameToUrl[longname]
  752. );
  753. }
  754. });
  755. if (env.opts.tutorials) {
  756. copyRecursiveSync(env.opts.tutorials, outdir + '/tutorials');
  757. }
  758. // TODO: move the tutorial functions to templateHelper.js
  759. function generateTutorial(
  760. title,
  761. tutorial,
  762. fileName,
  763. originalFileName,
  764. isHtmlTutorial
  765. ) {
  766. var tutorialData = {
  767. docs: null, // If there is no "docs" prop, Erros in layout.tmpl. (For left-nav member listing control)
  768. isTutorial: true,
  769. env: env,
  770. title: title,
  771. header: tutorial.title,
  772. children: tutorial.children,
  773. isHtmlTutorial: isHtmlTutorial,
  774. package: find({ kind: 'package' })[0]
  775. };
  776. if (isHtmlTutorial) {
  777. _.extend(
  778. tutorialData,
  779. generateHtmlTutorialData(tutorial, fileName, originalFileName)
  780. );
  781. } else {
  782. tutorialData.content = tutorial.parse();
  783. }
  784. var tutorialPath = path.join(outdir, fileName),
  785. html = view.render('tutorial.tmpl', tutorialData);
  786. // yes, you can use {@link} in tutorials too!
  787. html = helper.resolveLinks(html); // turn {@link foo} into <a href="foodoc.html">foo</a>
  788. fs.writeFileSync(tutorialPath, html, 'utf8');
  789. }
  790. function generateHtmlTutorialData(tutorial, filename, originalFileName) {
  791. var $ = cheerio.load(tutorial.parse(), {
  792. decodeEntities: false,
  793. normalizeWhitespace: false
  794. });
  795. return {
  796. codeHtml: htmlsafe($('div.code-html').html() || ''),
  797. codeJs: htmlsafe($('script.code-js').html() || ''),
  798. originalFileName: originalFileName
  799. };
  800. }
  801. // tutorials can have only one parent so there is no risk for loops
  802. function saveChildren(node) {
  803. node.children.forEach(function(child) {
  804. var originalFileName = child.name;
  805. var isHtmlTutorial = child.type === tutorial.TYPES.HTML;
  806. var title = 'Tutorial: ' + child.title;
  807. var fileName = helper.tutorialToUrl(child.name);
  808. generateTutorial(
  809. title,
  810. child,
  811. fileName,
  812. originalFileName,
  813. isHtmlTutorial
  814. );
  815. saveChildren(child);
  816. });
  817. }
  818. saveChildren(tutorials);
  819. };