index.js 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. /**
  2. * merge-estraverse-visitors
  3. * merge multiple estraverse visitors into one
  4. *
  5. * https://github.com/twada/merge-estraverse-visitors
  6. *
  7. * Copyright (c) 2016 Takuto Wada
  8. * Licensed under the MIT license.
  9. * https://twada.mit-license.org/
  10. */
  11. 'use strict';
  12. var estraverse = require('estraverse');
  13. function SubVisitor () {
  14. this.skipStartNode = null;
  15. this.broken = false;
  16. }
  17. SubVisitor.prototype.isBroken = function () {
  18. return !!this.broken;
  19. };
  20. SubVisitor.prototype.markBroken = function () {
  21. return this.broken = true;
  22. };
  23. SubVisitor.prototype.isSkipping = function (controller) {
  24. return this.skipStartNode && (this.skipStartNode !== controller.current());
  25. };
  26. SubVisitor.prototype.startSkipping = function (controller) {
  27. this.skipStartNode = controller.current();
  28. };
  29. SubVisitor.prototype.finishSkippingIfLeavingFrom = function (controller) {
  30. if (this.skipStartNode === controller.current()) {
  31. this.skipStartNode = null;
  32. }
  33. };
  34. function noop () {
  35. }
  36. function createSubVisitors (visitors) {
  37. var enters = [];
  38. var leaves = [];
  39. var subVisitor, i, v, len = visitors.length;
  40. for(i = 0; i < len; i += 1) {
  41. v = visitors[i];
  42. subVisitor = new SubVisitor();
  43. subVisitor.enter = (typeof v.enter === 'function') ? v.enter : noop;
  44. subVisitor.leave = (typeof v.leave === 'function') ? v.leave : noop;
  45. enters.push(subVisitor);
  46. leaves.unshift(subVisitor);
  47. }
  48. return {
  49. enters: enters,
  50. leaves: leaves
  51. };
  52. }
  53. module.exports = function mergeVisitors (visitors) {
  54. var subVisitors = createSubVisitors(visitors);
  55. return {
  56. enter: function (currentNode, parentNode) {
  57. var orig = this;
  58. subVisitors.enters.forEach(function (subVisitor) {
  59. var controller = Object.create(orig);
  60. if (subVisitor.isBroken()) {
  61. return;
  62. }
  63. if (subVisitor.isSkipping(controller)) {
  64. return;
  65. }
  66. controller.notify = function notify (flag) {
  67. switch (flag) {
  68. case estraverse.VisitorOption.Skip:
  69. subVisitor.startSkipping(controller);
  70. return;
  71. case estraverse.VisitorOption.Break:
  72. subVisitor.markBroken();
  73. return;
  74. default:
  75. orig.notify.call(orig, flag);
  76. }
  77. };
  78. var ret = subVisitor.enter.call(controller, currentNode, parentNode);
  79. switch (ret) {
  80. case estraverse.VisitorOption.Skip:
  81. subVisitor.startSkipping(controller);
  82. break;
  83. case estraverse.VisitorOption.Break:
  84. subVisitor.markBroken();
  85. break;
  86. }
  87. });
  88. },
  89. leave: function (currentNode, parentNode) {
  90. var orig = this;
  91. var replacements = [];
  92. subVisitors.leaves.forEach(function (subVisitor) {
  93. var controller = Object.create(orig);
  94. if (subVisitor.isBroken()) {
  95. return;
  96. }
  97. if (subVisitor.isSkipping(controller)) {
  98. return;
  99. }
  100. subVisitor.finishSkippingIfLeavingFrom(controller);
  101. controller.notify = function notify (flag) {
  102. switch (flag) {
  103. case estraverse.VisitorOption.Skip:
  104. // subVisitor.startSkipping(controller); // meaningless
  105. return;
  106. case estraverse.VisitorOption.Break:
  107. subVisitor.markBroken();
  108. return;
  109. default:
  110. orig.notify.call(orig, flag);
  111. }
  112. };
  113. var ret = subVisitor.leave.call(controller, currentNode, parentNode);
  114. switch (ret) {
  115. case estraverse.VisitorOption.Skip:
  116. // subVisitor.startSkipping(controller); // meaningless
  117. return;
  118. case estraverse.VisitorOption.Break:
  119. subVisitor.markBroken();
  120. return;
  121. }
  122. if (typeof ret === 'object' && ret !== null && typeof ret.type === 'string') {
  123. replacements.push(ret);
  124. }
  125. });
  126. if (replacements.length === 1) {
  127. return replacements[0];
  128. }
  129. return undefined;
  130. }
  131. };
  132. };