jquery.inputmask.regex.extensions.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. /*
  2. Input Mask plugin extensions
  3. http://github.com/RobinHerbots/jquery.inputmask
  4. Copyright (c) 2010 - 2014 Robin Herbots
  5. Licensed under the MIT license (http://www.opensource.org/licenses/mit-license.php)
  6. Version: 0.0.0
  7. Regex extensions on the jquery.inputmask base
  8. Allows for using regular expressions as a mask
  9. */
  10. (function (factory) {
  11. if (typeof define === 'function' && define.amd) {
  12. // AMD. Register as an anonymous module.
  13. define("jquery.inputmask.regex.extensions", ['jquery', 'jquery.inputmask'], factory);
  14. } else {
  15. // Browser globals
  16. factory(jQuery);
  17. }
  18. }(function ($) {
  19. $.extend($.inputmask.defaults.aliases, { // $(selector).inputmask("Regex", { regex: "[0-9]*"}
  20. 'Regex': {
  21. mask: "r",
  22. greedy: false,
  23. repeat: "*",
  24. regex: null,
  25. regexTokens: null,
  26. //Thx to https://github.com/slevithan/regex-colorizer for the tokenizer regex
  27. tokenizer: /\[\^?]?(?:[^\\\]]+|\\[\S\s]?)*]?|\\(?:0(?:[0-3][0-7]{0,2}|[4-7][0-7]?)?|[1-9][0-9]*|x[0-9A-Fa-f]{2}|u[0-9A-Fa-f]{4}|c[A-Za-z]|[\S\s]?)|\((?:\?[:=!]?)?|(?:[?*+]|\{[0-9]+(?:,[0-9]*)?\})\??|[^.?*+^${[()|\\]+|./g,
  28. quantifierFilter: /[0-9]+[^,]/,
  29. isComplete: function(buffer, opts){
  30. return new RegExp(opts.regex).test(buffer.join(''));
  31. },
  32. definitions: {
  33. 'r': {
  34. validator: function (chrs, maskset, pos, strict, opts) {
  35. function regexToken(isGroup, isQuantifier) {
  36. this.matches = [];
  37. this.isGroup = isGroup || false;
  38. this.isQuantifier = isQuantifier || false;
  39. this.quantifier = { min: 1, max: 1 };
  40. this.repeaterPart = undefined;
  41. }
  42. function analyseRegex() {
  43. var currentToken = new regexToken(), match, m, opengroups = [];
  44. opts.regexTokens = [];
  45. // The tokenizer regex does most of the tokenization grunt work
  46. while (match = opts.tokenizer.exec(opts.regex)) {
  47. m = match[0];
  48. switch (m.charAt(0)) {
  49. case "(": // Group opening
  50. opengroups.push(new regexToken(true));
  51. break;
  52. case ")": // Group closing
  53. var groupToken = opengroups.pop();
  54. if (opengroups.length > 0) {
  55. opengroups[opengroups.length - 1]["matches"].push(groupToken);
  56. } else {
  57. currentToken.matches.push(groupToken);
  58. }
  59. break;
  60. case "{": case "+": case "*": //Quantifier
  61. var quantifierToken = new regexToken(false, true);
  62. m = m.replace(/[{}]/g, "");
  63. var mq = m.split(","), mq0 = isNaN(mq[0]) ? mq[0] : parseInt(mq[0]), mq1 = mq.length == 1 ? mq0 : (isNaN(mq[1]) ? mq[1] : parseInt(mq[1]));
  64. quantifierToken.quantifier = { min: mq0, max: mq1 };
  65. if (opengroups.length > 0) {
  66. var matches = opengroups[opengroups.length - 1]["matches"];
  67. match = matches.pop();
  68. if (!match["isGroup"]) {
  69. var groupToken = new regexToken(true);
  70. groupToken.matches.push(match);
  71. match = groupToken;
  72. }
  73. matches.push(match);
  74. matches.push(quantifierToken);
  75. } else {
  76. match = currentToken.matches.pop();
  77. if (!match["isGroup"]) {
  78. var groupToken = new regexToken(true);
  79. groupToken.matches.push(match);
  80. match = groupToken;
  81. }
  82. currentToken.matches.push(match);
  83. currentToken.matches.push(quantifierToken);
  84. }
  85. break;
  86. default:
  87. if (opengroups.length > 0) {
  88. opengroups[opengroups.length - 1]["matches"].push(m);
  89. } else {
  90. currentToken.matches.push(m);
  91. }
  92. break;
  93. }
  94. }
  95. if (currentToken.matches.length > 0)
  96. opts.regexTokens.push(currentToken);
  97. };
  98. function validateRegexToken(token, fromGroup) {
  99. var isvalid = false;
  100. if (fromGroup) {
  101. regexPart += "(";
  102. openGroupCount++;
  103. }
  104. for (var mndx = 0; mndx < token["matches"].length; mndx++) {
  105. var matchToken = token["matches"][mndx];
  106. if (matchToken["isGroup"] == true) {
  107. isvalid = validateRegexToken(matchToken, true);
  108. } else if (matchToken["isQuantifier"] == true) {
  109. var crrntndx = $.inArray(matchToken, token["matches"]),
  110. matchGroup = token["matches"][crrntndx - 1];
  111. var regexPartBak = regexPart;
  112. if (isNaN(matchToken.quantifier.max)) {
  113. while (matchToken["repeaterPart"] && matchToken["repeaterPart"] != regexPart && matchToken["repeaterPart"].length > regexPart.length) {
  114. isvalid = validateRegexToken(matchGroup, true);
  115. if (isvalid) break;
  116. }
  117. isvalid = isvalid || validateRegexToken(matchGroup, true);
  118. if (isvalid) matchToken["repeaterPart"] = regexPart;
  119. regexPart = regexPartBak + matchToken.quantifier.max;
  120. } else {
  121. for (var i = 0, qm = matchToken.quantifier.max - 1; i < qm; i++) {
  122. isvalid = validateRegexToken(matchGroup, true);
  123. if (isvalid) break;
  124. }
  125. regexPart = regexPartBak + "{" + matchToken.quantifier.min + "," + matchToken.quantifier.max + "}";
  126. }
  127. } else if (matchToken["matches"] != undefined) {
  128. for (var k = 0; k < matchToken.length; k++) {
  129. isvalid = validateRegexToken(matchToken[k], fromGroup);
  130. if (isvalid) break;
  131. }
  132. } else {
  133. var testExp;
  134. if (matchToken[0] == "[") {
  135. testExp = regexPart;
  136. testExp += matchToken;
  137. for (var j = 0; j < openGroupCount; j++) {
  138. testExp += ")";
  139. }
  140. var exp = new RegExp("^(" + testExp + ")$");
  141. isvalid = exp.test(bufferStr);
  142. } else {
  143. for (var l = 0, tl = matchToken.length; l < tl; l++) {
  144. if (matchToken[l] == "\\") continue;
  145. testExp = regexPart;
  146. testExp += matchToken.substr(0, l + 1);
  147. testExp = testExp.replace(/\|$/, "");
  148. for (var j = 0; j < openGroupCount; j++) {
  149. testExp += ")";
  150. }
  151. var exp = new RegExp("^(" + testExp + ")$");
  152. isvalid = exp.test(bufferStr);
  153. if (isvalid) break;
  154. }
  155. }
  156. regexPart += matchToken;
  157. }
  158. if (isvalid) break;
  159. }
  160. if (fromGroup) {
  161. regexPart += ")";
  162. openGroupCount--;
  163. }
  164. return isvalid;
  165. }
  166. if (opts.regexTokens == null) {
  167. analyseRegex();
  168. }
  169. var cbuffer = maskset.buffer.slice(), regexPart = "", isValid = false, openGroupCount = 0;
  170. cbuffer.splice(pos, 0, chrs);
  171. var bufferStr = cbuffer.join('');
  172. for (var i = 0; i < opts.regexTokens.length; i++) {
  173. var regexToken = opts.regexTokens[i];
  174. isValid = validateRegexToken(regexToken, regexToken["isGroup"]);
  175. if (isValid) break;
  176. }
  177. return isValid;
  178. },
  179. cardinality: 1
  180. }
  181. }
  182. }
  183. });
  184. }));