printer.js 46 KB

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