jade.js 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. /*!
  2. * Jade
  3. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  4. * MIT Licensed
  5. */
  6. /**
  7. * Module dependencies.
  8. */
  9. var Parser = require('./parser')
  10. , Lexer = require('./lexer')
  11. , Compiler = require('./compiler')
  12. , runtime = require('./runtime')
  13. // if node
  14. , fs = require('fs');
  15. // end
  16. /**
  17. * Library version.
  18. */
  19. exports.version = '0.28.2';
  20. /**
  21. * Expose self closing tags.
  22. */
  23. exports.selfClosing = require('./self-closing');
  24. /**
  25. * Default supported doctypes.
  26. */
  27. exports.doctypes = require('./doctypes');
  28. /**
  29. * Text filters.
  30. */
  31. exports.filters = require('./filters');
  32. /**
  33. * Utilities.
  34. */
  35. exports.utils = require('./utils');
  36. /**
  37. * Expose `Compiler`.
  38. */
  39. exports.Compiler = Compiler;
  40. /**
  41. * Expose `Parser`.
  42. */
  43. exports.Parser = Parser;
  44. /**
  45. * Expose `Lexer`.
  46. */
  47. exports.Lexer = Lexer;
  48. /**
  49. * Nodes.
  50. */
  51. exports.nodes = require('./nodes');
  52. /**
  53. * Jade runtime helpers.
  54. */
  55. exports.runtime = runtime;
  56. /**
  57. * Template function cache.
  58. */
  59. exports.cache = {};
  60. /**
  61. * Parse the given `str` of jade and return a function body.
  62. *
  63. * @param {String} str
  64. * @param {Object} options
  65. * @return {String}
  66. * @api private
  67. */
  68. function parse(str, options){
  69. try {
  70. // Parse
  71. var parser = new Parser(str, options.filename, options);
  72. // Compile
  73. var compiler = new (options.compiler || Compiler)(parser.parse(), options)
  74. , js = compiler.compile();
  75. // Debug compiler
  76. if (options.debug) {
  77. console.error('\nCompiled Function:\n\n\033[90m%s\033[0m', js.replace(/^/gm, ' '));
  78. }
  79. return ''
  80. + 'var buf = [];\n'
  81. + (options.self
  82. ? 'var self = locals || {};\n' + js
  83. : 'with (locals || {}) {\n' + js + '\n}\n')
  84. + 'return buf.join("");';
  85. } catch (err) {
  86. parser = parser.context();
  87. runtime.rethrow(err, parser.filename, parser.lexer.lineno);
  88. }
  89. }
  90. /**
  91. * Strip any UTF-8 BOM off of the start of `str`, if it exists.
  92. *
  93. * @param {String} str
  94. * @return {String}
  95. * @api private
  96. */
  97. function stripBOM(str){
  98. return 0xFEFF == str.charCodeAt(0)
  99. ? str.substring(1)
  100. : str;
  101. }
  102. /**
  103. * Compile a `Function` representation of the given jade `str`.
  104. *
  105. * Options:
  106. *
  107. * - `compileDebug` when `false` debugging code is stripped from the compiled template
  108. * - `client` when `true` the helper functions `escape()` etc will reference `jade.escape()`
  109. * for use with the Jade client-side runtime.js
  110. *
  111. * @param {String} str
  112. * @param {Options} options
  113. * @return {Function}
  114. * @api public
  115. */
  116. exports.compile = function(str, options){
  117. var options = options || {}
  118. , client = options.client
  119. , filename = options.filename
  120. ? JSON.stringify(options.filename)
  121. : 'undefined'
  122. , fn;
  123. str = stripBOM(String(str));
  124. if (options.compileDebug !== false) {
  125. fn = [
  126. 'var __jade = [{ lineno: 1, filename: ' + filename + ' }];'
  127. , 'try {'
  128. , parse(str, options)
  129. , '} catch (err) {'
  130. , ' rethrow(err, __jade[0].filename, __jade[0].lineno);'
  131. , '}'
  132. ].join('\n');
  133. } else {
  134. fn = parse(str, options);
  135. }
  136. if (client) {
  137. fn = 'attrs = attrs || jade.attrs; escape = escape || jade.escape; rethrow = rethrow || jade.rethrow; merge = merge || jade.merge;\n' + fn;
  138. }
  139. fn = new Function('locals, attrs, escape, rethrow, merge', fn);
  140. if (client) return fn;
  141. return function(locals){
  142. return fn(locals, runtime.attrs, runtime.escape, runtime.rethrow, runtime.merge);
  143. };
  144. };
  145. /**
  146. * Render the given `str` of jade and invoke
  147. * the callback `fn(err, str)`.
  148. *
  149. * Options:
  150. *
  151. * - `cache` enable template caching
  152. * - `filename` filename required for `include` / `extends` and caching
  153. *
  154. * @param {String} str
  155. * @param {Object|Function} options or fn
  156. * @param {Function} fn
  157. * @api public
  158. */
  159. exports.render = function(str, options, fn){
  160. // swap args
  161. if ('function' == typeof options) {
  162. fn = options, options = {};
  163. }
  164. // cache requires .filename
  165. if (options.cache && !options.filename) {
  166. return fn(new Error('the "filename" option is required for caching'));
  167. }
  168. try {
  169. var path = options.filename;
  170. var tmpl = options.cache
  171. ? exports.cache[path] || (exports.cache[path] = exports.compile(str, options))
  172. : exports.compile(str, options);
  173. fn(null, tmpl(options));
  174. } catch (err) {
  175. fn(err);
  176. }
  177. };
  178. /**
  179. * Render a Jade file at the given `path` and callback `fn(err, str)`.
  180. *
  181. * @param {String} path
  182. * @param {Object|Function} options or callback
  183. * @param {Function} fn
  184. * @api public
  185. */
  186. exports.renderFile = function(path, options, fn){
  187. var key = path + ':string';
  188. if ('function' == typeof options) {
  189. fn = options, options = {};
  190. }
  191. try {
  192. options.filename = path;
  193. var str = options.cache
  194. ? exports.cache[key] || (exports.cache[key] = fs.readFileSync(path, 'utf8'))
  195. : fs.readFileSync(path, 'utf8');
  196. exports.render(str, options, fn);
  197. } catch (err) {
  198. fn(err);
  199. }
  200. };
  201. /**
  202. * Express support.
  203. */
  204. exports.__express = exports.renderFile;