jade.js 74 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654
  1. (function() {
  2. // CommonJS require()
  3. function require(p){
  4. var path = require.resolve(p)
  5. , mod = require.modules[path];
  6. if (!mod) throw new Error('failed to require "' + p + '"');
  7. if (!mod.exports) {
  8. mod.exports = {};
  9. mod.call(mod.exports, mod, mod.exports, require.relative(path));
  10. }
  11. return mod.exports;
  12. }
  13. require.modules = {};
  14. require.resolve = function (path){
  15. var orig = path
  16. , reg = path + '.js'
  17. , index = path + '/index.js';
  18. return require.modules[reg] && reg
  19. || require.modules[index] && index
  20. || orig;
  21. };
  22. require.register = function (path, fn){
  23. require.modules[path] = fn;
  24. };
  25. require.relative = function (parent) {
  26. return function(p){
  27. if ('.' != p.charAt(0)) return require(p);
  28. var path = parent.split('/')
  29. , segs = p.split('/');
  30. path.pop();
  31. for (var i = 0; i < segs.length; i++) {
  32. var seg = segs[i];
  33. if ('..' == seg) path.pop();
  34. else if ('.' != seg) path.push(seg);
  35. }
  36. return require(path.join('/'));
  37. };
  38. };
  39. require.register("compiler.js", function(module, exports, require){
  40. /*!
  41. * Jade - Compiler
  42. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  43. * MIT Licensed
  44. */
  45. /**
  46. * Module dependencies.
  47. */
  48. var nodes = require('./nodes')
  49. , filters = require('./filters')
  50. , doctypes = require('./doctypes')
  51. , selfClosing = require('./self-closing')
  52. , runtime = require('./runtime')
  53. , utils = require('./utils');
  54. if (!Object.keys) {
  55. Object.keys = function(obj){
  56. var arr = [];
  57. for (var key in obj) {
  58. if (obj.hasOwnProperty(key)) {
  59. arr.push(key);
  60. }
  61. }
  62. return arr;
  63. }
  64. }
  65. if (!String.prototype.trimLeft) {
  66. String.prototype.trimLeft = function(){
  67. return this.replace(/^\s+/, '');
  68. }
  69. }
  70. /**
  71. * Initialize `Compiler` with the given `node`.
  72. *
  73. * @param {Node} node
  74. * @param {Object} options
  75. * @api public
  76. */
  77. var Compiler = module.exports = function Compiler(node, options) {
  78. this.options = options = options || {};
  79. this.node = node;
  80. this.hasCompiledDoctype = false;
  81. this.hasCompiledTag = false;
  82. this.pp = options.pretty || false;
  83. this.debug = false !== options.compileDebug;
  84. this.indents = 0;
  85. this.parentIndents = 0;
  86. if (options.doctype) this.setDoctype(options.doctype);
  87. };
  88. /**
  89. * Compiler prototype.
  90. */
  91. Compiler.prototype = {
  92. /**
  93. * Compile parse tree to JavaScript.
  94. *
  95. * @api public
  96. */
  97. compile: function(){
  98. this.buf = ['var interp;'];
  99. if (this.pp) this.buf.push("var __indent = [];");
  100. this.lastBufferedIdx = -1;
  101. this.visit(this.node);
  102. return this.buf.join('\n');
  103. },
  104. /**
  105. * Sets the default doctype `name`. Sets terse mode to `true` when
  106. * html 5 is used, causing self-closing tags to end with ">" vs "/>",
  107. * and boolean attributes are not mirrored.
  108. *
  109. * @param {string} name
  110. * @api public
  111. */
  112. setDoctype: function(name){
  113. name = (name && name.toLowerCase()) || 'default';
  114. this.doctype = doctypes[name] || '<!DOCTYPE ' + name + '>';
  115. this.terse = this.doctype.toLowerCase() == '<!doctype html>';
  116. this.xml = 0 == this.doctype.indexOf('<?xml');
  117. },
  118. /**
  119. * Buffer the given `str` optionally escaped.
  120. *
  121. * @param {String} str
  122. * @param {Boolean} esc
  123. * @api public
  124. */
  125. buffer: function(str, esc){
  126. if (esc) str = utils.escape(str);
  127. if (this.lastBufferedIdx == this.buf.length) {
  128. this.lastBuffered += str;
  129. this.buf[this.lastBufferedIdx - 1] = "buf.push('" + this.lastBuffered + "');"
  130. } else {
  131. this.buf.push("buf.push('" + str + "');");
  132. this.lastBuffered = str;
  133. this.lastBufferedIdx = this.buf.length;
  134. }
  135. },
  136. /**
  137. * Buffer an indent based on the current `indent`
  138. * property and an additional `offset`.
  139. *
  140. * @param {Number} offset
  141. * @param {Boolean} newline
  142. * @api public
  143. */
  144. prettyIndent: function(offset, newline){
  145. offset = offset || 0;
  146. newline = newline ? '\\n' : '';
  147. this.buffer(newline + Array(this.indents + offset).join(' '));
  148. if (this.parentIndents)
  149. this.buf.push("buf.push.apply(buf, __indent);");
  150. },
  151. /**
  152. * Visit `node`.
  153. *
  154. * @param {Node} node
  155. * @api public
  156. */
  157. visit: function(node){
  158. var debug = this.debug;
  159. if (debug) {
  160. this.buf.push('__jade.unshift({ lineno: ' + node.line
  161. + ', filename: ' + (node.filename
  162. ? JSON.stringify(node.filename)
  163. : '__jade[0].filename')
  164. + ' });');
  165. }
  166. // Massive hack to fix our context
  167. // stack for - else[ if] etc
  168. if (false === node.debug && this.debug) {
  169. this.buf.pop();
  170. this.buf.pop();
  171. }
  172. this.visitNode(node);
  173. if (debug) this.buf.push('__jade.shift();');
  174. },
  175. /**
  176. * Visit `node`.
  177. *
  178. * @param {Node} node
  179. * @api public
  180. */
  181. visitNode: function(node){
  182. var name = node.constructor.name
  183. || node.constructor.toString().match(/function ([^(\s]+)()/)[1];
  184. return this['visit' + name](node);
  185. },
  186. /**
  187. * Visit case `node`.
  188. *
  189. * @param {Literal} node
  190. * @api public
  191. */
  192. visitCase: function(node){
  193. var _ = this.withinCase;
  194. this.withinCase = true;
  195. this.buf.push('switch (' + node.expr + '){');
  196. this.visit(node.block);
  197. this.buf.push('}');
  198. this.withinCase = _;
  199. },
  200. /**
  201. * Visit when `node`.
  202. *
  203. * @param {Literal} node
  204. * @api public
  205. */
  206. visitWhen: function(node){
  207. if ('default' == node.expr) {
  208. this.buf.push('default:');
  209. } else {
  210. this.buf.push('case ' + node.expr + ':');
  211. }
  212. this.visit(node.block);
  213. this.buf.push(' break;');
  214. },
  215. /**
  216. * Visit literal `node`.
  217. *
  218. * @param {Literal} node
  219. * @api public
  220. */
  221. visitLiteral: function(node){
  222. var str = node.str.replace(/\n/g, '\\\\n');
  223. this.buffer(str);
  224. },
  225. /**
  226. * Visit all nodes in `block`.
  227. *
  228. * @param {Block} block
  229. * @api public
  230. */
  231. visitBlock: function(block){
  232. var len = block.nodes.length
  233. , escape = this.escape
  234. , pp = this.pp
  235. // Block keyword has a special meaning in mixins
  236. if (this.parentIndents && block.mode) {
  237. if (pp) this.buf.push("__indent.push('" + Array(this.indents + 1).join(' ') + "');")
  238. this.buf.push('block && block();');
  239. if (pp) this.buf.push("__indent.pop();")
  240. return;
  241. }
  242. // Pretty print multi-line text
  243. if (pp && len > 1 && !escape && block.nodes[0].isText && block.nodes[1].isText)
  244. this.prettyIndent(1, true);
  245. for (var i = 0; i < len; ++i) {
  246. // Pretty print text
  247. if (pp && i > 0 && !escape && block.nodes[i].isText && block.nodes[i-1].isText)
  248. this.prettyIndent(1, false);
  249. this.visit(block.nodes[i]);
  250. // Multiple text nodes are separated by newlines
  251. if (block.nodes[i+1] && block.nodes[i].isText && block.nodes[i+1].isText)
  252. this.buffer('\\n');
  253. }
  254. },
  255. /**
  256. * Visit `doctype`. Sets terse mode to `true` when html 5
  257. * is used, causing self-closing tags to end with ">" vs "/>",
  258. * and boolean attributes are not mirrored.
  259. *
  260. * @param {Doctype} doctype
  261. * @api public
  262. */
  263. visitDoctype: function(doctype){
  264. if (doctype && (doctype.val || !this.doctype)) {
  265. this.setDoctype(doctype.val || 'default');
  266. }
  267. if (this.doctype) this.buffer(this.doctype);
  268. this.hasCompiledDoctype = true;
  269. },
  270. /**
  271. * Visit `mixin`, generating a function that
  272. * may be called within the template.
  273. *
  274. * @param {Mixin} mixin
  275. * @api public
  276. */
  277. visitMixin: function(mixin){
  278. var name = mixin.name.replace(/-/g, '_') + '_mixin'
  279. , args = mixin.args || ''
  280. , block = mixin.block
  281. , attrs = mixin.attrs
  282. , pp = this.pp;
  283. if (mixin.call) {
  284. if (pp) this.buf.push("__indent.push('" + Array(this.indents + 1).join(' ') + "');")
  285. if (block || attrs.length) {
  286. this.buf.push(name + '.call({');
  287. if (block) {
  288. this.buf.push('block: function(){');
  289. // Render block with no indents, dynamically added when rendered
  290. this.parentIndents++;
  291. var _indents = this.indents;
  292. this.indents = 0;
  293. this.visit(mixin.block);
  294. this.indents = _indents;
  295. this.parentIndents--;
  296. if (attrs.length) {
  297. this.buf.push('},');
  298. } else {
  299. this.buf.push('}');
  300. }
  301. }
  302. if (attrs.length) {
  303. var val = this.attrs(attrs);
  304. if (val.inherits) {
  305. this.buf.push('attributes: merge({' + val.buf
  306. + '}, attributes), escaped: merge(' + val.escaped + ', escaped, true)');
  307. } else {
  308. this.buf.push('attributes: {' + val.buf + '}, escaped: ' + val.escaped);
  309. }
  310. }
  311. if (args) {
  312. this.buf.push('}, ' + args + ');');
  313. } else {
  314. this.buf.push('});');
  315. }
  316. } else {
  317. this.buf.push(name + '(' + args + ');');
  318. }
  319. if (pp) this.buf.push("__indent.pop();")
  320. } else {
  321. this.buf.push('var ' + name + ' = function(' + args + '){');
  322. this.buf.push('var block = this.block, attributes = this.attributes || {}, escaped = this.escaped || {};');
  323. this.parentIndents++;
  324. this.visit(block);
  325. this.parentIndents--;
  326. this.buf.push('};');
  327. }
  328. },
  329. /**
  330. * Visit `tag` buffering tag markup, generating
  331. * attributes, visiting the `tag`'s code and block.
  332. *
  333. * @param {Tag} tag
  334. * @api public
  335. */
  336. visitTag: function(tag){
  337. this.indents++;
  338. var name = tag.name
  339. , pp = this.pp;
  340. if (tag.buffer) name = "' + (" + name + ") + '";
  341. if (!this.hasCompiledTag) {
  342. if (!this.hasCompiledDoctype && 'html' == name) {
  343. this.visitDoctype();
  344. }
  345. this.hasCompiledTag = true;
  346. }
  347. // pretty print
  348. if (pp && !tag.isInline())
  349. this.prettyIndent(0, true);
  350. if ((~selfClosing.indexOf(name) || tag.selfClosing) && !this.xml) {
  351. this.buffer('<' + name);
  352. this.visitAttributes(tag.attrs);
  353. this.terse
  354. ? this.buffer('>')
  355. : this.buffer('/>');
  356. } else {
  357. // Optimize attributes buffering
  358. if (tag.attrs.length) {
  359. this.buffer('<' + name);
  360. if (tag.attrs.length) this.visitAttributes(tag.attrs);
  361. this.buffer('>');
  362. } else {
  363. this.buffer('<' + name + '>');
  364. }
  365. if (tag.code) this.visitCode(tag.code);
  366. this.escape = 'pre' == tag.name;
  367. this.visit(tag.block);
  368. // pretty print
  369. if (pp && !tag.isInline() && 'pre' != tag.name && !tag.canInline())
  370. this.prettyIndent(0, true);
  371. this.buffer('</' + name + '>');
  372. }
  373. this.indents--;
  374. },
  375. /**
  376. * Visit `filter`, throwing when the filter does not exist.
  377. *
  378. * @param {Filter} filter
  379. * @api public
  380. */
  381. visitFilter: function(filter){
  382. var fn = filters[filter.name];
  383. // unknown filter
  384. if (!fn) {
  385. if (filter.isASTFilter) {
  386. throw new Error('unknown ast filter "' + filter.name + ':"');
  387. } else {
  388. throw new Error('unknown filter ":' + filter.name + '"');
  389. }
  390. }
  391. if (filter.isASTFilter) {
  392. this.buf.push(fn(filter.block, this, filter.attrs));
  393. } else {
  394. var text = filter.block.nodes.map(function(node){ return node.val }).join('\n');
  395. filter.attrs = filter.attrs || {};
  396. filter.attrs.filename = this.options.filename;
  397. this.buffer(utils.text(fn(text, filter.attrs)));
  398. }
  399. },
  400. /**
  401. * Visit `text` node.
  402. *
  403. * @param {Text} text
  404. * @api public
  405. */
  406. visitText: function(text){
  407. text = utils.text(text.val.replace(/\\/g, '_SLASH_'));
  408. if (this.escape) text = escape(text);
  409. text = text.replace(/_SLASH_/g, '\\\\');
  410. this.buffer(text);
  411. },
  412. /**
  413. * Visit a `comment`, only buffering when the buffer flag is set.
  414. *
  415. * @param {Comment} comment
  416. * @api public
  417. */
  418. visitComment: function(comment){
  419. if (!comment.buffer) return;
  420. if (this.pp) this.prettyIndent(1, true);
  421. this.buffer('<!--' + utils.escape(comment.val) + '-->');
  422. },
  423. /**
  424. * Visit a `BlockComment`.
  425. *
  426. * @param {Comment} comment
  427. * @api public
  428. */
  429. visitBlockComment: function(comment){
  430. if (!comment.buffer) return;
  431. if (0 == comment.val.trim().indexOf('if')) {
  432. this.buffer('<!--[' + comment.val.trim() + ']>');
  433. this.visit(comment.block);
  434. this.buffer('<![endif]-->');
  435. } else {
  436. this.buffer('<!--' + comment.val);
  437. this.visit(comment.block);
  438. this.buffer('-->');
  439. }
  440. },
  441. /**
  442. * Visit `code`, respecting buffer / escape flags.
  443. * If the code is followed by a block, wrap it in
  444. * a self-calling function.
  445. *
  446. * @param {Code} code
  447. * @api public
  448. */
  449. visitCode: function(code){
  450. // Wrap code blocks with {}.
  451. // we only wrap unbuffered code blocks ATM
  452. // since they are usually flow control
  453. // Buffer code
  454. if (code.buffer) {
  455. var val = code.val.trimLeft();
  456. this.buf.push('var __val__ = ' + val);
  457. val = 'null == __val__ ? "" : __val__';
  458. if (code.escape) val = 'escape(' + val + ')';
  459. this.buf.push("buf.push(" + val + ");");
  460. } else {
  461. this.buf.push(code.val);
  462. }
  463. // Block support
  464. if (code.block) {
  465. if (!code.buffer) this.buf.push('{');
  466. this.visit(code.block);
  467. if (!code.buffer) this.buf.push('}');
  468. }
  469. },
  470. /**
  471. * Visit `each` block.
  472. *
  473. * @param {Each} each
  474. * @api public
  475. */
  476. visitEach: function(each){
  477. this.buf.push(''
  478. + '// iterate ' + each.obj + '\n'
  479. + ';(function(){\n'
  480. + ' if (\'number\' == typeof ' + each.obj + '.length) {\n');
  481. if (each.alternative) {
  482. this.buf.push(' if (' + each.obj + '.length) {');
  483. }
  484. this.buf.push(''
  485. + ' for (var ' + each.key + ' = 0, $$l = ' + each.obj + '.length; ' + each.key + ' < $$l; ' + each.key + '++) {\n'
  486. + ' var ' + each.val + ' = ' + each.obj + '[' + each.key + '];\n');
  487. this.visit(each.block);
  488. this.buf.push(' }\n');
  489. if (each.alternative) {
  490. this.buf.push(' } else {');
  491. this.visit(each.alternative);
  492. this.buf.push(' }');
  493. }
  494. this.buf.push(''
  495. + ' } else {\n'
  496. + ' var $$l = 0;\n'
  497. + ' for (var ' + each.key + ' in ' + each.obj + ') {\n'
  498. + ' $$l++;'
  499. + ' if (' + each.obj + '.hasOwnProperty(' + each.key + ')){'
  500. + ' var ' + each.val + ' = ' + each.obj + '[' + each.key + '];\n');
  501. this.visit(each.block);
  502. this.buf.push(' }\n');
  503. this.buf.push(' }\n');
  504. if (each.alternative) {
  505. this.buf.push(' if ($$l === 0) {');
  506. this.visit(each.alternative);
  507. this.buf.push(' }');
  508. }
  509. this.buf.push(' }\n}).call(this);\n');
  510. },
  511. /**
  512. * Visit `attrs`.
  513. *
  514. * @param {Array} attrs
  515. * @api public
  516. */
  517. visitAttributes: function(attrs){
  518. var val = this.attrs(attrs);
  519. if (val.inherits) {
  520. this.buf.push("buf.push(attrs(merge({ " + val.buf +
  521. " }, attributes), merge(" + val.escaped + ", escaped, true)));");
  522. } else if (val.constant) {
  523. eval('var buf={' + val.buf + '};');
  524. this.buffer(runtime.attrs(buf, JSON.parse(val.escaped)), true);
  525. } else {
  526. this.buf.push("buf.push(attrs({ " + val.buf + " }, " + val.escaped + "));");
  527. }
  528. },
  529. /**
  530. * Compile attributes.
  531. */
  532. attrs: function(attrs){
  533. var buf = []
  534. , classes = []
  535. , escaped = {}
  536. , constant = attrs.every(function(attr){ return isConstant(attr.val) })
  537. , inherits = false;
  538. if (this.terse) buf.push('terse: true');
  539. attrs.forEach(function(attr){
  540. if (attr.name == 'attributes') return inherits = true;
  541. escaped[attr.name] = attr.escaped;
  542. if (attr.name == 'class') {
  543. classes.push('(' + attr.val + ')');
  544. } else {
  545. var pair = "'" + attr.name + "':(" + attr.val + ')';
  546. buf.push(pair);
  547. }
  548. });
  549. if (classes.length) {
  550. classes = classes.join(" + ' ' + ");
  551. buf.push("class: " + classes);
  552. }
  553. return {
  554. buf: buf.join(', ').replace('class:', '"class":'),
  555. escaped: JSON.stringify(escaped),
  556. inherits: inherits,
  557. constant: constant
  558. };
  559. }
  560. };
  561. /**
  562. * Check if expression can be evaluated to a constant
  563. *
  564. * @param {String} expression
  565. * @return {Boolean}
  566. * @api private
  567. */
  568. function isConstant(val){
  569. // Check strings/literals
  570. if (/^ *("([^"\\]*(\\.[^"\\]*)*)"|'([^'\\]*(\\.[^'\\]*)*)'|true|false|null|undefined) *$/i.test(val))
  571. return true;
  572. // Check numbers
  573. if (!isNaN(Number(val)))
  574. return true;
  575. // Check arrays
  576. var matches;
  577. if (matches = /^ *\[(.*)\] *$/.exec(val))
  578. return matches[1].split(',').every(isConstant);
  579. return false;
  580. }
  581. /**
  582. * Escape the given string of `html`.
  583. *
  584. * @param {String} html
  585. * @return {String}
  586. * @api private
  587. */
  588. function escape(html){
  589. return String(html)
  590. .replace(/&(?!\w+;)/g, '&amp;')
  591. .replace(/</g, '&lt;')
  592. .replace(/>/g, '&gt;')
  593. .replace(/"/g, '&quot;');
  594. }
  595. }); // module: compiler.js
  596. require.register("doctypes.js", function(module, exports, require){
  597. /*!
  598. * Jade - doctypes
  599. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  600. * MIT Licensed
  601. */
  602. module.exports = {
  603. '5': '<!DOCTYPE html>'
  604. , 'default': '<!DOCTYPE html>'
  605. , 'xml': '<?xml version="1.0" encoding="utf-8" ?>'
  606. , 'transitional': '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">'
  607. , 'strict': '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">'
  608. , 'frameset': '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Frameset//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-frameset.dtd">'
  609. , '1.1': '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" "http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">'
  610. , 'basic': '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML Basic 1.1//EN" "http://www.w3.org/TR/xhtml-basic/xhtml-basic11.dtd">'
  611. , 'mobile': '<!DOCTYPE html PUBLIC "-//WAPFORUM//DTD XHTML Mobile 1.2//EN" "http://www.openmobilealliance.org/tech/DTD/xhtml-mobile12.dtd">'
  612. };
  613. }); // module: doctypes.js
  614. require.register("filters.js", function(module, exports, require){
  615. /*!
  616. * Jade - filters
  617. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  618. * MIT Licensed
  619. */
  620. module.exports = {
  621. /**
  622. * Wrap text with CDATA block.
  623. */
  624. cdata: function(str){
  625. return '<![CDATA[\\n' + str + '\\n]]>';
  626. },
  627. /**
  628. * Transform sass to css, wrapped in style tags.
  629. */
  630. sass: function(str){
  631. str = str.replace(/\\n/g, '\n');
  632. var sass = require('sass').render(str).replace(/\n/g, '\\n');
  633. return '<style type="text/css">' + sass + '</style>';
  634. },
  635. /**
  636. * Transform stylus to css, wrapped in style tags.
  637. */
  638. stylus: function(str, options){
  639. var ret;
  640. str = str.replace(/\\n/g, '\n');
  641. var stylus = require('stylus');
  642. stylus(str, options).render(function(err, css){
  643. if (err) throw err;
  644. ret = css.replace(/\n/g, '\\n');
  645. });
  646. return '<style type="text/css">' + ret + '</style>';
  647. },
  648. /**
  649. * Transform less to css, wrapped in style tags.
  650. */
  651. less: function(str){
  652. var ret;
  653. str = str.replace(/\\n/g, '\n');
  654. require('less').render(str, function(err, css){
  655. if (err) throw err;
  656. ret = '<style type="text/css">' + css.replace(/\n/g, '\\n') + '</style>';
  657. });
  658. return ret;
  659. },
  660. /**
  661. * Transform markdown to html.
  662. */
  663. markdown: function(str){
  664. var md;
  665. // support markdown / discount
  666. try {
  667. md = require('markdown');
  668. } catch (err){
  669. try {
  670. md = require('discount');
  671. } catch (err) {
  672. try {
  673. md = require('markdown-js');
  674. } catch (err) {
  675. try {
  676. md = require('marked');
  677. } catch (err) {
  678. throw new
  679. Error('Cannot find markdown library, install markdown, discount, or marked.');
  680. }
  681. }
  682. }
  683. }
  684. str = str.replace(/\\n/g, '\n');
  685. return md.parse(str).replace(/\n/g, '\\n').replace(/'/g,'&#39;');
  686. },
  687. /**
  688. * Transform coffeescript to javascript.
  689. */
  690. coffeescript: function(str){
  691. var js = require('coffee-script').compile(str).replace(/\\/g, '\\\\').replace(/\n/g, '\\n');
  692. return '<script type="text/javascript">\\n' + js + '</script>';
  693. }
  694. };
  695. }); // module: filters.js
  696. require.register("inline-tags.js", function(module, exports, require){
  697. /*!
  698. * Jade - inline tags
  699. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  700. * MIT Licensed
  701. */
  702. module.exports = [
  703. 'a'
  704. , 'abbr'
  705. , 'acronym'
  706. , 'b'
  707. , 'br'
  708. , 'code'
  709. , 'em'
  710. , 'font'
  711. , 'i'
  712. , 'img'
  713. , 'ins'
  714. , 'kbd'
  715. , 'map'
  716. , 'samp'
  717. , 'small'
  718. , 'span'
  719. , 'strong'
  720. , 'sub'
  721. , 'sup'
  722. ];
  723. }); // module: inline-tags.js
  724. require.register("jade.js", function(module, exports, require){
  725. /*!
  726. * Jade
  727. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  728. * MIT Licensed
  729. */
  730. /**
  731. * Module dependencies.
  732. */
  733. var Parser = require('./parser')
  734. , Lexer = require('./lexer')
  735. , Compiler = require('./compiler')
  736. , runtime = require('./runtime')
  737. /**
  738. * Library version.
  739. */
  740. exports.version = '0.27.6';
  741. /**
  742. * Expose self closing tags.
  743. */
  744. exports.selfClosing = require('./self-closing');
  745. /**
  746. * Default supported doctypes.
  747. */
  748. exports.doctypes = require('./doctypes');
  749. /**
  750. * Text filters.
  751. */
  752. exports.filters = require('./filters');
  753. /**
  754. * Utilities.
  755. */
  756. exports.utils = require('./utils');
  757. /**
  758. * Expose `Compiler`.
  759. */
  760. exports.Compiler = Compiler;
  761. /**
  762. * Expose `Parser`.
  763. */
  764. exports.Parser = Parser;
  765. /**
  766. * Expose `Lexer`.
  767. */
  768. exports.Lexer = Lexer;
  769. /**
  770. * Nodes.
  771. */
  772. exports.nodes = require('./nodes');
  773. /**
  774. * Jade runtime helpers.
  775. */
  776. exports.runtime = runtime;
  777. /**
  778. * Template function cache.
  779. */
  780. exports.cache = {};
  781. /**
  782. * Parse the given `str` of jade and return a function body.
  783. *
  784. * @param {String} str
  785. * @param {Object} options
  786. * @return {String}
  787. * @api private
  788. */
  789. function parse(str, options){
  790. try {
  791. // Parse
  792. var parser = new Parser(str, options.filename, options);
  793. // Compile
  794. var compiler = new (options.compiler || Compiler)(parser.parse(), options)
  795. , js = compiler.compile();
  796. // Debug compiler
  797. if (options.debug) {
  798. console.error('\nCompiled Function:\n\n\033[90m%s\033[0m', js.replace(/^/gm, ' '));
  799. }
  800. return ''
  801. + 'var buf = [];\n'
  802. + (options.self
  803. ? 'var self = locals || {};\n' + js
  804. : 'with (locals || {}) {\n' + js + '\n}\n')
  805. + 'return buf.join("");';
  806. } catch (err) {
  807. parser = parser.context();
  808. runtime.rethrow(err, parser.filename, parser.lexer.lineno);
  809. }
  810. }
  811. /**
  812. * Strip any UTF-8 BOM off of the start of `str`, if it exists.
  813. *
  814. * @param {String} str
  815. * @return {String}
  816. * @api private
  817. */
  818. function stripBOM(str){
  819. return 0xFEFF == str.charCodeAt(0)
  820. ? str.substring(1)
  821. : str;
  822. }
  823. /**
  824. * Compile a `Function` representation of the given jade `str`.
  825. *
  826. * Options:
  827. *
  828. * - `compileDebug` when `false` debugging code is stripped from the compiled template
  829. * - `client` when `true` the helper functions `escape()` etc will reference `jade.escape()`
  830. * for use with the Jade client-side runtime.js
  831. *
  832. * @param {String} str
  833. * @param {Options} options
  834. * @return {Function}
  835. * @api public
  836. */
  837. exports.compile = function(str, options){
  838. var options = options || {}
  839. , client = options.client
  840. , filename = options.filename
  841. ? JSON.stringify(options.filename)
  842. : 'undefined'
  843. , fn;
  844. str = stripBOM(String(str));
  845. if (options.compileDebug !== false) {
  846. fn = [
  847. 'var __jade = [{ lineno: 1, filename: ' + filename + ' }];'
  848. , 'try {'
  849. , parse(str, options)
  850. , '} catch (err) {'
  851. , ' rethrow(err, __jade[0].filename, __jade[0].lineno);'
  852. , '}'
  853. ].join('\n');
  854. } else {
  855. fn = parse(str, options);
  856. }
  857. if (client) {
  858. fn = 'attrs = attrs || jade.attrs; escape = escape || jade.escape; rethrow = rethrow || jade.rethrow; merge = merge || jade.merge;\n' + fn;
  859. }
  860. fn = new Function('locals, attrs, escape, rethrow, merge', fn);
  861. if (client) return fn;
  862. return function(locals){
  863. return fn(locals, runtime.attrs, runtime.escape, runtime.rethrow, runtime.merge);
  864. };
  865. };
  866. /**
  867. * Render the given `str` of jade and invoke
  868. * the callback `fn(err, str)`.
  869. *
  870. * Options:
  871. *
  872. * - `cache` enable template caching
  873. * - `filename` filename required for `include` / `extends` and caching
  874. *
  875. * @param {String} str
  876. * @param {Object|Function} options or fn
  877. * @param {Function} fn
  878. * @api public
  879. */
  880. exports.render = function(str, options, fn){
  881. // swap args
  882. if ('function' == typeof options) {
  883. fn = options, options = {};
  884. }
  885. // cache requires .filename
  886. if (options.cache && !options.filename) {
  887. return fn(new Error('the "filename" option is required for caching'));
  888. }
  889. try {
  890. var path = options.filename;
  891. var tmpl = options.cache
  892. ? exports.cache[path] || (exports.cache[path] = exports.compile(str, options))
  893. : exports.compile(str, options);
  894. fn(null, tmpl(options));
  895. } catch (err) {
  896. fn(err);
  897. }
  898. };
  899. /**
  900. * Render a Jade file at the given `path` and callback `fn(err, str)`.
  901. *
  902. * @param {String} path
  903. * @param {Object|Function} options or callback
  904. * @param {Function} fn
  905. * @api public
  906. */
  907. exports.renderFile = function(path, options, fn){
  908. var key = path + ':string';
  909. if ('function' == typeof options) {
  910. fn = options, options = {};
  911. }
  912. try {
  913. options.filename = path;
  914. var str = options.cache
  915. ? exports.cache[key] || (exports.cache[key] = fs.readFileSync(path, 'utf8'))
  916. : fs.readFileSync(path, 'utf8');
  917. exports.render(str, options, fn);
  918. } catch (err) {
  919. fn(err);
  920. }
  921. };
  922. /**
  923. * Express support.
  924. */
  925. exports.__express = exports.renderFile;
  926. }); // module: jade.js
  927. require.register("lexer.js", function(module, exports, require){
  928. /*!
  929. * Jade - Lexer
  930. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  931. * MIT Licensed
  932. */
  933. var utils = require('./utils');
  934. /**
  935. * Initialize `Lexer` with the given `str`.
  936. *
  937. * Options:
  938. *
  939. * - `colons` allow colons for attr delimiters
  940. *
  941. * @param {String} str
  942. * @param {Object} options
  943. * @api private
  944. */
  945. var Lexer = module.exports = function Lexer(str, options) {
  946. options = options || {};
  947. this.input = str.replace(/\r\n|\r/g, '\n');
  948. this.colons = options.colons;
  949. this.deferredTokens = [];
  950. this.lastIndents = 0;
  951. this.lineno = 1;
  952. this.stash = [];
  953. this.indentStack = [];
  954. this.indentRe = null;
  955. this.pipeless = false;
  956. };
  957. /**
  958. * Lexer prototype.
  959. */
  960. Lexer.prototype = {
  961. /**
  962. * Construct a token with the given `type` and `val`.
  963. *
  964. * @param {String} type
  965. * @param {String} val
  966. * @return {Object}
  967. * @api private
  968. */
  969. tok: function(type, val){
  970. return {
  971. type: type
  972. , line: this.lineno
  973. , val: val
  974. }
  975. },
  976. /**
  977. * Consume the given `len` of input.
  978. *
  979. * @param {Number} len
  980. * @api private
  981. */
  982. consume: function(len){
  983. this.input = this.input.substr(len);
  984. },
  985. /**
  986. * Scan for `type` with the given `regexp`.
  987. *
  988. * @param {String} type
  989. * @param {RegExp} regexp
  990. * @return {Object}
  991. * @api private
  992. */
  993. scan: function(regexp, type){
  994. var captures;
  995. if (captures = regexp.exec(this.input)) {
  996. this.consume(captures[0].length);
  997. return this.tok(type, captures[1]);
  998. }
  999. },
  1000. /**
  1001. * Defer the given `tok`.
  1002. *
  1003. * @param {Object} tok
  1004. * @api private
  1005. */
  1006. defer: function(tok){
  1007. this.deferredTokens.push(tok);
  1008. },
  1009. /**
  1010. * Lookahead `n` tokens.
  1011. *
  1012. * @param {Number} n
  1013. * @return {Object}
  1014. * @api private
  1015. */
  1016. lookahead: function(n){
  1017. var fetch = n - this.stash.length;
  1018. while (fetch-- > 0) this.stash.push(this.next());
  1019. return this.stash[--n];
  1020. },
  1021. /**
  1022. * Return the indexOf `start` / `end` delimiters.
  1023. *
  1024. * @param {String} start
  1025. * @param {String} end
  1026. * @return {Number}
  1027. * @api private
  1028. */
  1029. indexOfDelimiters: function(start, end){
  1030. var str = this.input
  1031. , nstart = 0
  1032. , nend = 0
  1033. , pos = 0;
  1034. for (var i = 0, len = str.length; i < len; ++i) {
  1035. if (start == str.charAt(i)) {
  1036. ++nstart;
  1037. } else if (end == str.charAt(i)) {
  1038. if (++nend == nstart) {
  1039. pos = i;
  1040. break;
  1041. }
  1042. }
  1043. }
  1044. return pos;
  1045. },
  1046. /**
  1047. * Stashed token.
  1048. */
  1049. stashed: function() {
  1050. return this.stash.length
  1051. && this.stash.shift();
  1052. },
  1053. /**
  1054. * Deferred token.
  1055. */
  1056. deferred: function() {
  1057. return this.deferredTokens.length
  1058. && this.deferredTokens.shift();
  1059. },
  1060. /**
  1061. * end-of-source.
  1062. */
  1063. eos: function() {
  1064. if (this.input.length) return;
  1065. if (this.indentStack.length) {
  1066. this.indentStack.shift();
  1067. return this.tok('outdent');
  1068. } else {
  1069. return this.tok('eos');
  1070. }
  1071. },
  1072. /**
  1073. * Blank line.
  1074. */
  1075. blank: function() {
  1076. var captures;
  1077. if (captures = /^\n *\n/.exec(this.input)) {
  1078. this.consume(captures[0].length - 1);
  1079. ++this.lineno;
  1080. if (this.pipeless) return this.tok('text', '');
  1081. return this.next();
  1082. }
  1083. },
  1084. /**
  1085. * Comment.
  1086. */
  1087. comment: function() {
  1088. var captures;
  1089. if (captures = /^ *\/\/(-)?([^\n]*)/.exec(this.input)) {
  1090. this.consume(captures[0].length);
  1091. var tok = this.tok('comment', captures[2]);
  1092. tok.buffer = '-' != captures[1];
  1093. return tok;
  1094. }
  1095. },
  1096. /**
  1097. * Interpolated tag.
  1098. */
  1099. interpolation: function() {
  1100. var captures;
  1101. if (captures = /^#\{(.*?)\}/.exec(this.input)) {
  1102. this.consume(captures[0].length);
  1103. return this.tok('interpolation', captures[1]);
  1104. }
  1105. },
  1106. /**
  1107. * Tag.
  1108. */
  1109. tag: function() {
  1110. var captures;
  1111. if (captures = /^(\w[-:\w]*)(\/?)/.exec(this.input)) {
  1112. this.consume(captures[0].length);
  1113. var tok, name = captures[1];
  1114. if (':' == name[name.length - 1]) {
  1115. name = name.slice(0, -1);
  1116. tok = this.tok('tag', name);
  1117. this.defer(this.tok(':'));
  1118. while (' ' == this.input[0]) this.input = this.input.substr(1);
  1119. } else {
  1120. tok = this.tok('tag', name);
  1121. }
  1122. tok.selfClosing = !! captures[2];
  1123. return tok;
  1124. }
  1125. },
  1126. /**
  1127. * Filter.
  1128. */
  1129. filter: function() {
  1130. return this.scan(/^:(\w+)/, 'filter');
  1131. },
  1132. /**
  1133. * Doctype.
  1134. */
  1135. doctype: function() {
  1136. return this.scan(/^(?:!!!|doctype) *([^\n]+)?/, 'doctype');
  1137. },
  1138. /**
  1139. * Id.
  1140. */
  1141. id: function() {
  1142. return this.scan(/^#([\w-]+)/, 'id');
  1143. },
  1144. /**
  1145. * Class.
  1146. */
  1147. className: function() {
  1148. return this.scan(/^\.([\w-]+)/, 'class');
  1149. },
  1150. /**
  1151. * Text.
  1152. */
  1153. text: function() {
  1154. return this.scan(/^(?:\| ?| ?)?([^\n]+)/, 'text');
  1155. },
  1156. /**
  1157. * Extends.
  1158. */
  1159. "extends": function() {
  1160. return this.scan(/^extends? +([^\n]+)/, 'extends');
  1161. },
  1162. /**
  1163. * Block prepend.
  1164. */
  1165. prepend: function() {
  1166. var captures;
  1167. if (captures = /^prepend +([^\n]+)/.exec(this.input)) {
  1168. this.consume(captures[0].length);
  1169. var mode = 'prepend'
  1170. , name = captures[1]
  1171. , tok = this.tok('block', name);
  1172. tok.mode = mode;
  1173. return tok;
  1174. }
  1175. },
  1176. /**
  1177. * Block append.
  1178. */
  1179. append: function() {
  1180. var captures;
  1181. if (captures = /^append +([^\n]+)/.exec(this.input)) {
  1182. this.consume(captures[0].length);
  1183. var mode = 'append'
  1184. , name = captures[1]
  1185. , tok = this.tok('block', name);
  1186. tok.mode = mode;
  1187. return tok;
  1188. }
  1189. },
  1190. /**
  1191. * Block.
  1192. */
  1193. block: function() {
  1194. var captures;
  1195. if (captures = /^block\b *(?:(prepend|append) +)?([^\n]*)/.exec(this.input)) {
  1196. this.consume(captures[0].length);
  1197. var mode = captures[1] || 'replace'
  1198. , name = captures[2]
  1199. , tok = this.tok('block', name);
  1200. tok.mode = mode;
  1201. return tok;
  1202. }
  1203. },
  1204. /**
  1205. * Yield.
  1206. */
  1207. yield: function() {
  1208. return this.scan(/^yield */, 'yield');
  1209. },
  1210. /**
  1211. * Include.
  1212. */
  1213. include: function() {
  1214. return this.scan(/^include +([^\n]+)/, 'include');
  1215. },
  1216. /**
  1217. * Case.
  1218. */
  1219. "case": function() {
  1220. return this.scan(/^case +([^\n]+)/, 'case');
  1221. },
  1222. /**
  1223. * When.
  1224. */
  1225. when: function() {
  1226. return this.scan(/^when +([^:\n]+)/, 'when');
  1227. },
  1228. /**
  1229. * Default.
  1230. */
  1231. "default": function() {
  1232. return this.scan(/^default */, 'default');
  1233. },
  1234. /**
  1235. * Assignment.
  1236. */
  1237. assignment: function() {
  1238. var captures;
  1239. if (captures = /^(\w+) += *([^;\n]+)( *;? *)/.exec(this.input)) {
  1240. this.consume(captures[0].length);
  1241. var name = captures[1]
  1242. , val = captures[2];
  1243. return this.tok('code', 'var ' + name + ' = (' + val + ');');
  1244. }
  1245. },
  1246. /**
  1247. * Call mixin.
  1248. */
  1249. call: function(){
  1250. var captures;
  1251. if (captures = /^\+([-\w]+)/.exec(this.input)) {
  1252. this.consume(captures[0].length);
  1253. var tok = this.tok('call', captures[1]);
  1254. // Check for args (not attributes)
  1255. if (captures = /^ *\((.*?)\)/.exec(this.input)) {
  1256. if (!/^ *[-\w]+ *=/.test(captures[1])) {
  1257. this.consume(captures[0].length);
  1258. tok.args = captures[1];
  1259. }
  1260. }
  1261. return tok;
  1262. }
  1263. },
  1264. /**
  1265. * Mixin.
  1266. */
  1267. mixin: function(){
  1268. var captures;
  1269. if (captures = /^mixin +([-\w]+)(?: *\((.*)\))?/.exec(this.input)) {
  1270. this.consume(captures[0].length);
  1271. var tok = this.tok('mixin', captures[1]);
  1272. tok.args = captures[2];
  1273. return tok;
  1274. }
  1275. },
  1276. /**
  1277. * Conditional.
  1278. */
  1279. conditional: function() {
  1280. var captures;
  1281. if (captures = /^(if|unless|else if|else)\b([^\n]*)/.exec(this.input)) {
  1282. this.consume(captures[0].length);
  1283. var type = captures[1]
  1284. , js = captures[2];
  1285. switch (type) {
  1286. case 'if': js = 'if (' + js + ')'; break;
  1287. case 'unless': js = 'if (!(' + js + '))'; break;
  1288. case 'else if': js = 'else if (' + js + ')'; break;
  1289. case 'else': js = 'else'; break;
  1290. }
  1291. return this.tok('code', js);
  1292. }
  1293. },
  1294. /**
  1295. * While.
  1296. */
  1297. "while": function() {
  1298. var captures;
  1299. if (captures = /^while +([^\n]+)/.exec(this.input)) {
  1300. this.consume(captures[0].length);
  1301. return this.tok('code', 'while (' + captures[1] + ')');
  1302. }
  1303. },
  1304. /**
  1305. * Each.
  1306. */
  1307. each: function() {
  1308. var captures;
  1309. if (captures = /^(?:- *)?(?:each|for) +(\w+)(?: *, *(\w+))? * in *([^\n]+)/.exec(this.input)) {
  1310. this.consume(captures[0].length);
  1311. var tok = this.tok('each', captures[1]);
  1312. tok.key = captures[2] || '$index';
  1313. tok.code = captures[3];
  1314. return tok;
  1315. }
  1316. },
  1317. /**
  1318. * Code.
  1319. */
  1320. code: function() {
  1321. var captures;
  1322. if (captures = /^(!?=|-)([^\n]+)/.exec(this.input)) {
  1323. this.consume(captures[0].length);
  1324. var flags = captures[1];
  1325. captures[1] = captures[2];
  1326. var tok = this.tok('code', captures[1]);
  1327. tok.escape = flags.charAt(0) === '=';
  1328. tok.buffer = flags.charAt(0) === '=' || flags.charAt(1) === '=';
  1329. return tok;
  1330. }
  1331. },
  1332. /**
  1333. * Attributes.
  1334. */
  1335. attrs: function() {
  1336. if ('(' == this.input.charAt(0)) {
  1337. var index = this.indexOfDelimiters('(', ')')
  1338. , str = this.input.substr(1, index-1)
  1339. , tok = this.tok('attrs')
  1340. , len = str.length
  1341. , colons = this.colons
  1342. , states = ['key']
  1343. , escapedAttr
  1344. , key = ''
  1345. , val = ''
  1346. , quote
  1347. , c
  1348. , p;
  1349. function state(){
  1350. return states[states.length - 1];
  1351. }
  1352. function interpolate(attr) {
  1353. return attr.replace(/(\\)?#\{([^}]+)\}/g, function(_, escape, expr){
  1354. return escape
  1355. ? _
  1356. : quote + " + (" + expr + ") + " + quote;
  1357. });
  1358. }
  1359. this.consume(index + 1);
  1360. tok.attrs = {};
  1361. tok.escaped = {};
  1362. function parse(c) {
  1363. var real = c;
  1364. // TODO: remove when people fix ":"
  1365. if (colons && ':' == c) c = '=';
  1366. switch (c) {
  1367. case ',':
  1368. case '\n':
  1369. switch (state()) {
  1370. case 'expr':
  1371. case 'array':
  1372. case 'string':
  1373. case 'object':
  1374. val += c;
  1375. break;
  1376. default:
  1377. states.push('key');
  1378. val = val.trim();
  1379. key = key.trim();
  1380. if ('' == key) return;
  1381. key = key.replace(/^['"]|['"]$/g, '').replace('!', '');
  1382. tok.escaped[key] = escapedAttr;
  1383. tok.attrs[key] = '' == val
  1384. ? true
  1385. : interpolate(val);
  1386. key = val = '';
  1387. }
  1388. break;
  1389. case '=':
  1390. switch (state()) {
  1391. case 'key char':
  1392. key += real;
  1393. break;
  1394. case 'val':
  1395. case 'expr':
  1396. case 'array':
  1397. case 'string':
  1398. case 'object':
  1399. val += real;
  1400. break;
  1401. default:
  1402. escapedAttr = '!' != p;
  1403. states.push('val');
  1404. }
  1405. break;
  1406. case '(':
  1407. if ('val' == state()
  1408. || 'expr' == state()) states.push('expr');
  1409. val += c;
  1410. break;
  1411. case ')':
  1412. if ('expr' == state()
  1413. || 'val' == state()) states.pop();
  1414. val += c;
  1415. break;
  1416. case '{':
  1417. if ('val' == state()) states.push('object');
  1418. val += c;
  1419. break;
  1420. case '}':
  1421. if ('object' == state()) states.pop();
  1422. val += c;
  1423. break;
  1424. case '[':
  1425. if ('val' == state()) states.push('array');
  1426. val += c;
  1427. break;
  1428. case ']':
  1429. if ('array' == state()) states.pop();
  1430. val += c;
  1431. break;
  1432. case '"':
  1433. case "'":
  1434. switch (state()) {
  1435. case 'key':
  1436. states.push('key char');
  1437. break;
  1438. case 'key char':
  1439. states.pop();
  1440. break;
  1441. case 'string':
  1442. if (c == quote) states.pop();
  1443. val += c;
  1444. break;
  1445. default:
  1446. states.push('string');
  1447. val += c;
  1448. quote = c;
  1449. }
  1450. break;
  1451. case '':
  1452. break;
  1453. default:
  1454. switch (state()) {
  1455. case 'key':
  1456. case 'key char':
  1457. key += c;
  1458. break;
  1459. default:
  1460. val += c;
  1461. }
  1462. }
  1463. p = c;
  1464. }
  1465. for (var i = 0; i < len; ++i) {
  1466. parse(str.charAt(i));
  1467. }
  1468. parse(',');
  1469. if ('/' == this.input.charAt(0)) {
  1470. this.consume(1);
  1471. tok.selfClosing = true;
  1472. }
  1473. return tok;
  1474. }
  1475. },
  1476. /**
  1477. * Indent | Outdent | Newline.
  1478. */
  1479. indent: function() {
  1480. var captures, re;
  1481. // established regexp
  1482. if (this.indentRe) {
  1483. captures = this.indentRe.exec(this.input);
  1484. // determine regexp
  1485. } else {
  1486. // tabs
  1487. re = /^\n(\t*) */;
  1488. captures = re.exec(this.input);
  1489. // spaces
  1490. if (captures && !captures[1].length) {
  1491. re = /^\n( *)/;
  1492. captures = re.exec(this.input);
  1493. }
  1494. // established
  1495. if (captures && captures[1].length) this.indentRe = re;
  1496. }
  1497. if (captures) {
  1498. var tok
  1499. , indents = captures[1].length;
  1500. ++this.lineno;
  1501. this.consume(indents + 1);
  1502. if (' ' == this.input[0] || '\t' == this.input[0]) {
  1503. throw new Error('Invalid indentation, you can use tabs or spaces but not both');
  1504. }
  1505. // blank line
  1506. if ('\n' == this.input[0]) return this.tok('newline');
  1507. // outdent
  1508. if (this.indentStack.length && indents < this.indentStack[0]) {
  1509. while (this.indentStack.length && this.indentStack[0] > indents) {
  1510. this.stash.push(this.tok('outdent'));
  1511. this.indentStack.shift();
  1512. }
  1513. tok = this.stash.pop();
  1514. // indent
  1515. } else if (indents && indents != this.indentStack[0]) {
  1516. this.indentStack.unshift(indents);
  1517. tok = this.tok('indent', indents);
  1518. // newline
  1519. } else {
  1520. tok = this.tok('newline');
  1521. }
  1522. return tok;
  1523. }
  1524. },
  1525. /**
  1526. * Pipe-less text consumed only when
  1527. * pipeless is true;
  1528. */
  1529. pipelessText: function() {
  1530. if (this.pipeless) {
  1531. if ('\n' == this.input[0]) return;
  1532. var i = this.input.indexOf('\n');
  1533. if (-1 == i) i = this.input.length;
  1534. var str = this.input.substr(0, i);
  1535. this.consume(str.length);
  1536. return this.tok('text', str);
  1537. }
  1538. },
  1539. /**
  1540. * ':'
  1541. */
  1542. colon: function() {
  1543. return this.scan(/^: */, ':');
  1544. },
  1545. /**
  1546. * Return the next token object, or those
  1547. * previously stashed by lookahead.
  1548. *
  1549. * @return {Object}
  1550. * @api private
  1551. */
  1552. advance: function(){
  1553. return this.stashed()
  1554. || this.next();
  1555. },
  1556. /**
  1557. * Return the next token object.
  1558. *
  1559. * @return {Object}
  1560. * @api private
  1561. */
  1562. next: function() {
  1563. return this.deferred()
  1564. || this.blank()
  1565. || this.eos()
  1566. || this.pipelessText()
  1567. || this.yield()
  1568. || this.doctype()
  1569. || this.interpolation()
  1570. || this["case"]()
  1571. || this.when()
  1572. || this["default"]()
  1573. || this["extends"]()
  1574. || this.append()
  1575. || this.prepend()
  1576. || this.block()
  1577. || this.include()
  1578. || this.mixin()
  1579. || this.call()
  1580. || this.conditional()
  1581. || this.each()
  1582. || this["while"]()
  1583. || this.assignment()
  1584. || this.tag()
  1585. || this.filter()
  1586. || this.code()
  1587. || this.id()
  1588. || this.className()
  1589. || this.attrs()
  1590. || this.indent()
  1591. || this.comment()
  1592. || this.colon()
  1593. || this.text();
  1594. }
  1595. };
  1596. }); // module: lexer.js
  1597. require.register("nodes/attrs.js", function(module, exports, require){
  1598. /*!
  1599. * Jade - nodes - Attrs
  1600. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  1601. * MIT Licensed
  1602. */
  1603. /**
  1604. * Module dependencies.
  1605. */
  1606. var Node = require('./node'),
  1607. Block = require('./block');
  1608. /**
  1609. * Initialize a `Attrs` node.
  1610. *
  1611. * @api public
  1612. */
  1613. var Attrs = module.exports = function Attrs() {
  1614. this.attrs = [];
  1615. };
  1616. /**
  1617. * Inherit from `Node`.
  1618. */
  1619. Attrs.prototype = new Node;
  1620. Attrs.prototype.constructor = Attrs;
  1621. /**
  1622. * Set attribute `name` to `val`, keep in mind these become
  1623. * part of a raw js object literal, so to quote a value you must
  1624. * '"quote me"', otherwise or example 'user.name' is literal JavaScript.
  1625. *
  1626. * @param {String} name
  1627. * @param {String} val
  1628. * @param {Boolean} escaped
  1629. * @return {Tag} for chaining
  1630. * @api public
  1631. */
  1632. Attrs.prototype.setAttribute = function(name, val, escaped){
  1633. this.attrs.push({ name: name, val: val, escaped: escaped });
  1634. return this;
  1635. };
  1636. /**
  1637. * Remove attribute `name` when present.
  1638. *
  1639. * @param {String} name
  1640. * @api public
  1641. */
  1642. Attrs.prototype.removeAttribute = function(name){
  1643. for (var i = 0, len = this.attrs.length; i < len; ++i) {
  1644. if (this.attrs[i] && this.attrs[i].name == name) {
  1645. delete this.attrs[i];
  1646. }
  1647. }
  1648. };
  1649. /**
  1650. * Get attribute value by `name`.
  1651. *
  1652. * @param {String} name
  1653. * @return {String}
  1654. * @api public
  1655. */
  1656. Attrs.prototype.getAttribute = function(name){
  1657. for (var i = 0, len = this.attrs.length; i < len; ++i) {
  1658. if (this.attrs[i] && this.attrs[i].name == name) {
  1659. return this.attrs[i].val;
  1660. }
  1661. }
  1662. };
  1663. }); // module: nodes/attrs.js
  1664. require.register("nodes/block-comment.js", function(module, exports, require){
  1665. /*!
  1666. * Jade - nodes - BlockComment
  1667. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  1668. * MIT Licensed
  1669. */
  1670. /**
  1671. * Module dependencies.
  1672. */
  1673. var Node = require('./node');
  1674. /**
  1675. * Initialize a `BlockComment` with the given `block`.
  1676. *
  1677. * @param {String} val
  1678. * @param {Block} block
  1679. * @param {Boolean} buffer
  1680. * @api public
  1681. */
  1682. var BlockComment = module.exports = function BlockComment(val, block, buffer) {
  1683. this.block = block;
  1684. this.val = val;
  1685. this.buffer = buffer;
  1686. };
  1687. /**
  1688. * Inherit from `Node`.
  1689. */
  1690. BlockComment.prototype = new Node;
  1691. BlockComment.prototype.constructor = BlockComment;
  1692. }); // module: nodes/block-comment.js
  1693. require.register("nodes/block.js", function(module, exports, require){
  1694. /*!
  1695. * Jade - nodes - Block
  1696. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  1697. * MIT Licensed
  1698. */
  1699. /**
  1700. * Module dependencies.
  1701. */
  1702. var Node = require('./node');
  1703. /**
  1704. * Initialize a new `Block` with an optional `node`.
  1705. *
  1706. * @param {Node} node
  1707. * @api public
  1708. */
  1709. var Block = module.exports = function Block(node){
  1710. this.nodes = [];
  1711. if (node) this.push(node);
  1712. };
  1713. /**
  1714. * Inherit from `Node`.
  1715. */
  1716. Block.prototype = new Node;
  1717. Block.prototype.constructor = Block;
  1718. /**
  1719. * Block flag.
  1720. */
  1721. Block.prototype.isBlock = true;
  1722. /**
  1723. * Replace the nodes in `other` with the nodes
  1724. * in `this` block.
  1725. *
  1726. * @param {Block} other
  1727. * @api private
  1728. */
  1729. Block.prototype.replace = function(other){
  1730. other.nodes = this.nodes;
  1731. };
  1732. /**
  1733. * Pust the given `node`.
  1734. *
  1735. * @param {Node} node
  1736. * @return {Number}
  1737. * @api public
  1738. */
  1739. Block.prototype.push = function(node){
  1740. return this.nodes.push(node);
  1741. };
  1742. /**
  1743. * Check if this block is empty.
  1744. *
  1745. * @return {Boolean}
  1746. * @api public
  1747. */
  1748. Block.prototype.isEmpty = function(){
  1749. return 0 == this.nodes.length;
  1750. };
  1751. /**
  1752. * Unshift the given `node`.
  1753. *
  1754. * @param {Node} node
  1755. * @return {Number}
  1756. * @api public
  1757. */
  1758. Block.prototype.unshift = function(node){
  1759. return this.nodes.unshift(node);
  1760. };
  1761. /**
  1762. * Return the "last" block, or the first `yield` node.
  1763. *
  1764. * @return {Block}
  1765. * @api private
  1766. */
  1767. Block.prototype.includeBlock = function(){
  1768. var ret = this
  1769. , node;
  1770. for (var i = 0, len = this.nodes.length; i < len; ++i) {
  1771. node = this.nodes[i];
  1772. if (node.yield) return node;
  1773. else if (node.textOnly) continue;
  1774. else if (node.includeBlock) ret = node.includeBlock();
  1775. else if (node.block && !node.block.isEmpty()) ret = node.block.includeBlock();
  1776. if (ret.yield) return ret;
  1777. }
  1778. return ret;
  1779. };
  1780. /**
  1781. * Return a clone of this block.
  1782. *
  1783. * @return {Block}
  1784. * @api private
  1785. */
  1786. Block.prototype.clone = function(){
  1787. var clone = new Block;
  1788. for (var i = 0, len = this.nodes.length; i < len; ++i) {
  1789. clone.push(this.nodes[i].clone());
  1790. }
  1791. return clone;
  1792. };
  1793. }); // module: nodes/block.js
  1794. require.register("nodes/case.js", function(module, exports, require){
  1795. /*!
  1796. * Jade - nodes - Case
  1797. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  1798. * MIT Licensed
  1799. */
  1800. /**
  1801. * Module dependencies.
  1802. */
  1803. var Node = require('./node');
  1804. /**
  1805. * Initialize a new `Case` with `expr`.
  1806. *
  1807. * @param {String} expr
  1808. * @api public
  1809. */
  1810. var Case = exports = module.exports = function Case(expr, block){
  1811. this.expr = expr;
  1812. this.block = block;
  1813. };
  1814. /**
  1815. * Inherit from `Node`.
  1816. */
  1817. Case.prototype = new Node;
  1818. Case.prototype.constructor = Case;
  1819. var When = exports.When = function When(expr, block){
  1820. this.expr = expr;
  1821. this.block = block;
  1822. this.debug = false;
  1823. };
  1824. /**
  1825. * Inherit from `Node`.
  1826. */
  1827. When.prototype = new Node;
  1828. When.prototype.constructor = When;
  1829. }); // module: nodes/case.js
  1830. require.register("nodes/code.js", function(module, exports, require){
  1831. /*!
  1832. * Jade - nodes - Code
  1833. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  1834. * MIT Licensed
  1835. */
  1836. /**
  1837. * Module dependencies.
  1838. */
  1839. var Node = require('./node');
  1840. /**
  1841. * Initialize a `Code` node with the given code `val`.
  1842. * Code may also be optionally buffered and escaped.
  1843. *
  1844. * @param {String} val
  1845. * @param {Boolean} buffer
  1846. * @param {Boolean} escape
  1847. * @api public
  1848. */
  1849. var Code = module.exports = function Code(val, buffer, escape) {
  1850. this.val = val;
  1851. this.buffer = buffer;
  1852. this.escape = escape;
  1853. if (val.match(/^ *else/)) this.debug = false;
  1854. };
  1855. /**
  1856. * Inherit from `Node`.
  1857. */
  1858. Code.prototype = new Node;
  1859. Code.prototype.constructor = Code;
  1860. }); // module: nodes/code.js
  1861. require.register("nodes/comment.js", function(module, exports, require){
  1862. /*!
  1863. * Jade - nodes - Comment
  1864. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  1865. * MIT Licensed
  1866. */
  1867. /**
  1868. * Module dependencies.
  1869. */
  1870. var Node = require('./node');
  1871. /**
  1872. * Initialize a `Comment` with the given `val`, optionally `buffer`,
  1873. * otherwise the comment may render in the output.
  1874. *
  1875. * @param {String} val
  1876. * @param {Boolean} buffer
  1877. * @api public
  1878. */
  1879. var Comment = module.exports = function Comment(val, buffer) {
  1880. this.val = val;
  1881. this.buffer = buffer;
  1882. };
  1883. /**
  1884. * Inherit from `Node`.
  1885. */
  1886. Comment.prototype = new Node;
  1887. Comment.prototype.constructor = Comment;
  1888. }); // module: nodes/comment.js
  1889. require.register("nodes/doctype.js", function(module, exports, require){
  1890. /*!
  1891. * Jade - nodes - Doctype
  1892. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  1893. * MIT Licensed
  1894. */
  1895. /**
  1896. * Module dependencies.
  1897. */
  1898. var Node = require('./node');
  1899. /**
  1900. * Initialize a `Doctype` with the given `val`.
  1901. *
  1902. * @param {String} val
  1903. * @api public
  1904. */
  1905. var Doctype = module.exports = function Doctype(val) {
  1906. this.val = val;
  1907. };
  1908. /**
  1909. * Inherit from `Node`.
  1910. */
  1911. Doctype.prototype = new Node;
  1912. Doctype.prototype.constructor = Doctype;
  1913. }); // module: nodes/doctype.js
  1914. require.register("nodes/each.js", function(module, exports, require){
  1915. /*!
  1916. * Jade - nodes - Each
  1917. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  1918. * MIT Licensed
  1919. */
  1920. /**
  1921. * Module dependencies.
  1922. */
  1923. var Node = require('./node');
  1924. /**
  1925. * Initialize an `Each` node, representing iteration
  1926. *
  1927. * @param {String} obj
  1928. * @param {String} val
  1929. * @param {String} key
  1930. * @param {Block} block
  1931. * @api public
  1932. */
  1933. var Each = module.exports = function Each(obj, val, key, block) {
  1934. this.obj = obj;
  1935. this.val = val;
  1936. this.key = key;
  1937. this.block = block;
  1938. };
  1939. /**
  1940. * Inherit from `Node`.
  1941. */
  1942. Each.prototype = new Node;
  1943. Each.prototype.constructor = Each;
  1944. }); // module: nodes/each.js
  1945. require.register("nodes/filter.js", function(module, exports, require){
  1946. /*!
  1947. * Jade - nodes - Filter
  1948. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  1949. * MIT Licensed
  1950. */
  1951. /**
  1952. * Module dependencies.
  1953. */
  1954. var Node = require('./node')
  1955. , Block = require('./block');
  1956. /**
  1957. * Initialize a `Filter` node with the given
  1958. * filter `name` and `block`.
  1959. *
  1960. * @param {String} name
  1961. * @param {Block|Node} block
  1962. * @api public
  1963. */
  1964. var Filter = module.exports = function Filter(name, block, attrs) {
  1965. this.name = name;
  1966. this.block = block;
  1967. this.attrs = attrs;
  1968. this.isASTFilter = !block.nodes.every(function(node){ return node.isText });
  1969. };
  1970. /**
  1971. * Inherit from `Node`.
  1972. */
  1973. Filter.prototype = new Node;
  1974. Filter.prototype.constructor = Filter;
  1975. }); // module: nodes/filter.js
  1976. require.register("nodes/index.js", function(module, exports, require){
  1977. /*!
  1978. * Jade - nodes
  1979. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  1980. * MIT Licensed
  1981. */
  1982. exports.Node = require('./node');
  1983. exports.Tag = require('./tag');
  1984. exports.Code = require('./code');
  1985. exports.Each = require('./each');
  1986. exports.Case = require('./case');
  1987. exports.Text = require('./text');
  1988. exports.Block = require('./block');
  1989. exports.Mixin = require('./mixin');
  1990. exports.Filter = require('./filter');
  1991. exports.Comment = require('./comment');
  1992. exports.Literal = require('./literal');
  1993. exports.BlockComment = require('./block-comment');
  1994. exports.Doctype = require('./doctype');
  1995. }); // module: nodes/index.js
  1996. require.register("nodes/literal.js", function(module, exports, require){
  1997. /*!
  1998. * Jade - nodes - Literal
  1999. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  2000. * MIT Licensed
  2001. */
  2002. /**
  2003. * Module dependencies.
  2004. */
  2005. var Node = require('./node');
  2006. /**
  2007. * Initialize a `Literal` node with the given `str.
  2008. *
  2009. * @param {String} str
  2010. * @api public
  2011. */
  2012. var Literal = module.exports = function Literal(str) {
  2013. this.str = str
  2014. .replace(/\\/g, "\\\\")
  2015. .replace(/\n|\r\n/g, "\\n")
  2016. .replace(/'/g, "\\'");
  2017. };
  2018. /**
  2019. * Inherit from `Node`.
  2020. */
  2021. Literal.prototype = new Node;
  2022. Literal.prototype.constructor = Literal;
  2023. }); // module: nodes/literal.js
  2024. require.register("nodes/mixin.js", function(module, exports, require){
  2025. /*!
  2026. * Jade - nodes - Mixin
  2027. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  2028. * MIT Licensed
  2029. */
  2030. /**
  2031. * Module dependencies.
  2032. */
  2033. var Attrs = require('./attrs');
  2034. /**
  2035. * Initialize a new `Mixin` with `name` and `block`.
  2036. *
  2037. * @param {String} name
  2038. * @param {String} args
  2039. * @param {Block} block
  2040. * @api public
  2041. */
  2042. var Mixin = module.exports = function Mixin(name, args, block, call){
  2043. this.name = name;
  2044. this.args = args;
  2045. this.block = block;
  2046. this.attrs = [];
  2047. this.call = call;
  2048. };
  2049. /**
  2050. * Inherit from `Attrs`.
  2051. */
  2052. Mixin.prototype = new Attrs;
  2053. Mixin.prototype.constructor = Mixin;
  2054. }); // module: nodes/mixin.js
  2055. require.register("nodes/node.js", function(module, exports, require){
  2056. /*!
  2057. * Jade - nodes - Node
  2058. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  2059. * MIT Licensed
  2060. */
  2061. /**
  2062. * Initialize a `Node`.
  2063. *
  2064. * @api public
  2065. */
  2066. var Node = module.exports = function Node(){};
  2067. /**
  2068. * Clone this node (return itself)
  2069. *
  2070. * @return {Node}
  2071. * @api private
  2072. */
  2073. Node.prototype.clone = function(){
  2074. return this;
  2075. };
  2076. }); // module: nodes/node.js
  2077. require.register("nodes/tag.js", function(module, exports, require){
  2078. /*!
  2079. * Jade - nodes - Tag
  2080. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  2081. * MIT Licensed
  2082. */
  2083. /**
  2084. * Module dependencies.
  2085. */
  2086. var Attrs = require('./attrs'),
  2087. Block = require('./block'),
  2088. inlineTags = require('../inline-tags');
  2089. /**
  2090. * Initialize a `Tag` node with the given tag `name` and optional `block`.
  2091. *
  2092. * @param {String} name
  2093. * @param {Block} block
  2094. * @api public
  2095. */
  2096. var Tag = module.exports = function Tag(name, block) {
  2097. this.name = name;
  2098. this.attrs = [];
  2099. this.block = block || new Block;
  2100. };
  2101. /**
  2102. * Inherit from `Attrs`.
  2103. */
  2104. Tag.prototype = new Attrs;
  2105. Tag.prototype.constructor = Tag;
  2106. /**
  2107. * Clone this tag.
  2108. *
  2109. * @return {Tag}
  2110. * @api private
  2111. */
  2112. Tag.prototype.clone = function(){
  2113. var clone = new Tag(this.name, this.block.clone());
  2114. clone.line = this.line;
  2115. clone.attrs = this.attrs;
  2116. clone.textOnly = this.textOnly;
  2117. return clone;
  2118. };
  2119. /**
  2120. * Check if this tag is an inline tag.
  2121. *
  2122. * @return {Boolean}
  2123. * @api private
  2124. */
  2125. Tag.prototype.isInline = function(){
  2126. return ~inlineTags.indexOf(this.name);
  2127. };
  2128. /**
  2129. * Check if this tag's contents can be inlined. Used for pretty printing.
  2130. *
  2131. * @return {Boolean}
  2132. * @api private
  2133. */
  2134. Tag.prototype.canInline = function(){
  2135. var nodes = this.block.nodes;
  2136. function isInline(node){
  2137. // Recurse if the node is a block
  2138. if (node.isBlock) return node.nodes.every(isInline);
  2139. return node.isText || (node.isInline && node.isInline());
  2140. }
  2141. // Empty tag
  2142. if (!nodes.length) return true;
  2143. // Text-only or inline-only tag
  2144. if (1 == nodes.length) return isInline(nodes[0]);
  2145. // Multi-line inline-only tag
  2146. if (this.block.nodes.every(isInline)) {
  2147. for (var i = 1, len = nodes.length; i < len; ++i) {
  2148. if (nodes[i-1].isText && nodes[i].isText)
  2149. return false;
  2150. }
  2151. return true;
  2152. }
  2153. // Mixed tag
  2154. return false;
  2155. };
  2156. }); // module: nodes/tag.js
  2157. require.register("nodes/text.js", function(module, exports, require){
  2158. /*!
  2159. * Jade - nodes - Text
  2160. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  2161. * MIT Licensed
  2162. */
  2163. /**
  2164. * Module dependencies.
  2165. */
  2166. var Node = require('./node');
  2167. /**
  2168. * Initialize a `Text` node with optional `line`.
  2169. *
  2170. * @param {String} line
  2171. * @api public
  2172. */
  2173. var Text = module.exports = function Text(line) {
  2174. this.val = '';
  2175. if ('string' == typeof line) this.val = line;
  2176. };
  2177. /**
  2178. * Inherit from `Node`.
  2179. */
  2180. Text.prototype = new Node;
  2181. Text.prototype.constructor = Text;
  2182. /**
  2183. * Flag as text.
  2184. */
  2185. Text.prototype.isText = true;
  2186. }); // module: nodes/text.js
  2187. require.register("parser.js", function(module, exports, require){
  2188. /*!
  2189. * Jade - Parser
  2190. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  2191. * MIT Licensed
  2192. */
  2193. /**
  2194. * Module dependencies.
  2195. */
  2196. var Lexer = require('./lexer')
  2197. , nodes = require('./nodes')
  2198. , utils = require('./utils');
  2199. /**
  2200. * Initialize `Parser` with the given input `str` and `filename`.
  2201. *
  2202. * @param {String} str
  2203. * @param {String} filename
  2204. * @param {Object} options
  2205. * @api public
  2206. */
  2207. var Parser = exports = module.exports = function Parser(str, filename, options){
  2208. this.input = str;
  2209. this.lexer = new Lexer(str, options);
  2210. this.filename = filename;
  2211. this.blocks = {};
  2212. this.mixins = {};
  2213. this.options = options;
  2214. this.contexts = [this];
  2215. };
  2216. /**
  2217. * Tags that may not contain tags.
  2218. */
  2219. var textOnly = exports.textOnly = ['script', 'style'];
  2220. /**
  2221. * Parser prototype.
  2222. */
  2223. Parser.prototype = {
  2224. /**
  2225. * Push `parser` onto the context stack,
  2226. * or pop and return a `Parser`.
  2227. */
  2228. context: function(parser){
  2229. if (parser) {
  2230. this.contexts.push(parser);
  2231. } else {
  2232. return this.contexts.pop();
  2233. }
  2234. },
  2235. /**
  2236. * Return the next token object.
  2237. *
  2238. * @return {Object}
  2239. * @api private
  2240. */
  2241. advance: function(){
  2242. return this.lexer.advance();
  2243. },
  2244. /**
  2245. * Skip `n` tokens.
  2246. *
  2247. * @param {Number} n
  2248. * @api private
  2249. */
  2250. skip: function(n){
  2251. while (n--) this.advance();
  2252. },
  2253. /**
  2254. * Single token lookahead.
  2255. *
  2256. * @return {Object}
  2257. * @api private
  2258. */
  2259. peek: function() {
  2260. return this.lookahead(1);
  2261. },
  2262. /**
  2263. * Return lexer lineno.
  2264. *
  2265. * @return {Number}
  2266. * @api private
  2267. */
  2268. line: function() {
  2269. return this.lexer.lineno;
  2270. },
  2271. /**
  2272. * `n` token lookahead.
  2273. *
  2274. * @param {Number} n
  2275. * @return {Object}
  2276. * @api private
  2277. */
  2278. lookahead: function(n){
  2279. return this.lexer.lookahead(n);
  2280. },
  2281. /**
  2282. * Parse input returning a string of js for evaluation.
  2283. *
  2284. * @return {String}
  2285. * @api public
  2286. */
  2287. parse: function(){
  2288. var block = new nodes.Block, parser;
  2289. block.line = this.line();
  2290. while ('eos' != this.peek().type) {
  2291. if ('newline' == this.peek().type) {
  2292. this.advance();
  2293. } else {
  2294. block.push(this.parseExpr());
  2295. }
  2296. }
  2297. if (parser = this.extending) {
  2298. this.context(parser);
  2299. var ast = parser.parse();
  2300. this.context();
  2301. // hoist mixins
  2302. for (var name in this.mixins)
  2303. ast.unshift(this.mixins[name]);
  2304. return ast;
  2305. }
  2306. return block;
  2307. },
  2308. /**
  2309. * Expect the given type, or throw an exception.
  2310. *
  2311. * @param {String} type
  2312. * @api private
  2313. */
  2314. expect: function(type){
  2315. if (this.peek().type === type) {
  2316. return this.advance();
  2317. } else {
  2318. throw new Error('expected "' + type + '", but got "' + this.peek().type + '"');
  2319. }
  2320. },
  2321. /**
  2322. * Accept the given `type`.
  2323. *
  2324. * @param {String} type
  2325. * @api private
  2326. */
  2327. accept: function(type){
  2328. if (this.peek().type === type) {
  2329. return this.advance();
  2330. }
  2331. },
  2332. /**
  2333. * tag
  2334. * | doctype
  2335. * | mixin
  2336. * | include
  2337. * | filter
  2338. * | comment
  2339. * | text
  2340. * | each
  2341. * | code
  2342. * | yield
  2343. * | id
  2344. * | class
  2345. * | interpolation
  2346. */
  2347. parseExpr: function(){
  2348. switch (this.peek().type) {
  2349. case 'tag':
  2350. return this.parseTag();
  2351. case 'mixin':
  2352. return this.parseMixin();
  2353. case 'block':
  2354. return this.parseBlock();
  2355. case 'case':
  2356. return this.parseCase();
  2357. case 'when':
  2358. return this.parseWhen();
  2359. case 'default':
  2360. return this.parseDefault();
  2361. case 'extends':
  2362. return this.parseExtends();
  2363. case 'include':
  2364. return this.parseInclude();
  2365. case 'doctype':
  2366. return this.parseDoctype();
  2367. case 'filter':
  2368. return this.parseFilter();
  2369. case 'comment':
  2370. return this.parseComment();
  2371. case 'text':
  2372. return this.parseText();
  2373. case 'each':
  2374. return this.parseEach();
  2375. case 'code':
  2376. return this.parseCode();
  2377. case 'call':
  2378. return this.parseCall();
  2379. case 'interpolation':
  2380. return this.parseInterpolation();
  2381. case 'yield':
  2382. this.advance();
  2383. var block = new nodes.Block;
  2384. block.yield = true;
  2385. return block;
  2386. case 'id':
  2387. case 'class':
  2388. var tok = this.advance();
  2389. this.lexer.defer(this.lexer.tok('tag', 'div'));
  2390. this.lexer.defer(tok);
  2391. return this.parseExpr();
  2392. default:
  2393. throw new Error('unexpected token "' + this.peek().type + '"');
  2394. }
  2395. },
  2396. /**
  2397. * Text
  2398. */
  2399. parseText: function(){
  2400. var tok = this.expect('text')
  2401. , node = new nodes.Text(tok.val);
  2402. node.line = this.line();
  2403. return node;
  2404. },
  2405. /**
  2406. * ':' expr
  2407. * | block
  2408. */
  2409. parseBlockExpansion: function(){
  2410. if (':' == this.peek().type) {
  2411. this.advance();
  2412. return new nodes.Block(this.parseExpr());
  2413. } else {
  2414. return this.block();
  2415. }
  2416. },
  2417. /**
  2418. * case
  2419. */
  2420. parseCase: function(){
  2421. var val = this.expect('case').val
  2422. , node = new nodes.Case(val);
  2423. node.line = this.line();
  2424. node.block = this.block();
  2425. return node;
  2426. },
  2427. /**
  2428. * when
  2429. */
  2430. parseWhen: function(){
  2431. var val = this.expect('when').val
  2432. return new nodes.Case.When(val, this.parseBlockExpansion());
  2433. },
  2434. /**
  2435. * default
  2436. */
  2437. parseDefault: function(){
  2438. this.expect('default');
  2439. return new nodes.Case.When('default', this.parseBlockExpansion());
  2440. },
  2441. /**
  2442. * code
  2443. */
  2444. parseCode: function(){
  2445. var tok = this.expect('code')
  2446. , node = new nodes.Code(tok.val, tok.buffer, tok.escape)
  2447. , block
  2448. , i = 1;
  2449. node.line = this.line();
  2450. while (this.lookahead(i) && 'newline' == this.lookahead(i).type) ++i;
  2451. block = 'indent' == this.lookahead(i).type;
  2452. if (block) {
  2453. this.skip(i-1);
  2454. node.block = this.block();
  2455. }
  2456. return node;
  2457. },
  2458. /**
  2459. * comment
  2460. */
  2461. parseComment: function(){
  2462. var tok = this.expect('comment')
  2463. , node;
  2464. if ('indent' == this.peek().type) {
  2465. node = new nodes.BlockComment(tok.val, this.block(), tok.buffer);
  2466. } else {
  2467. node = new nodes.Comment(tok.val, tok.buffer);
  2468. }
  2469. node.line = this.line();
  2470. return node;
  2471. },
  2472. /**
  2473. * doctype
  2474. */
  2475. parseDoctype: function(){
  2476. var tok = this.expect('doctype')
  2477. , node = new nodes.Doctype(tok.val);
  2478. node.line = this.line();
  2479. return node;
  2480. },
  2481. /**
  2482. * filter attrs? text-block
  2483. */
  2484. parseFilter: function(){
  2485. var block
  2486. , tok = this.expect('filter')
  2487. , attrs = this.accept('attrs');
  2488. this.lexer.pipeless = true;
  2489. block = this.parseTextBlock();
  2490. this.lexer.pipeless = false;
  2491. var node = new nodes.Filter(tok.val, block, attrs && attrs.attrs);
  2492. node.line = this.line();
  2493. return node;
  2494. },
  2495. /**
  2496. * tag ':' attrs? block
  2497. */
  2498. parseASTFilter: function(){
  2499. var block
  2500. , tok = this.expect('tag')
  2501. , attrs = this.accept('attrs');
  2502. this.expect(':');
  2503. block = this.block();
  2504. var node = new nodes.Filter(tok.val, block, attrs && attrs.attrs);
  2505. node.line = this.line();
  2506. return node;
  2507. },
  2508. /**
  2509. * each block
  2510. */
  2511. parseEach: function(){
  2512. var tok = this.expect('each')
  2513. , node = new nodes.Each(tok.code, tok.val, tok.key);
  2514. node.line = this.line();
  2515. node.block = this.block();
  2516. if (this.peek().type == 'code' && this.peek().val == 'else') {
  2517. this.advance();
  2518. node.alternative = this.block();
  2519. }
  2520. return node;
  2521. },
  2522. /**
  2523. * 'extends' name
  2524. */
  2525. parseExtends: function(){
  2526. var path = require('path')
  2527. , fs = require('fs')
  2528. , dirname = path.dirname
  2529. , basename = path.basename
  2530. , join = path.join;
  2531. if (!this.filename)
  2532. throw new Error('the "filename" option is required to extend templates');
  2533. var path = this.expect('extends').val.trim()
  2534. , dir = dirname(this.filename);
  2535. var path = join(dir, path + '.jade')
  2536. , str = fs.readFileSync(path, 'utf8')
  2537. , parser = new Parser(str, path, this.options);
  2538. parser.blocks = this.blocks;
  2539. parser.contexts = this.contexts;
  2540. this.extending = parser;
  2541. // TODO: null node
  2542. return new nodes.Literal('');
  2543. },
  2544. /**
  2545. * 'block' name block
  2546. */
  2547. parseBlock: function(){
  2548. var block = this.expect('block')
  2549. , mode = block.mode
  2550. , name = block.val.trim();
  2551. block = 'indent' == this.peek().type
  2552. ? this.block()
  2553. : new nodes.Block(new nodes.Literal(''));
  2554. var prev = this.blocks[name];
  2555. if (prev) {
  2556. switch (prev.mode) {
  2557. case 'append':
  2558. block.nodes = block.nodes.concat(prev.nodes);
  2559. prev = block;
  2560. break;
  2561. case 'prepend':
  2562. block.nodes = prev.nodes.concat(block.nodes);
  2563. prev = block;
  2564. break;
  2565. }
  2566. }
  2567. block.mode = mode;
  2568. return this.blocks[name] = prev || block;
  2569. },
  2570. /**
  2571. * include block?
  2572. */
  2573. parseInclude: function(){
  2574. var path = require('path')
  2575. , fs = require('fs')
  2576. , dirname = path.dirname
  2577. , basename = path.basename
  2578. , join = path.join;
  2579. var path = this.expect('include').val.trim()
  2580. , dir = dirname(this.filename);
  2581. if (!this.filename)
  2582. throw new Error('the "filename" option is required to use includes');
  2583. // no extension
  2584. if (!~basename(path).indexOf('.')) {
  2585. path += '.jade';
  2586. }
  2587. // non-jade
  2588. if ('.jade' != path.substr(-5)) {
  2589. var path = join(dir, path)
  2590. , str = fs.readFileSync(path, 'utf8');
  2591. return new nodes.Literal(str);
  2592. }
  2593. var path = join(dir, path)
  2594. , str = fs.readFileSync(path, 'utf8')
  2595. , parser = new Parser(str, path, this.options);
  2596. parser.blocks = utils.merge({}, this.blocks);
  2597. parser.mixins = this.mixins;
  2598. this.context(parser);
  2599. var ast = parser.parse();
  2600. this.context();
  2601. ast.filename = path;
  2602. if ('indent' == this.peek().type) {
  2603. ast.includeBlock().push(this.block());
  2604. }
  2605. return ast;
  2606. },
  2607. /**
  2608. * call ident block
  2609. */
  2610. parseCall: function(){
  2611. var tok = this.expect('call')
  2612. , name = tok.val
  2613. , args = tok.args
  2614. , mixin = new nodes.Mixin(name, args, new nodes.Block, true);
  2615. this.tag(mixin);
  2616. if (mixin.block.isEmpty()) mixin.block = null;
  2617. return mixin;
  2618. },
  2619. /**
  2620. * mixin block
  2621. */
  2622. parseMixin: function(){
  2623. var tok = this.expect('mixin')
  2624. , name = tok.val
  2625. , args = tok.args
  2626. , mixin;
  2627. // definition
  2628. if ('indent' == this.peek().type) {
  2629. mixin = new nodes.Mixin(name, args, this.block(), false);
  2630. this.mixins[name] = mixin;
  2631. return mixin;
  2632. // call
  2633. } else {
  2634. return new nodes.Mixin(name, args, null, true);
  2635. }
  2636. },
  2637. /**
  2638. * indent (text | newline)* outdent
  2639. */
  2640. parseTextBlock: function(){
  2641. var block = new nodes.Block;
  2642. block.line = this.line();
  2643. var spaces = this.expect('indent').val;
  2644. if (null == this._spaces) this._spaces = spaces;
  2645. var indent = Array(spaces - this._spaces + 1).join(' ');
  2646. while ('outdent' != this.peek().type) {
  2647. switch (this.peek().type) {
  2648. case 'newline':
  2649. this.advance();
  2650. break;
  2651. case 'indent':
  2652. this.parseTextBlock().nodes.forEach(function(node){
  2653. block.push(node);
  2654. });
  2655. break;
  2656. default:
  2657. var text = new nodes.Text(indent + this.advance().val);
  2658. text.line = this.line();
  2659. block.push(text);
  2660. }
  2661. }
  2662. if (spaces == this._spaces) this._spaces = null;
  2663. this.expect('outdent');
  2664. return block;
  2665. },
  2666. /**
  2667. * indent expr* outdent
  2668. */
  2669. block: function(){
  2670. var block = new nodes.Block;
  2671. block.line = this.line();
  2672. this.expect('indent');
  2673. while ('outdent' != this.peek().type) {
  2674. if ('newline' == this.peek().type) {
  2675. this.advance();
  2676. } else {
  2677. block.push(this.parseExpr());
  2678. }
  2679. }
  2680. this.expect('outdent');
  2681. return block;
  2682. },
  2683. /**
  2684. * interpolation (attrs | class | id)* (text | code | ':')? newline* block?
  2685. */
  2686. parseInterpolation: function(){
  2687. var tok = this.advance();
  2688. var tag = new nodes.Tag(tok.val);
  2689. tag.buffer = true;
  2690. return this.tag(tag);
  2691. },
  2692. /**
  2693. * tag (attrs | class | id)* (text | code | ':')? newline* block?
  2694. */
  2695. parseTag: function(){
  2696. // ast-filter look-ahead
  2697. var i = 2;
  2698. if ('attrs' == this.lookahead(i).type) ++i;
  2699. if (':' == this.lookahead(i).type) {
  2700. if ('indent' == this.lookahead(++i).type) {
  2701. return this.parseASTFilter();
  2702. }
  2703. }
  2704. var tok = this.advance()
  2705. , tag = new nodes.Tag(tok.val);
  2706. tag.selfClosing = tok.selfClosing;
  2707. return this.tag(tag);
  2708. },
  2709. /**
  2710. * Parse tag.
  2711. */
  2712. tag: function(tag){
  2713. var dot;
  2714. tag.line = this.line();
  2715. // (attrs | class | id)*
  2716. out:
  2717. while (true) {
  2718. switch (this.peek().type) {
  2719. case 'id':
  2720. case 'class':
  2721. var tok = this.advance();
  2722. tag.setAttribute(tok.type, "'" + tok.val + "'");
  2723. continue;
  2724. case 'attrs':
  2725. var tok = this.advance()
  2726. , obj = tok.attrs
  2727. , escaped = tok.escaped
  2728. , names = Object.keys(obj);
  2729. if (tok.selfClosing) tag.selfClosing = true;
  2730. for (var i = 0, len = names.length; i < len; ++i) {
  2731. var name = names[i]
  2732. , val = obj[name];
  2733. tag.setAttribute(name, val, escaped[name]);
  2734. }
  2735. continue;
  2736. default:
  2737. break out;
  2738. }
  2739. }
  2740. // check immediate '.'
  2741. if ('.' == this.peek().val) {
  2742. dot = tag.textOnly = true;
  2743. this.advance();
  2744. }
  2745. // (text | code | ':')?
  2746. switch (this.peek().type) {
  2747. case 'text':
  2748. tag.block.push(this.parseText());
  2749. break;
  2750. case 'code':
  2751. tag.code = this.parseCode();
  2752. break;
  2753. case ':':
  2754. this.advance();
  2755. tag.block = new nodes.Block;
  2756. tag.block.push(this.parseExpr());
  2757. break;
  2758. }
  2759. // newline*
  2760. while ('newline' == this.peek().type) this.advance();
  2761. tag.textOnly = tag.textOnly || ~textOnly.indexOf(tag.name);
  2762. // script special-case
  2763. if ('script' == tag.name) {
  2764. var type = tag.getAttribute('type');
  2765. if (!dot && type && 'text/javascript' != type.replace(/^['"]|['"]$/g, '')) {
  2766. tag.textOnly = false;
  2767. }
  2768. }
  2769. // block?
  2770. if ('indent' == this.peek().type) {
  2771. if (tag.textOnly) {
  2772. this.lexer.pipeless = true;
  2773. tag.block = this.parseTextBlock();
  2774. this.lexer.pipeless = false;
  2775. } else {
  2776. var block = this.block();
  2777. if (tag.block) {
  2778. for (var i = 0, len = block.nodes.length; i < len; ++i) {
  2779. tag.block.push(block.nodes[i]);
  2780. }
  2781. } else {
  2782. tag.block = block;
  2783. }
  2784. }
  2785. }
  2786. return tag;
  2787. }
  2788. };
  2789. }); // module: parser.js
  2790. require.register("runtime.js", function(module, exports, require){
  2791. /*!
  2792. * Jade - runtime
  2793. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  2794. * MIT Licensed
  2795. */
  2796. /**
  2797. * Lame Array.isArray() polyfill for now.
  2798. */
  2799. if (!Array.isArray) {
  2800. Array.isArray = function(arr){
  2801. return '[object Array]' == Object.prototype.toString.call(arr);
  2802. };
  2803. }
  2804. /**
  2805. * Lame Object.keys() polyfill for now.
  2806. */
  2807. if (!Object.keys) {
  2808. Object.keys = function(obj){
  2809. var arr = [];
  2810. for (var key in obj) {
  2811. if (obj.hasOwnProperty(key)) {
  2812. arr.push(key);
  2813. }
  2814. }
  2815. return arr;
  2816. }
  2817. }
  2818. /**
  2819. * Merge two attribute objects giving precedence
  2820. * to values in object `b`. Classes are special-cased
  2821. * allowing for arrays and merging/joining appropriately
  2822. * resulting in a string.
  2823. *
  2824. * @param {Object} a
  2825. * @param {Object} b
  2826. * @return {Object} a
  2827. * @api private
  2828. */
  2829. exports.merge = function merge(a, b) {
  2830. var ac = a['class'];
  2831. var bc = b['class'];
  2832. if (ac || bc) {
  2833. ac = ac || [];
  2834. bc = bc || [];
  2835. if (!Array.isArray(ac)) ac = [ac];
  2836. if (!Array.isArray(bc)) bc = [bc];
  2837. ac = ac.filter(nulls);
  2838. bc = bc.filter(nulls);
  2839. a['class'] = ac.concat(bc).join(' ');
  2840. }
  2841. for (var key in b) {
  2842. if (key != 'class') {
  2843. a[key] = b[key];
  2844. }
  2845. }
  2846. return a;
  2847. };
  2848. /**
  2849. * Filter null `val`s.
  2850. *
  2851. * @param {Mixed} val
  2852. * @return {Mixed}
  2853. * @api private
  2854. */
  2855. function nulls(val) {
  2856. return val != null;
  2857. }
  2858. /**
  2859. * Render the given attributes object.
  2860. *
  2861. * @param {Object} obj
  2862. * @param {Object} escaped
  2863. * @return {String}
  2864. * @api private
  2865. */
  2866. exports.attrs = function attrs(obj, escaped){
  2867. var buf = []
  2868. , terse = obj.terse;
  2869. delete obj.terse;
  2870. var keys = Object.keys(obj)
  2871. , len = keys.length;
  2872. if (len) {
  2873. buf.push('');
  2874. for (var i = 0; i < len; ++i) {
  2875. var key = keys[i]
  2876. , val = obj[key];
  2877. if ('boolean' == typeof val || null == val) {
  2878. if (val) {
  2879. terse
  2880. ? buf.push(key)
  2881. : buf.push(key + '="' + key + '"');
  2882. }
  2883. } else if (0 == key.indexOf('data') && 'string' != typeof val) {
  2884. buf.push(key + "='" + JSON.stringify(val) + "'");
  2885. } else if ('class' == key && Array.isArray(val)) {
  2886. buf.push(key + '="' + exports.escape(val.join(' ')) + '"');
  2887. } else if (escaped && escaped[key]) {
  2888. buf.push(key + '="' + exports.escape(val) + '"');
  2889. } else {
  2890. buf.push(key + '="' + val + '"');
  2891. }
  2892. }
  2893. }
  2894. return buf.join(' ');
  2895. };
  2896. /**
  2897. * Escape the given string of `html`.
  2898. *
  2899. * @param {String} html
  2900. * @return {String}
  2901. * @api private
  2902. */
  2903. exports.escape = function escape(html){
  2904. return String(html)
  2905. .replace(/&(?!(\w+|\#\d+);)/g, '&amp;')
  2906. .replace(/</g, '&lt;')
  2907. .replace(/>/g, '&gt;')
  2908. .replace(/"/g, '&quot;');
  2909. };
  2910. /**
  2911. * Re-throw the given `err` in context to the
  2912. * the jade in `filename` at the given `lineno`.
  2913. *
  2914. * @param {Error} err
  2915. * @param {String} filename
  2916. * @param {String} lineno
  2917. * @api private
  2918. */
  2919. exports.rethrow = function rethrow(err, filename, lineno){
  2920. if (!filename) throw err;
  2921. var context = 3
  2922. , str = require('fs').readFileSync(filename, 'utf8')
  2923. , lines = str.split('\n')
  2924. , start = Math.max(lineno - context, 0)
  2925. , end = Math.min(lines.length, lineno + context);
  2926. // Error context
  2927. var context = lines.slice(start, end).map(function(line, i){
  2928. var curr = i + start + 1;
  2929. return (curr == lineno ? ' > ' : ' ')
  2930. + curr
  2931. + '| '
  2932. + line;
  2933. }).join('\n');
  2934. // Alter exception message
  2935. err.path = filename;
  2936. err.message = (filename || 'Jade') + ':' + lineno
  2937. + '\n' + context + '\n\n' + err.message;
  2938. throw err;
  2939. };
  2940. }); // module: runtime.js
  2941. require.register("self-closing.js", function(module, exports, require){
  2942. /*!
  2943. * Jade - self closing tags
  2944. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  2945. * MIT Licensed
  2946. */
  2947. module.exports = [
  2948. 'meta'
  2949. , 'img'
  2950. , 'link'
  2951. , 'input'
  2952. , 'source'
  2953. , 'area'
  2954. , 'base'
  2955. , 'col'
  2956. , 'br'
  2957. , 'hr'
  2958. ];
  2959. }); // module: self-closing.js
  2960. require.register("utils.js", function(module, exports, require){
  2961. /*!
  2962. * Jade - utils
  2963. * Copyright(c) 2010 TJ Holowaychuk <tj@vision-media.ca>
  2964. * MIT Licensed
  2965. */
  2966. /**
  2967. * Convert interpolation in the given string to JavaScript.
  2968. *
  2969. * @param {String} str
  2970. * @return {String}
  2971. * @api private
  2972. */
  2973. var interpolate = exports.interpolate = function(str){
  2974. return str.replace(/(_SLASH_)?([#!]){(.*?)}/g, function(str, escape, flag, code){
  2975. code = code
  2976. .replace(/\\'/g, "'")
  2977. .replace(/_SLASH_/g, '\\');
  2978. return escape
  2979. ? str.slice(7)
  2980. : "' + "
  2981. + ('!' == flag ? '' : 'escape')
  2982. + "((interp = " + code
  2983. + ") == null ? '' : interp) + '";
  2984. });
  2985. };
  2986. /**
  2987. * Escape single quotes in `str`.
  2988. *
  2989. * @param {String} str
  2990. * @return {String}
  2991. * @api private
  2992. */
  2993. var escape = exports.escape = function(str) {
  2994. return str.replace(/'/g, "\\'");
  2995. };
  2996. /**
  2997. * Interpolate, and escape the given `str`.
  2998. *
  2999. * @param {String} str
  3000. * @return {String}
  3001. * @api private
  3002. */
  3003. exports.text = function(str){
  3004. return interpolate(escape(str));
  3005. };
  3006. /**
  3007. * Merge `b` into `a`.
  3008. *
  3009. * @param {Object} a
  3010. * @param {Object} b
  3011. * @return {Object}
  3012. * @api public
  3013. */
  3014. exports.merge = function(a, b) {
  3015. for (var key in b) a[key] = b[key];
  3016. return a;
  3017. };
  3018. }); // module: utils.js
  3019. window.jade = require("jade");
  3020. })();