printer.js 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526
  1. import { escapeAttrValue, escapeText, sortByLoc } from './util';
  2. export const voidMap = Object.create(null);
  3. let voidTagNames = 'area base br col command embed hr img input keygen link meta param source track wbr';
  4. voidTagNames.split(' ').forEach(tagName => {
  5. voidMap[tagName] = true;
  6. });
  7. const NON_WHITESPACE = /\S/;
  8. export default class Printer {
  9. constructor(options) {
  10. this.buffer = '';
  11. this.options = options;
  12. }
  13. /*
  14. This is used by _all_ methods on this Printer class that add to `this.buffer`,
  15. it allows consumers of the printer to use alternate string representations for
  16. a given node.
  17. The primary use case for this are things like source -> source codemod utilities.
  18. For example, ember-template-recast attempts to always preserve the original string
  19. formatting in each AST node if no modifications are made to it.
  20. */
  21. handledByOverride(node, ensureLeadingWhitespace = false) {
  22. if (this.options.override !== undefined) {
  23. let result = this.options.override(node, this.options);
  24. if (typeof result === 'string') {
  25. if (ensureLeadingWhitespace && result !== '' && NON_WHITESPACE.test(result[0])) {
  26. result = ` ${result}`;
  27. }
  28. this.buffer += result;
  29. return true;
  30. }
  31. }
  32. return false;
  33. }
  34. Node(node) {
  35. switch (node.type) {
  36. case 'MustacheStatement':
  37. case 'BlockStatement':
  38. case 'PartialStatement':
  39. case 'MustacheCommentStatement':
  40. case 'CommentStatement':
  41. case 'TextNode':
  42. case 'ElementNode':
  43. case 'AttrNode':
  44. case 'Block':
  45. case 'Template':
  46. return this.TopLevelStatement(node);
  47. case 'StringLiteral':
  48. case 'BooleanLiteral':
  49. case 'NumberLiteral':
  50. case 'UndefinedLiteral':
  51. case 'NullLiteral':
  52. case 'PathExpression':
  53. case 'SubExpression':
  54. return this.Expression(node);
  55. case 'Program':
  56. return this.Block(node);
  57. case 'ConcatStatement':
  58. // should have an AttrNode parent
  59. return this.ConcatStatement(node);
  60. case 'Hash':
  61. return this.Hash(node);
  62. case 'HashPair':
  63. return this.HashPair(node);
  64. case 'ElementModifierStatement':
  65. return this.ElementModifierStatement(node);
  66. }
  67. }
  68. Expression(expression) {
  69. switch (expression.type) {
  70. case 'StringLiteral':
  71. case 'BooleanLiteral':
  72. case 'NumberLiteral':
  73. case 'UndefinedLiteral':
  74. case 'NullLiteral':
  75. return this.Literal(expression);
  76. case 'PathExpression':
  77. return this.PathExpression(expression);
  78. case 'SubExpression':
  79. return this.SubExpression(expression);
  80. }
  81. }
  82. Literal(literal) {
  83. switch (literal.type) {
  84. case 'StringLiteral':
  85. return this.StringLiteral(literal);
  86. case 'BooleanLiteral':
  87. return this.BooleanLiteral(literal);
  88. case 'NumberLiteral':
  89. return this.NumberLiteral(literal);
  90. case 'UndefinedLiteral':
  91. return this.UndefinedLiteral(literal);
  92. case 'NullLiteral':
  93. return this.NullLiteral(literal);
  94. }
  95. }
  96. TopLevelStatement(statement) {
  97. switch (statement.type) {
  98. case 'MustacheStatement':
  99. return this.MustacheStatement(statement);
  100. case 'BlockStatement':
  101. return this.BlockStatement(statement);
  102. case 'PartialStatement':
  103. return this.PartialStatement(statement);
  104. case 'MustacheCommentStatement':
  105. return this.MustacheCommentStatement(statement);
  106. case 'CommentStatement':
  107. return this.CommentStatement(statement);
  108. case 'TextNode':
  109. return this.TextNode(statement);
  110. case 'ElementNode':
  111. return this.ElementNode(statement);
  112. case 'Block':
  113. case 'Template':
  114. return this.Block(statement);
  115. case 'AttrNode':
  116. // should have element
  117. return this.AttrNode(statement);
  118. }
  119. }
  120. Block(block) {
  121. /*
  122. When processing a template like:
  123. ```hbs
  124. {{#if whatever}}
  125. whatever
  126. {{else if somethingElse}}
  127. something else
  128. {{else}}
  129. fallback
  130. {{/if}}
  131. ```
  132. The AST still _effectively_ looks like:
  133. ```hbs
  134. {{#if whatever}}
  135. whatever
  136. {{else}}{{#if somethingElse}}
  137. something else
  138. {{else}}
  139. fallback
  140. {{/if}}{{/if}}
  141. ```
  142. The only way we can tell if that is the case is by checking for
  143. `block.chained`, but unfortunately when the actual statements are
  144. processed the `block.body[0]` node (which will always be a
  145. `BlockStatement`) has no clue that its ancestor `Block` node was
  146. chained.
  147. This "forwards" the `chained` setting so that we can check
  148. it later when processing the `BlockStatement`.
  149. */
  150. if (block.chained) {
  151. let firstChild = block.body[0];
  152. firstChild.chained = true;
  153. }
  154. if (this.handledByOverride(block)) {
  155. return;
  156. }
  157. this.TopLevelStatements(block.body);
  158. }
  159. TopLevelStatements(statements) {
  160. statements.forEach(statement => this.TopLevelStatement(statement));
  161. }
  162. ElementNode(el) {
  163. if (this.handledByOverride(el)) {
  164. return;
  165. }
  166. this.OpenElementNode(el);
  167. this.TopLevelStatements(el.children);
  168. this.CloseElementNode(el);
  169. }
  170. OpenElementNode(el) {
  171. this.buffer += `<${el.tag}`;
  172. const parts = [...el.attributes, ...el.modifiers, ...el.comments].sort(sortByLoc);
  173. for (const part of parts) {
  174. this.buffer += ' ';
  175. switch (part.type) {
  176. case 'AttrNode':
  177. this.AttrNode(part);
  178. break;
  179. case 'ElementModifierStatement':
  180. this.ElementModifierStatement(part);
  181. break;
  182. case 'MustacheCommentStatement':
  183. this.MustacheCommentStatement(part);
  184. break;
  185. }
  186. }
  187. if (el.blockParams.length) {
  188. this.BlockParams(el.blockParams);
  189. }
  190. if (el.selfClosing) {
  191. this.buffer += ' /';
  192. }
  193. this.buffer += '>';
  194. }
  195. CloseElementNode(el) {
  196. if (el.selfClosing || voidMap[el.tag.toLowerCase()]) {
  197. return;
  198. }
  199. this.buffer += `</${el.tag}>`;
  200. }
  201. AttrNode(attr) {
  202. if (this.handledByOverride(attr)) {
  203. return;
  204. }
  205. let {
  206. name,
  207. value
  208. } = attr;
  209. this.buffer += name;
  210. if (value.type !== 'TextNode' || value.chars.length > 0) {
  211. this.buffer += '=';
  212. this.AttrNodeValue(value);
  213. }
  214. }
  215. AttrNodeValue(value) {
  216. if (value.type === 'TextNode') {
  217. this.buffer += '"';
  218. this.TextNode(value, true);
  219. this.buffer += '"';
  220. } else {
  221. this.Node(value);
  222. }
  223. }
  224. TextNode(text, isAttr) {
  225. if (this.handledByOverride(text)) {
  226. return;
  227. }
  228. if (this.options.entityEncoding === 'raw') {
  229. this.buffer += text.chars;
  230. } else if (isAttr) {
  231. this.buffer += escapeAttrValue(text.chars);
  232. } else {
  233. this.buffer += escapeText(text.chars);
  234. }
  235. }
  236. MustacheStatement(mustache) {
  237. if (this.handledByOverride(mustache)) {
  238. return;
  239. }
  240. this.buffer += mustache.escaped ? '{{' : '{{{';
  241. if (mustache.strip.open) {
  242. this.buffer += '~';
  243. }
  244. this.Expression(mustache.path);
  245. this.Params(mustache.params);
  246. this.Hash(mustache.hash);
  247. if (mustache.strip.close) {
  248. this.buffer += '~';
  249. }
  250. this.buffer += mustache.escaped ? '}}' : '}}}';
  251. }
  252. BlockStatement(block) {
  253. if (this.handledByOverride(block)) {
  254. return;
  255. }
  256. if (block.chained) {
  257. this.buffer += block.inverseStrip.open ? '{{~' : '{{';
  258. this.buffer += 'else ';
  259. } else {
  260. this.buffer += block.openStrip.open ? '{{~#' : '{{#';
  261. }
  262. this.Expression(block.path);
  263. this.Params(block.params);
  264. this.Hash(block.hash);
  265. if (block.program.blockParams.length) {
  266. this.BlockParams(block.program.blockParams);
  267. }
  268. if (block.chained) {
  269. this.buffer += block.inverseStrip.close ? '~}}' : '}}';
  270. } else {
  271. this.buffer += block.openStrip.close ? '~}}' : '}}';
  272. }
  273. this.Block(block.program);
  274. if (block.inverse) {
  275. if (!block.inverse.chained) {
  276. this.buffer += block.inverseStrip.open ? '{{~' : '{{';
  277. this.buffer += 'else';
  278. this.buffer += block.inverseStrip.close ? '~}}' : '}}';
  279. }
  280. this.Block(block.inverse);
  281. }
  282. if (!block.chained) {
  283. this.buffer += block.closeStrip.open ? '{{~/' : '{{/';
  284. this.Expression(block.path);
  285. this.buffer += block.closeStrip.close ? '~}}' : '}}';
  286. }
  287. }
  288. BlockParams(blockParams) {
  289. this.buffer += ` as |${blockParams.join(' ')}|`;
  290. }
  291. PartialStatement(partial) {
  292. if (this.handledByOverride(partial)) {
  293. return;
  294. }
  295. this.buffer += '{{>';
  296. this.Expression(partial.name);
  297. this.Params(partial.params);
  298. this.Hash(partial.hash);
  299. this.buffer += '}}';
  300. }
  301. ConcatStatement(concat) {
  302. if (this.handledByOverride(concat)) {
  303. return;
  304. }
  305. this.buffer += '"';
  306. concat.parts.forEach(part => {
  307. if (part.type === 'TextNode') {
  308. this.TextNode(part, true);
  309. } else {
  310. this.Node(part);
  311. }
  312. });
  313. this.buffer += '"';
  314. }
  315. MustacheCommentStatement(comment) {
  316. if (this.handledByOverride(comment)) {
  317. return;
  318. }
  319. this.buffer += `{{!--${comment.value}--}}`;
  320. }
  321. ElementModifierStatement(mod) {
  322. if (this.handledByOverride(mod)) {
  323. return;
  324. }
  325. this.buffer += '{{';
  326. this.Expression(mod.path);
  327. this.Params(mod.params);
  328. this.Hash(mod.hash);
  329. this.buffer += '}}';
  330. }
  331. CommentStatement(comment) {
  332. if (this.handledByOverride(comment)) {
  333. return;
  334. }
  335. this.buffer += `<!--${comment.value}-->`;
  336. }
  337. PathExpression(path) {
  338. if (this.handledByOverride(path)) {
  339. return;
  340. }
  341. this.buffer += path.original;
  342. }
  343. SubExpression(sexp) {
  344. if (this.handledByOverride(sexp)) {
  345. return;
  346. }
  347. this.buffer += '(';
  348. this.Expression(sexp.path);
  349. this.Params(sexp.params);
  350. this.Hash(sexp.hash);
  351. this.buffer += ')';
  352. }
  353. Params(params) {
  354. // TODO: implement a top level Params AST node (just like the Hash object)
  355. // so that this can also be overridden
  356. if (params.length) {
  357. params.forEach(param => {
  358. this.buffer += ' ';
  359. this.Expression(param);
  360. });
  361. }
  362. }
  363. Hash(hash) {
  364. if (this.handledByOverride(hash, true)) {
  365. return;
  366. }
  367. hash.pairs.forEach(pair => {
  368. this.buffer += ' ';
  369. this.HashPair(pair);
  370. });
  371. }
  372. HashPair(pair) {
  373. if (this.handledByOverride(pair)) {
  374. return;
  375. }
  376. this.buffer += pair.key;
  377. this.buffer += '=';
  378. this.Node(pair.value);
  379. }
  380. StringLiteral(str) {
  381. if (this.handledByOverride(str)) {
  382. return;
  383. }
  384. this.buffer += JSON.stringify(str.value);
  385. }
  386. BooleanLiteral(bool) {
  387. if (this.handledByOverride(bool)) {
  388. return;
  389. }
  390. this.buffer += bool.value;
  391. }
  392. NumberLiteral(number) {
  393. if (this.handledByOverride(number)) {
  394. return;
  395. }
  396. this.buffer += number.value;
  397. }
  398. UndefinedLiteral(node) {
  399. if (this.handledByOverride(node)) {
  400. return;
  401. }
  402. this.buffer += 'undefined';
  403. }
  404. NullLiteral(node) {
  405. if (this.handledByOverride(node)) {
  406. return;
  407. }
  408. this.buffer += 'null';
  409. }
  410. print(node) {
  411. let {
  412. options
  413. } = this;
  414. if (options.override) {
  415. let result = options.override(node, options);
  416. if (result !== undefined) {
  417. return result;
  418. }
  419. }
  420. this.buffer = '';
  421. this.Node(node);
  422. return this.buffer;
  423. }
  424. }
  425. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../packages/@glimmer/syntax/lib/generation/printer.ts"],"names":[],"mappings":"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","sourcesContent":["import * as ASTv1 from '../v1/api';\nimport { escapeAttrValue, escapeText, sortByLoc } from './util';\n\nexport const voidMap: {\n  [tagName: string]: boolean;\n} = Object.create(null);\n\nlet voidTagNames =\n  'area base br col command embed hr img input keygen link meta param source track wbr';\nvoidTagNames.split(' ').forEach((tagName) => {\n  voidMap[tagName] = true;\n});\n\nconst NON_WHITESPACE = /\\S/;\n\nexport interface PrinterOptions {\n  entityEncoding: 'transformed' | 'raw';\n\n  /**\n   * Used to override the mechanism of printing a given AST.Node.\n   *\n   * This will generally only be useful to source -> source codemods\n   * where you would like to specialize/override the way a given node is\n   * printed (e.g. you would like to preserve as much of the original\n   * formatting as possible).\n   *\n   * When the provided override returns undefined, the default built in printing\n   * will be done for the AST.Node.\n   *\n   * @param ast the ast node to be printed\n   * @param options the options specified during the print() invocation\n   */\n  override?(ast: ASTv1.Node, options: PrinterOptions): void | string;\n}\n\nexport default class Printer {\n  private buffer = '';\n  private options: PrinterOptions;\n\n  constructor(options: PrinterOptions) {\n    this.options = options;\n  }\n\n  /*\n    This is used by _all_ methods on this Printer class that add to `this.buffer`,\n    it allows consumers of the printer to use alternate string representations for\n    a given node.\n\n    The primary use case for this are things like source -> source codemod utilities.\n    For example, ember-template-recast attempts to always preserve the original string\n    formatting in each AST node if no modifications are made to it.\n  */\n  handledByOverride(node: ASTv1.Node, ensureLeadingWhitespace = false): boolean {\n    if (this.options.override !== undefined) {\n      let result = this.options.override(node, this.options);\n      if (typeof result === 'string') {\n        if (ensureLeadingWhitespace && result !== '' && NON_WHITESPACE.test(result[0])) {\n          result = ` ${result}`;\n        }\n\n        this.buffer += result;\n        return true;\n      }\n    }\n\n    return false;\n  }\n\n  Node(node: ASTv1.Node): void {\n    switch (node.type) {\n      case 'MustacheStatement':\n      case 'BlockStatement':\n      case 'PartialStatement':\n      case 'MustacheCommentStatement':\n      case 'CommentStatement':\n      case 'TextNode':\n      case 'ElementNode':\n      case 'AttrNode':\n      case 'Block':\n      case 'Template':\n        return this.TopLevelStatement(node);\n      case 'StringLiteral':\n      case 'BooleanLiteral':\n      case 'NumberLiteral':\n      case 'UndefinedLiteral':\n      case 'NullLiteral':\n      case 'PathExpression':\n      case 'SubExpression':\n        return this.Expression(node);\n      case 'Program':\n        return this.Block(node);\n      case 'ConcatStatement':\n        // should have an AttrNode parent\n        return this.ConcatStatement(node);\n      case 'Hash':\n        return this.Hash(node);\n      case 'HashPair':\n        return this.HashPair(node);\n      case 'ElementModifierStatement':\n        return this.ElementModifierStatement(node);\n    }\n  }\n\n  Expression(expression: ASTv1.Expression): void {\n    switch (expression.type) {\n      case 'StringLiteral':\n      case 'BooleanLiteral':\n      case 'NumberLiteral':\n      case 'UndefinedLiteral':\n      case 'NullLiteral':\n        return this.Literal(expression);\n      case 'PathExpression':\n        return this.PathExpression(expression);\n      case 'SubExpression':\n        return this.SubExpression(expression);\n    }\n  }\n\n  Literal(literal: ASTv1.Literal): void {\n    switch (literal.type) {\n      case 'StringLiteral':\n        return this.StringLiteral(literal);\n      case 'BooleanLiteral':\n        return this.BooleanLiteral(literal);\n      case 'NumberLiteral':\n        return this.NumberLiteral(literal);\n      case 'UndefinedLiteral':\n        return this.UndefinedLiteral(literal);\n      case 'NullLiteral':\n        return this.NullLiteral(literal);\n    }\n  }\n\n  TopLevelStatement(statement: ASTv1.TopLevelStatement | ASTv1.Template | ASTv1.AttrNode): void {\n    switch (statement.type) {\n      case 'MustacheStatement':\n        return this.MustacheStatement(statement);\n      case 'BlockStatement':\n        return this.BlockStatement(statement);\n      case 'PartialStatement':\n        return this.PartialStatement(statement);\n      case 'MustacheCommentStatement':\n        return this.MustacheCommentStatement(statement);\n      case 'CommentStatement':\n        return this.CommentStatement(statement);\n      case 'TextNode':\n        return this.TextNode(statement);\n      case 'ElementNode':\n        return this.ElementNode(statement);\n      case 'Block':\n      case 'Template':\n        return this.Block(statement);\n      case 'AttrNode':\n        // should have element\n        return this.AttrNode(statement);\n    }\n  }\n\n  Block(block: ASTv1.Block | ASTv1.Program | ASTv1.Template): void {\n    /*\n      When processing a template like:\n\n      ```hbs\n      {{#if whatever}}\n        whatever\n      {{else if somethingElse}}\n        something else\n      {{else}}\n        fallback\n      {{/if}}\n      ```\n\n      The AST still _effectively_ looks like:\n\n      ```hbs\n      {{#if whatever}}\n        whatever\n      {{else}}{{#if somethingElse}}\n        something else\n      {{else}}\n        fallback\n      {{/if}}{{/if}}\n      ```\n\n      The only way we can tell if that is the case is by checking for\n      `block.chained`, but unfortunately when the actual statements are\n      processed the `block.body[0]` node (which will always be a\n      `BlockStatement`) has no clue that its ancestor `Block` node was\n      chained.\n\n      This \"forwards\" the `chained` setting so that we can check\n      it later when processing the `BlockStatement`.\n    */\n    if (block.chained) {\n      let firstChild = block.body[0] as ASTv1.BlockStatement;\n      firstChild.chained = true;\n    }\n\n    if (this.handledByOverride(block)) {\n      return;\n    }\n\n    this.TopLevelStatements(block.body);\n  }\n\n  TopLevelStatements(statements: ASTv1.TopLevelStatement[]): void {\n    statements.forEach((statement) => this.TopLevelStatement(statement));\n  }\n\n  ElementNode(el: ASTv1.ElementNode): void {\n    if (this.handledByOverride(el)) {\n      return;\n    }\n\n    this.OpenElementNode(el);\n    this.TopLevelStatements(el.children);\n    this.CloseElementNode(el);\n  }\n\n  OpenElementNode(el: ASTv1.ElementNode): void {\n    this.buffer += `<${el.tag}`;\n    const parts = [...el.attributes, ...el.modifiers, ...el.comments].sort(sortByLoc);\n\n    for (const part of parts) {\n      this.buffer += ' ';\n      switch (part.type) {\n        case 'AttrNode':\n          this.AttrNode(part);\n          break;\n        case 'ElementModifierStatement':\n          this.ElementModifierStatement(part);\n          break;\n        case 'MustacheCommentStatement':\n          this.MustacheCommentStatement(part);\n          break;\n      }\n    }\n    if (el.blockParams.length) {\n      this.BlockParams(el.blockParams);\n    }\n    if (el.selfClosing) {\n      this.buffer += ' /';\n    }\n    this.buffer += '>';\n  }\n\n  CloseElementNode(el: ASTv1.ElementNode): void {\n    if (el.selfClosing || voidMap[el.tag.toLowerCase()]) {\n      return;\n    }\n    this.buffer += `</${el.tag}>`;\n  }\n\n  AttrNode(attr: ASTv1.AttrNode): void {\n    if (this.handledByOverride(attr)) {\n      return;\n    }\n\n    let { name, value } = attr;\n\n    this.buffer += name;\n    if (value.type !== 'TextNode' || value.chars.length > 0) {\n      this.buffer += '=';\n      this.AttrNodeValue(value);\n    }\n  }\n\n  AttrNodeValue(value: ASTv1.AttrNode['value']): void {\n    if (value.type === 'TextNode') {\n      this.buffer += '\"';\n      this.TextNode(value, true);\n      this.buffer += '\"';\n    } else {\n      this.Node(value);\n    }\n  }\n\n  TextNode(text: ASTv1.TextNode, isAttr?: boolean): void {\n    if (this.handledByOverride(text)) {\n      return;\n    }\n\n    if (this.options.entityEncoding === 'raw') {\n      this.buffer += text.chars;\n    } else if (isAttr) {\n      this.buffer += escapeAttrValue(text.chars);\n    } else {\n      this.buffer += escapeText(text.chars);\n    }\n  }\n\n  MustacheStatement(mustache: ASTv1.MustacheStatement): void {\n    if (this.handledByOverride(mustache)) {\n      return;\n    }\n\n    this.buffer += mustache.escaped ? '{{' : '{{{';\n\n    if (mustache.strip.open) {\n      this.buffer += '~';\n    }\n\n    this.Expression(mustache.path);\n    this.Params(mustache.params);\n    this.Hash(mustache.hash);\n\n    if (mustache.strip.close) {\n      this.buffer += '~';\n    }\n\n    this.buffer += mustache.escaped ? '}}' : '}}}';\n  }\n\n  BlockStatement(block: ASTv1.BlockStatement): void {\n    if (this.handledByOverride(block)) {\n      return;\n    }\n\n    if (block.chained) {\n      this.buffer += block.inverseStrip.open ? '{{~' : '{{';\n      this.buffer += 'else ';\n    } else {\n      this.buffer += block.openStrip.open ? '{{~#' : '{{#';\n    }\n\n    this.Expression(block.path);\n    this.Params(block.params);\n    this.Hash(block.hash);\n    if (block.program.blockParams.length) {\n      this.BlockParams(block.program.blockParams);\n    }\n\n    if (block.chained) {\n      this.buffer += block.inverseStrip.close ? '~}}' : '}}';\n    } else {\n      this.buffer += block.openStrip.close ? '~}}' : '}}';\n    }\n\n    this.Block(block.program);\n\n    if (block.inverse) {\n      if (!block.inverse.chained) {\n        this.buffer += block.inverseStrip.open ? '{{~' : '{{';\n        this.buffer += 'else';\n        this.buffer += block.inverseStrip.close ? '~}}' : '}}';\n      }\n\n      this.Block(block.inverse);\n    }\n\n    if (!block.chained) {\n      this.buffer += block.closeStrip.open ? '{{~/' : '{{/';\n      this.Expression(block.path);\n      this.buffer += block.closeStrip.close ? '~}}' : '}}';\n    }\n  }\n\n  BlockParams(blockParams: string[]): void {\n    this.buffer += ` as |${blockParams.join(' ')}|`;\n  }\n\n  PartialStatement(partial: ASTv1.PartialStatement): void {\n    if (this.handledByOverride(partial)) {\n      return;\n    }\n\n    this.buffer += '{{>';\n    this.Expression(partial.name);\n    this.Params(partial.params);\n    this.Hash(partial.hash);\n    this.buffer += '}}';\n  }\n\n  ConcatStatement(concat: ASTv1.ConcatStatement): void {\n    if (this.handledByOverride(concat)) {\n      return;\n    }\n\n    this.buffer += '\"';\n    concat.parts.forEach((part) => {\n      if (part.type === 'TextNode') {\n        this.TextNode(part, true);\n      } else {\n        this.Node(part);\n      }\n    });\n    this.buffer += '\"';\n  }\n\n  MustacheCommentStatement(comment: ASTv1.MustacheCommentStatement): void {\n    if (this.handledByOverride(comment)) {\n      return;\n    }\n\n    this.buffer += `{{!--${comment.value}--}}`;\n  }\n\n  ElementModifierStatement(mod: ASTv1.ElementModifierStatement): void {\n    if (this.handledByOverride(mod)) {\n      return;\n    }\n\n    this.buffer += '{{';\n    this.Expression(mod.path);\n    this.Params(mod.params);\n    this.Hash(mod.hash);\n    this.buffer += '}}';\n  }\n\n  CommentStatement(comment: ASTv1.CommentStatement): void {\n    if (this.handledByOverride(comment)) {\n      return;\n    }\n\n    this.buffer += `<!--${comment.value}-->`;\n  }\n\n  PathExpression(path: ASTv1.PathExpression): void {\n    if (this.handledByOverride(path)) {\n      return;\n    }\n\n    this.buffer += path.original;\n  }\n\n  SubExpression(sexp: ASTv1.SubExpression): void {\n    if (this.handledByOverride(sexp)) {\n      return;\n    }\n\n    this.buffer += '(';\n    this.Expression(sexp.path);\n    this.Params(sexp.params);\n    this.Hash(sexp.hash);\n    this.buffer += ')';\n  }\n\n  Params(params: ASTv1.Expression[]): void {\n    // TODO: implement a top level Params AST node (just like the Hash object)\n    // so that this can also be overridden\n    if (params.length) {\n      params.forEach((param) => {\n        this.buffer += ' ';\n        this.Expression(param);\n      });\n    }\n  }\n\n  Hash(hash: ASTv1.Hash): void {\n    if (this.handledByOverride(hash, true)) {\n      return;\n    }\n\n    hash.pairs.forEach((pair) => {\n      this.buffer += ' ';\n      this.HashPair(pair);\n    });\n  }\n\n  HashPair(pair: ASTv1.HashPair): void {\n    if (this.handledByOverride(pair)) {\n      return;\n    }\n\n    this.buffer += pair.key;\n    this.buffer += '=';\n    this.Node(pair.value);\n  }\n\n  StringLiteral(str: ASTv1.StringLiteral): void {\n    if (this.handledByOverride(str)) {\n      return;\n    }\n\n    this.buffer += JSON.stringify(str.value);\n  }\n\n  BooleanLiteral(bool: ASTv1.BooleanLiteral): void {\n    if (this.handledByOverride(bool)) {\n      return;\n    }\n\n    this.buffer += bool.value;\n  }\n\n  NumberLiteral(number: ASTv1.NumberLiteral): void {\n    if (this.handledByOverride(number)) {\n      return;\n    }\n\n    this.buffer += number.value;\n  }\n\n  UndefinedLiteral(node: ASTv1.UndefinedLiteral): void {\n    if (this.handledByOverride(node)) {\n      return;\n    }\n\n    this.buffer += 'undefined';\n  }\n\n  NullLiteral(node: ASTv1.NullLiteral): void {\n    if (this.handledByOverride(node)) {\n      return;\n    }\n\n    this.buffer += 'null';\n  }\n\n  print(node: ASTv1.Node): string {\n    let { options } = this;\n\n    if (options.override) {\n      let result = options.override(node, options);\n\n      if (result !== undefined) {\n        return result;\n      }\n    }\n\n    this.buffer = '';\n    this.Node(node);\n    return this.buffer;\n  }\n}\n"],"sourceRoot":""}