compiler.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655
  1. /*!
  2. * Jade - Compiler
  3. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  4. * MIT Licensed
  5. */
  6. /**
  7. * Module dependencies.
  8. */
  9. var nodes = require('./nodes')
  10. , filters = require('./filters')
  11. , doctypes = require('./doctypes')
  12. , selfClosing = require('./self-closing')
  13. , runtime = require('./runtime')
  14. , utils = require('./utils');
  15. // if browser
  16. //
  17. // if (!Object.keys) {
  18. // Object.keys = function(obj){
  19. // var arr = [];
  20. // for (var key in obj) {
  21. // if (obj.hasOwnProperty(key)) {
  22. // arr.push(key);
  23. // }
  24. // }
  25. // return arr;
  26. // }
  27. // }
  28. //
  29. // if (!String.prototype.trimLeft) {
  30. // String.prototype.trimLeft = function(){
  31. // return this.replace(/^\s+/, '');
  32. // }
  33. // }
  34. //
  35. // end
  36. /**
  37. * Initialize `Compiler` with the given `node`.
  38. *
  39. * @param {Node} node
  40. * @param {Object} options
  41. * @api public
  42. */
  43. var Compiler = module.exports = function Compiler(node, options) {
  44. this.options = options = options || {};
  45. this.node = node;
  46. this.hasCompiledDoctype = false;
  47. this.hasCompiledTag = false;
  48. this.pp = options.pretty || false;
  49. this.debug = false !== options.compileDebug;
  50. this.indents = 0;
  51. this.parentIndents = 0;
  52. if (options.doctype) this.setDoctype(options.doctype);
  53. };
  54. /**
  55. * Compiler prototype.
  56. */
  57. Compiler.prototype = {
  58. /**
  59. * Compile parse tree to JavaScript.
  60. *
  61. * @api public
  62. */
  63. compile: function(){
  64. this.buf = ['var interp;'];
  65. if (this.pp) this.buf.push("var __indent = [];");
  66. this.lastBufferedIdx = -1;
  67. this.visit(this.node);
  68. return this.buf.join('\n');
  69. },
  70. /**
  71. * Sets the default doctype `name`. Sets terse mode to `true` when
  72. * html 5 is used, causing self-closing tags to end with ">" vs "/>",
  73. * and boolean attributes are not mirrored.
  74. *
  75. * @param {string} name
  76. * @api public
  77. */
  78. setDoctype: function(name){
  79. name = (name && name.toLowerCase()) || 'default';
  80. this.doctype = doctypes[name] || '<!DOCTYPE ' + name + '>';
  81. this.terse = this.doctype.toLowerCase() == '<!doctype html>';
  82. this.xml = 0 == this.doctype.indexOf('<?xml');
  83. },
  84. /**
  85. * Buffer the given `str` optionally escaped.
  86. *
  87. * @param {String} str
  88. * @param {Boolean} esc
  89. * @api public
  90. */
  91. buffer: function(str, esc){
  92. if (esc) str = utils.escape(str);
  93. if (this.lastBufferedIdx == this.buf.length) {
  94. this.lastBuffered += str;
  95. this.buf[this.lastBufferedIdx - 1] = "buf.push('" + this.lastBuffered + "');"
  96. } else {
  97. this.buf.push("buf.push('" + str + "');");
  98. this.lastBuffered = str;
  99. this.lastBufferedIdx = this.buf.length;
  100. }
  101. },
  102. /**
  103. * Buffer an indent based on the current `indent`
  104. * property and an additional `offset`.
  105. *
  106. * @param {Number} offset
  107. * @param {Boolean} newline
  108. * @api public
  109. */
  110. prettyIndent: function(offset, newline){
  111. offset = offset || 0;
  112. newline = newline ? '\\n' : '';
  113. this.buffer(newline + Array(this.indents + offset).join(' '));
  114. if (this.parentIndents)
  115. this.buf.push("buf.push.apply(buf, __indent);");
  116. },
  117. /**
  118. * Visit `node`.
  119. *
  120. * @param {Node} node
  121. * @api public
  122. */
  123. visit: function(node){
  124. var debug = this.debug;
  125. if (debug) {
  126. this.buf.push('__jade.unshift({ lineno: ' + node.line
  127. + ', filename: ' + (node.filename
  128. ? JSON.stringify(node.filename)
  129. : '__jade[0].filename')
  130. + ' });');
  131. }
  132. // Massive hack to fix our context
  133. // stack for - else[ if] etc
  134. if (false === node.debug && this.debug) {
  135. this.buf.pop();
  136. this.buf.pop();
  137. }
  138. this.visitNode(node);
  139. if (debug) this.buf.push('__jade.shift();');
  140. },
  141. /**
  142. * Visit `node`.
  143. *
  144. * @param {Node} node
  145. * @api public
  146. */
  147. visitNode: function(node){
  148. var name = node.constructor.name
  149. || node.constructor.toString().match(/function ([^(\s]+)()/)[1];
  150. return this['visit' + name](node);
  151. },
  152. /**
  153. * Visit case `node`.
  154. *
  155. * @param {Literal} node
  156. * @api public
  157. */
  158. visitCase: function(node){
  159. var _ = this.withinCase;
  160. this.withinCase = true;
  161. this.buf.push('switch (' + node.expr + '){');
  162. this.visit(node.block);
  163. this.buf.push('}');
  164. this.withinCase = _;
  165. },
  166. /**
  167. * Visit when `node`.
  168. *
  169. * @param {Literal} node
  170. * @api public
  171. */
  172. visitWhen: function(node){
  173. if ('default' == node.expr) {
  174. this.buf.push('default:');
  175. } else {
  176. this.buf.push('case ' + node.expr + ':');
  177. }
  178. this.visit(node.block);
  179. this.buf.push(' break;');
  180. },
  181. /**
  182. * Visit literal `node`.
  183. *
  184. * @param {Literal} node
  185. * @api public
  186. */
  187. visitLiteral: function(node){
  188. var str = node.str.replace(/\n/g, '\\\\n');
  189. this.buffer(str);
  190. },
  191. /**
  192. * Visit all nodes in `block`.
  193. *
  194. * @param {Block} block
  195. * @api public
  196. */
  197. visitBlock: function(block){
  198. var len = block.nodes.length
  199. , escape = this.escape
  200. , pp = this.pp
  201. // Block keyword has a special meaning in mixins
  202. if (this.parentIndents && block.mode) {
  203. if (pp) this.buf.push("__indent.push('" + Array(this.indents + 1).join(' ') + "');")
  204. this.buf.push('block && block();');
  205. if (pp) this.buf.push("__indent.pop();")
  206. return;
  207. }
  208. // Pretty print multi-line text
  209. if (pp && len > 1 && !escape && block.nodes[0].isText && block.nodes[1].isText)
  210. this.prettyIndent(1, true);
  211. for (var i = 0; i < len; ++i) {
  212. // Pretty print text
  213. if (pp && i > 0 && !escape && block.nodes[i].isText && block.nodes[i-1].isText)
  214. this.prettyIndent(1, false);
  215. this.visit(block.nodes[i]);
  216. // Multiple text nodes are separated by newlines
  217. if (block.nodes[i+1] && block.nodes[i].isText && block.nodes[i+1].isText)
  218. this.buffer('\\n');
  219. }
  220. },
  221. /**
  222. * Visit `doctype`. Sets terse mode to `true` when html 5
  223. * is used, causing self-closing tags to end with ">" vs "/>",
  224. * and boolean attributes are not mirrored.
  225. *
  226. * @param {Doctype} doctype
  227. * @api public
  228. */
  229. visitDoctype: function(doctype){
  230. if (doctype && (doctype.val || !this.doctype)) {
  231. this.setDoctype(doctype.val || 'default');
  232. }
  233. if (this.doctype) this.buffer(this.doctype);
  234. this.hasCompiledDoctype = true;
  235. },
  236. /**
  237. * Visit `mixin`, generating a function that
  238. * may be called within the template.
  239. *
  240. * @param {Mixin} mixin
  241. * @api public
  242. */
  243. visitMixin: function(mixin){
  244. var name = mixin.name.replace(/-/g, '_') + '_mixin'
  245. , args = mixin.args || ''
  246. , block = mixin.block
  247. , attrs = mixin.attrs
  248. , pp = this.pp;
  249. if (mixin.call) {
  250. if (pp) this.buf.push("__indent.push('" + Array(this.indents + 1).join(' ') + "');")
  251. if (block || attrs.length) {
  252. this.buf.push(name + '.call({');
  253. if (block) {
  254. this.buf.push('block: function(){');
  255. // Render block with no indents, dynamically added when rendered
  256. this.parentIndents++;
  257. var _indents = this.indents;
  258. this.indents = 0;
  259. this.visit(mixin.block);
  260. this.indents = _indents;
  261. this.parentIndents--;
  262. if (attrs.length) {
  263. this.buf.push('},');
  264. } else {
  265. this.buf.push('}');
  266. }
  267. }
  268. if (attrs.length) {
  269. var val = this.attrs(attrs);
  270. if (val.inherits) {
  271. this.buf.push('attributes: merge({' + val.buf
  272. + '}, attributes), escaped: merge(' + val.escaped + ', escaped, true)');
  273. } else {
  274. this.buf.push('attributes: {' + val.buf + '}, escaped: ' + val.escaped);
  275. }
  276. }
  277. if (args) {
  278. this.buf.push('}, ' + args + ');');
  279. } else {
  280. this.buf.push('});');
  281. }
  282. } else {
  283. this.buf.push(name + '(' + args + ');');
  284. }
  285. if (pp) this.buf.push("__indent.pop();")
  286. } else {
  287. this.buf.push('var ' + name + ' = function(' + args + '){');
  288. this.buf.push('var block = this.block, attributes = this.attributes || {}, escaped = this.escaped || {};');
  289. this.parentIndents++;
  290. this.visit(block);
  291. this.parentIndents--;
  292. this.buf.push('};');
  293. }
  294. },
  295. /**
  296. * Visit `tag` buffering tag markup, generating
  297. * attributes, visiting the `tag`'s code and block.
  298. *
  299. * @param {Tag} tag
  300. * @api public
  301. */
  302. visitTag: function(tag){
  303. this.indents++;
  304. var name = tag.name
  305. , pp = this.pp;
  306. if (tag.buffer) name = "' + (" + name + ") + '";
  307. if (!this.hasCompiledTag) {
  308. if (!this.hasCompiledDoctype && 'html' == name) {
  309. this.visitDoctype();
  310. }
  311. this.hasCompiledTag = true;
  312. }
  313. // pretty print
  314. if (pp && !tag.isInline())
  315. this.prettyIndent(0, true);
  316. if ((~selfClosing.indexOf(name) || tag.selfClosing) && !this.xml) {
  317. this.buffer('<' + name);
  318. this.visitAttributes(tag.attrs);
  319. this.terse
  320. ? this.buffer('>')
  321. : this.buffer('/>');
  322. } else {
  323. // Optimize attributes buffering
  324. if (tag.attrs.length) {
  325. this.buffer('<' + name);
  326. if (tag.attrs.length) this.visitAttributes(tag.attrs);
  327. this.buffer('>');
  328. } else {
  329. this.buffer('<' + name + '>');
  330. }
  331. if (tag.code) this.visitCode(tag.code);
  332. this.escape = 'pre' == tag.name;
  333. this.visit(tag.block);
  334. // pretty print
  335. if (pp && !tag.isInline() && 'pre' != tag.name && !tag.canInline())
  336. this.prettyIndent(0, true);
  337. this.buffer('</' + name + '>');
  338. }
  339. this.indents--;
  340. },
  341. /**
  342. * Visit `filter`, throwing when the filter does not exist.
  343. *
  344. * @param {Filter} filter
  345. * @api public
  346. */
  347. visitFilter: function(filter){
  348. var fn = filters[filter.name];
  349. // unknown filter
  350. if (!fn) throw new Error('unknown filter ":' + filter.name + '"');
  351. var text = filter.block.nodes.map(
  352. function(node){ return node.val; }
  353. ).join('\n');
  354. filter.attrs = filter.attrs || {};
  355. filter.attrs.filename = this.options.filename;
  356. this.buffer(utils.text(fn(text, filter.attrs)));
  357. },
  358. /**
  359. * Visit `text` node.
  360. *
  361. * @param {Text} text
  362. * @api public
  363. */
  364. visitText: function(text){
  365. text = utils.text(text.val.replace(/\\/g, '_SLASH_'));
  366. if (this.escape) text = escape(text);
  367. text = text.replace(/_SLASH_/g, '\\\\');
  368. this.buffer(text);
  369. },
  370. /**
  371. * Visit a `comment`, only buffering when the buffer flag is set.
  372. *
  373. * @param {Comment} comment
  374. * @api public
  375. */
  376. visitComment: function(comment){
  377. if (!comment.buffer) return;
  378. if (this.pp) this.prettyIndent(1, true);
  379. this.buffer('<!--' + utils.escape(comment.val) + '-->');
  380. },
  381. /**
  382. * Visit a `BlockComment`.
  383. *
  384. * @param {Comment} comment
  385. * @api public
  386. */
  387. visitBlockComment: function(comment){
  388. if (!comment.buffer) return;
  389. if (0 == comment.val.trim().indexOf('if')) {
  390. this.buffer('<!--[' + comment.val.trim() + ']>');
  391. this.visit(comment.block);
  392. this.buffer('<![endif]-->');
  393. } else {
  394. this.buffer('<!--' + comment.val);
  395. this.visit(comment.block);
  396. this.buffer('-->');
  397. }
  398. },
  399. /**
  400. * Visit `code`, respecting buffer / escape flags.
  401. * If the code is followed by a block, wrap it in
  402. * a self-calling function.
  403. *
  404. * @param {Code} code
  405. * @api public
  406. */
  407. visitCode: function(code){
  408. // Wrap code blocks with {}.
  409. // we only wrap unbuffered code blocks ATM
  410. // since they are usually flow control
  411. // Buffer code
  412. if (code.buffer) {
  413. var val = code.val.trimLeft();
  414. this.buf.push('var __val__ = ' + val);
  415. val = 'null == __val__ ? "" : __val__';
  416. if (code.escape) val = 'escape(' + val + ')';
  417. this.buf.push("buf.push(" + val + ");");
  418. } else {
  419. this.buf.push(code.val);
  420. }
  421. // Block support
  422. if (code.block) {
  423. if (!code.buffer) this.buf.push('{');
  424. this.visit(code.block);
  425. if (!code.buffer) this.buf.push('}');
  426. }
  427. },
  428. /**
  429. * Visit `each` block.
  430. *
  431. * @param {Each} each
  432. * @api public
  433. */
  434. visitEach: function(each){
  435. this.buf.push(''
  436. + '// iterate ' + each.obj + '\n'
  437. + ';(function(){\n'
  438. + ' if (\'number\' == typeof ' + each.obj + '.length) {\n');
  439. if (each.alternative) {
  440. this.buf.push(' if (' + each.obj + '.length) {');
  441. }
  442. this.buf.push(''
  443. + ' for (var ' + each.key + ' = 0, $$l = ' + each.obj + '.length; ' + each.key + ' < $$l; ' + each.key + '++) {\n'
  444. + ' var ' + each.val + ' = ' + each.obj + '[' + each.key + '];\n');
  445. this.visit(each.block);
  446. this.buf.push(' }\n');
  447. if (each.alternative) {
  448. this.buf.push(' } else {');
  449. this.visit(each.alternative);
  450. this.buf.push(' }');
  451. }
  452. this.buf.push(''
  453. + ' } else {\n'
  454. + ' var $$l = 0;\n'
  455. + ' for (var ' + each.key + ' in ' + each.obj + ') {\n'
  456. + ' $$l++;'
  457. // if browser
  458. // + ' if (' + each.obj + '.hasOwnProperty(' + each.key + ')){'
  459. // end
  460. + ' var ' + each.val + ' = ' + each.obj + '[' + each.key + '];\n');
  461. this.visit(each.block);
  462. // if browser
  463. // this.buf.push(' }\n');
  464. // end
  465. this.buf.push(' }\n');
  466. if (each.alternative) {
  467. this.buf.push(' if ($$l === 0) {');
  468. this.visit(each.alternative);
  469. this.buf.push(' }');
  470. }
  471. this.buf.push(' }\n}).call(this);\n');
  472. },
  473. /**
  474. * Visit `attrs`.
  475. *
  476. * @param {Array} attrs
  477. * @api public
  478. */
  479. visitAttributes: function(attrs){
  480. var val = this.attrs(attrs);
  481. if (val.inherits) {
  482. this.buf.push("buf.push(attrs(merge({ " + val.buf +
  483. " }, attributes), merge(" + val.escaped + ", escaped, true)));");
  484. } else if (val.constant) {
  485. eval('var buf={' + val.buf + '};');
  486. this.buffer(runtime.attrs(buf, JSON.parse(val.escaped)), true);
  487. } else {
  488. this.buf.push("buf.push(attrs({ " + val.buf + " }, " + val.escaped + "));");
  489. }
  490. },
  491. /**
  492. * Compile attributes.
  493. */
  494. attrs: function(attrs){
  495. var buf = []
  496. , classes = []
  497. , escaped = {}
  498. , constant = attrs.every(function(attr){ return isConstant(attr.val) })
  499. , inherits = false;
  500. if (this.terse) buf.push('terse: true');
  501. attrs.forEach(function(attr){
  502. if (attr.name == 'attributes') return inherits = true;
  503. escaped[attr.name] = attr.escaped;
  504. if (attr.name == 'class') {
  505. classes.push('(' + attr.val + ')');
  506. } else {
  507. var pair = "'" + attr.name + "':(" + attr.val + ')';
  508. buf.push(pair);
  509. }
  510. });
  511. if (classes.length) {
  512. classes = classes.join(" + ' ' + ");
  513. buf.push('"class": ' + classes);
  514. }
  515. return {
  516. buf: buf.join(', '),
  517. escaped: JSON.stringify(escaped),
  518. inherits: inherits,
  519. constant: constant
  520. };
  521. }
  522. };
  523. /**
  524. * Check if expression can be evaluated to a constant
  525. *
  526. * @param {String} expression
  527. * @return {Boolean}
  528. * @api private
  529. */
  530. function isConstant(val){
  531. // Check strings/literals
  532. if (/^ *("([^"\\]*(\\.[^"\\]*)*)"|'([^'\\]*(\\.[^'\\]*)*)'|true|false|null|undefined) *$/i.test(val))
  533. return true;
  534. // Check numbers
  535. if (!isNaN(Number(val)))
  536. return true;
  537. // Check arrays
  538. var matches;
  539. if (matches = /^ *\[(.*)\] *$/.exec(val))
  540. return matches[1].split(',').every(isConstant);
  541. return false;
  542. }
  543. /**
  544. * Escape the given string of `html`.
  545. *
  546. * @param {String} html
  547. * @return {String}
  548. * @api private
  549. */
  550. function escape(html){
  551. return String(html)
  552. .replace(/&(?!\w+;)/g, '&amp;')
  553. .replace(/</g, '&lt;')
  554. .replace(/>/g, '&gt;')
  555. .replace(/"/g, '&quot;');
  556. }