TemplateLiteralTransformer.ts 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. import { inject, injectable, } from 'inversify';
  2. import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';
  3. import * as ESTree from 'estree';
  4. import { IOptions } from '../../interfaces/options/IOptions';
  5. import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
  6. import { IVisitor } from '../../interfaces/node-transformers/IVisitor';
  7. import { TransformationStage } from '../../enums/node-transformers/TransformationStage';
  8. import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
  9. import { NodeFactory } from '../../node/NodeFactory';
  10. import { NodeGuards } from '../../node/NodeGuards';
  11. /**
  12. * Transform ES2015 template literals to ES5
  13. * Thanks to Babel for algorithm
  14. */
  15. @injectable()
  16. export class TemplateLiteralTransformer extends AbstractNodeTransformer {
  17. /**
  18. * @param {IRandomGenerator} randomGenerator
  19. * @param {IOptions} options
  20. */
  21. constructor (
  22. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  23. @inject(ServiceIdentifiers.IOptions) options: IOptions
  24. ) {
  25. super(randomGenerator, options);
  26. }
  27. /**
  28. * @param {NodeGuards} node
  29. * @returns {boolean}
  30. */
  31. private static isLiteralNodeWithStringValue (node: ESTree.Node): boolean {
  32. return node && NodeGuards.isLiteralNode(node) && typeof node.value === 'string';
  33. }
  34. /**
  35. * @param {TransformationStage} transformationStage
  36. * @returns {IVisitor | null}
  37. */
  38. public getVisitor (transformationStage: TransformationStage): IVisitor | null {
  39. switch (transformationStage) {
  40. case TransformationStage.Converting:
  41. return {
  42. leave: (node: ESTree.Node, parentNode: ESTree.Node | null) => {
  43. if (parentNode && NodeGuards.isTemplateLiteralNode(node)) {
  44. return this.transformNode(node, parentNode);
  45. }
  46. }
  47. };
  48. default:
  49. return null;
  50. }
  51. }
  52. /**
  53. * @param {TemplateLiteral} templateLiteralNode
  54. * @param {NodeGuards} parentNode
  55. * @returns {NodeGuards}
  56. */
  57. public transformNode (templateLiteralNode: ESTree.TemplateLiteral, parentNode: ESTree.Node): ESTree.Node {
  58. const templateLiteralExpressions: ESTree.Expression[] = templateLiteralNode.expressions;
  59. let nodes: ESTree.Expression[] = [];
  60. templateLiteralNode.quasis.forEach((templateElement: ESTree.TemplateElement) => {
  61. nodes.push(NodeFactory.literalNode(templateElement.value.cooked));
  62. const expression: ESTree.Expression | undefined = templateLiteralExpressions.shift();
  63. if (!expression) {
  64. return;
  65. }
  66. nodes.push(expression);
  67. });
  68. nodes = nodes.filter((node: ESTree.Literal | ESTree.Expression) => {
  69. return !(NodeGuards.isLiteralNode(node) && node.value === '');
  70. });
  71. // since `+` is left-to-right associative
  72. // ensure the first node is a string if first/second isn't
  73. if (
  74. !TemplateLiteralTransformer.isLiteralNodeWithStringValue(nodes[0]) &&
  75. !TemplateLiteralTransformer.isLiteralNodeWithStringValue(nodes[1])
  76. ) {
  77. nodes.unshift(NodeFactory.literalNode(''));
  78. }
  79. if (nodes.length > 1) {
  80. let root: ESTree.BinaryExpression = NodeFactory.binaryExpressionNode(
  81. '+',
  82. <ESTree.Literal>nodes.shift(),
  83. <ESTree.Expression>nodes.shift()
  84. );
  85. nodes.forEach((node: ESTree.Literal | ESTree.Expression) => {
  86. root = NodeFactory.binaryExpressionNode('+', root, node);
  87. });
  88. return root;
  89. }
  90. return nodes[0];
  91. }
  92. }