babylon.andOrNotEvaluator.ts 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. module BABYLON.Internals {
  2. export class AndOrNotEvaluator {
  3. public static Eval(query: string, evaluateCallback: (val) => boolean): boolean {
  4. if (!query.match(/\([^\(\)]*\)/g)) {
  5. query = AndOrNotEvaluator._HandleParenthesisContent(query, evaluateCallback);
  6. }
  7. else {
  8. query = query.replace(/\([^\(\)]*\)/g, r => {
  9. // remove parenthesis
  10. r = r.slice(1, r.length - 1);
  11. return AndOrNotEvaluator._HandleParenthesisContent(r, evaluateCallback);
  12. });
  13. }
  14. if (query === "true") {
  15. return true;
  16. }
  17. if (query === "false") {
  18. return false;
  19. }
  20. return AndOrNotEvaluator.Eval(query, evaluateCallback);
  21. }
  22. private static _HandleParenthesisContent(parenthesisContent: string, evaluateCallback: (val) => boolean): string {
  23. evaluateCallback = evaluateCallback || ((r) => {
  24. return r === "true" ? true : false;
  25. });
  26. var result;
  27. var or = parenthesisContent.split("||");
  28. for (var i in or) {
  29. var ori = AndOrNotEvaluator._SimplifyNegation(or[i].trim());
  30. var and = ori.split("&&");
  31. if (and.length > 1) {
  32. for (var j = 0; j < and.length; ++j) {
  33. var andj = AndOrNotEvaluator._SimplifyNegation(and[j].trim());
  34. if (andj !== "true" && andj !== "false") {
  35. if (andj[0] === "!") {
  36. result = !evaluateCallback(andj.substring(1));
  37. }
  38. else {
  39. result = evaluateCallback(andj);
  40. }
  41. }
  42. else {
  43. result = andj === "true" ? true : false;
  44. }
  45. if (!result) { // no need to continue since 'false && ... && ...' will always return false
  46. ori = "false";
  47. break;
  48. }
  49. }
  50. }
  51. if (result || ori === "true") { // no need to continue since 'true || ... || ...' will always return true
  52. result = true;
  53. break;
  54. }
  55. // result equals false (or undefined)
  56. if (ori !== "true" && ori !== "false") {
  57. if (ori[0] === "!") {
  58. result = !evaluateCallback(ori.substring(1));
  59. }
  60. else {
  61. result = evaluateCallback(ori);
  62. }
  63. }
  64. else {
  65. result = ori === "true" ? true : false;
  66. }
  67. }
  68. // the whole parenthesis scope is replaced by 'true' or 'false'
  69. return result ? "true" : "false";
  70. }
  71. private static _SimplifyNegation(booleanString: string): string {
  72. booleanString = booleanString.replace(/^[\s!]+/, r => {
  73. // remove whitespaces
  74. r = r.replace(/[\s]/g, () => "");
  75. return r.length % 2 ? "!" : "";
  76. });
  77. booleanString = booleanString.trim();
  78. if (booleanString === "!true") {
  79. booleanString = "false";
  80. }
  81. else if (booleanString === "!false") {
  82. booleanString = "true";
  83. }
  84. return booleanString;
  85. }
  86. }
  87. }