yii.validation.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500
  1. /**
  2. * Yii validation module.
  3. *
  4. * This JavaScript module provides the validation methods for the built-in validators.
  5. *
  6. * @link https://www.yiiframework.com/
  7. * @copyright Copyright (c) 2008 Yii Software LLC
  8. * @license https://www.yiiframework.com/license/
  9. * @author Qiang Xue <qiang.xue@gmail.com>
  10. * @since 2.0
  11. */
  12. yii.validation = (function ($) {
  13. var pub = {
  14. isEmpty: function (value) {
  15. return value === null || value === undefined || ($.isArray(value) && value.length === 0) || value === '';
  16. },
  17. addMessage: function (messages, message, value) {
  18. messages.push(message.replace(/\{value\}/g, value));
  19. },
  20. required: function (value, messages, options) {
  21. var valid = false;
  22. if (options.requiredValue === undefined) {
  23. var isString = typeof value == 'string' || value instanceof String;
  24. if (options.strict && value !== undefined || !options.strict && !pub.isEmpty(isString ? trimString(value) : value)) {
  25. valid = true;
  26. }
  27. } else if (!options.strict && value == options.requiredValue || options.strict && value === options.requiredValue) {
  28. valid = true;
  29. }
  30. if (!valid) {
  31. pub.addMessage(messages, options.message, value);
  32. }
  33. },
  34. // "boolean" is a reserved keyword in older versions of ES so it's quoted for IE < 9 support
  35. 'boolean': function (value, messages, options) {
  36. if (options.skipOnEmpty && pub.isEmpty(value)) {
  37. return;
  38. }
  39. var valid = !options.strict && (value == options.trueValue || value == options.falseValue)
  40. || options.strict && (value === options.trueValue || value === options.falseValue);
  41. if (!valid) {
  42. pub.addMessage(messages, options.message, value);
  43. }
  44. },
  45. string: function (value, messages, options) {
  46. if (options.skipOnEmpty && pub.isEmpty(value)) {
  47. return;
  48. }
  49. if (typeof value !== 'string') {
  50. pub.addMessage(messages, options.message, value);
  51. return;
  52. }
  53. if (options.is !== undefined && value.length != options.is) {
  54. pub.addMessage(messages, options.notEqual, value);
  55. return;
  56. }
  57. if (options.min !== undefined && value.length < options.min) {
  58. pub.addMessage(messages, options.tooShort, value);
  59. }
  60. if (options.max !== undefined && value.length > options.max) {
  61. pub.addMessage(messages, options.tooLong, value);
  62. }
  63. },
  64. file: function (attribute, messages, options) {
  65. var files = getUploadedFiles(attribute, messages, options);
  66. $.each(files, function (i, file) {
  67. validateFile(file, messages, options);
  68. });
  69. },
  70. image: function (attribute, messages, options, deferredList) {
  71. var files = getUploadedFiles(attribute, messages, options);
  72. $.each(files, function (i, file) {
  73. validateFile(file, messages, options);
  74. // Skip image validation if FileReader API is not available
  75. if (typeof FileReader === "undefined") {
  76. return;
  77. }
  78. var deferred = $.Deferred();
  79. pub.validateImage(file, messages, options, deferred, new FileReader(), new Image());
  80. deferredList.push(deferred);
  81. });
  82. },
  83. validateImage: function (file, messages, options, deferred, fileReader, image) {
  84. image.onload = function() {
  85. validateImageSize(file, image, messages, options);
  86. deferred.resolve();
  87. };
  88. image.onerror = function () {
  89. messages.push(options.notImage.replace(/\{file\}/g, file.name));
  90. deferred.resolve();
  91. };
  92. fileReader.onload = function () {
  93. image.src = this.result;
  94. };
  95. // Resolve deferred if there was error while reading data
  96. fileReader.onerror = function () {
  97. deferred.resolve();
  98. };
  99. fileReader.readAsDataURL(file);
  100. },
  101. number: function (value, messages, options) {
  102. if (options.skipOnEmpty && pub.isEmpty(value)) {
  103. return;
  104. }
  105. if (typeof value === 'string' && !options.pattern.test(value)) {
  106. pub.addMessage(messages, options.message, value);
  107. return;
  108. }
  109. if (options.min !== undefined && value < options.min) {
  110. pub.addMessage(messages, options.tooSmall, value);
  111. }
  112. if (options.max !== undefined && value > options.max) {
  113. pub.addMessage(messages, options.tooBig, value);
  114. }
  115. },
  116. range: function (value, messages, options) {
  117. if (options.skipOnEmpty && pub.isEmpty(value)) {
  118. return;
  119. }
  120. if (!options.allowArray && $.isArray(value)) {
  121. pub.addMessage(messages, options.message, value);
  122. return;
  123. }
  124. var inArray = true;
  125. $.each($.isArray(value) ? value : [value], function (i, v) {
  126. if ($.inArray(v, options.range) == -1) {
  127. inArray = false;
  128. return false;
  129. } else {
  130. return true;
  131. }
  132. });
  133. if (options.not === undefined) {
  134. options.not = false;
  135. }
  136. if (options.not === inArray) {
  137. pub.addMessage(messages, options.message, value);
  138. }
  139. },
  140. regularExpression: function (value, messages, options) {
  141. if (options.skipOnEmpty && pub.isEmpty(value)) {
  142. return;
  143. }
  144. if (!options.not && !options.pattern.test(value) || options.not && options.pattern.test(value)) {
  145. pub.addMessage(messages, options.message, value);
  146. }
  147. },
  148. email: function (value, messages, options) {
  149. if (options.skipOnEmpty && pub.isEmpty(value)) {
  150. return;
  151. }
  152. var valid = true,
  153. regexp = /^((?:"?([^"]*)"?\s)?)(?:\s+)?(?:(<?)((.+)@([^>]+))(>?))$/,
  154. matches = regexp.exec(value);
  155. if (matches === null) {
  156. valid = false;
  157. } else {
  158. var localPart = matches[5],
  159. domain = matches[6];
  160. if (options.enableIDN) {
  161. localPart = punycode.toASCII(localPart);
  162. domain = punycode.toASCII(domain);
  163. value = matches[1] + matches[3] + localPart + '@' + domain + matches[7];
  164. }
  165. if (localPart.length > 64) {
  166. valid = false;
  167. } else if ((localPart + '@' + domain).length > 254) {
  168. valid = false;
  169. } else {
  170. valid = options.pattern.test(value) || (options.allowName && options.fullPattern.test(value));
  171. }
  172. }
  173. if (!valid) {
  174. pub.addMessage(messages, options.message, value);
  175. }
  176. },
  177. url: function (value, messages, options) {
  178. if (options.skipOnEmpty && pub.isEmpty(value)) {
  179. return;
  180. }
  181. if (options.defaultScheme && !/:\/\//.test(value)) {
  182. value = options.defaultScheme + '://' + value;
  183. }
  184. var valid = true;
  185. if (options.enableIDN) {
  186. var matches = /^([^:]+):\/\/([^\/]+)(.*)$/.exec(value);
  187. if (matches === null) {
  188. valid = false;
  189. } else {
  190. value = matches[1] + '://' + punycode.toASCII(matches[2]) + matches[3];
  191. }
  192. }
  193. if (!valid || !options.pattern.test(value)) {
  194. pub.addMessage(messages, options.message, value);
  195. }
  196. },
  197. trim: function ($form, attribute, options, value) {
  198. var $input = $form.find(attribute.input);
  199. if ($input.is(':checkbox, :radio')) {
  200. return value;
  201. }
  202. value = $input.val();
  203. if (
  204. (!options.skipOnEmpty || !pub.isEmpty(value))
  205. && (!options.skipOnArray || !Array.isArray(value))
  206. ) {
  207. if (Array.isArray(value)) {
  208. for (var i = 0; i < value.length; i++) {
  209. value[i] = trimString(value[i], options);
  210. }
  211. } else {
  212. value = trimString(value, options);
  213. }
  214. $input.val(value);
  215. }
  216. return value;
  217. },
  218. captcha: function (value, messages, options) {
  219. if (options.skipOnEmpty && pub.isEmpty(value)) {
  220. return;
  221. }
  222. // CAPTCHA may be updated via AJAX and the updated hash is stored in body data
  223. var hash = $('body').data(options.hashKey);
  224. hash = hash == null ? options.hash : hash[options.caseSensitive ? 0 : 1];
  225. var v = options.caseSensitive ? value : value.toLowerCase();
  226. for (var i = v.length - 1, h = 0; i >= 0; --i) {
  227. h += v.charCodeAt(i) << i;
  228. }
  229. if (h != hash) {
  230. pub.addMessage(messages, options.message, value);
  231. }
  232. },
  233. compare: function (value, messages, options, $form) {
  234. if (options.skipOnEmpty && pub.isEmpty(value)) {
  235. return;
  236. }
  237. var compareValue,
  238. valid = true;
  239. if (options.compareAttribute === undefined) {
  240. compareValue = options.compareValue;
  241. } else {
  242. var $target = $('#' + options.compareAttribute);
  243. if (!$target.length) {
  244. $target = $form.find('[name="' + options.compareAttributeName + '"]');
  245. }
  246. compareValue = $target.val();
  247. }
  248. if (options.type === 'number') {
  249. value = value ? parseFloat(value) : 0;
  250. compareValue = compareValue ? parseFloat(compareValue) : 0;
  251. }
  252. switch (options.operator) {
  253. case '==':
  254. valid = value == compareValue;
  255. break;
  256. case '===':
  257. valid = value === compareValue;
  258. break;
  259. case '!=':
  260. valid = value != compareValue;
  261. break;
  262. case '!==':
  263. valid = value !== compareValue;
  264. break;
  265. case '>':
  266. valid = value > compareValue;
  267. break;
  268. case '>=':
  269. valid = value >= compareValue;
  270. break;
  271. case '<':
  272. valid = value < compareValue;
  273. break;
  274. case '<=':
  275. valid = value <= compareValue;
  276. break;
  277. default:
  278. valid = false;
  279. break;
  280. }
  281. if (!valid) {
  282. pub.addMessage(messages, options.message, value);
  283. }
  284. },
  285. ip: function (value, messages, options) {
  286. if (options.skipOnEmpty && pub.isEmpty(value)) {
  287. return;
  288. }
  289. var negation = null,
  290. cidr = null,
  291. matches = new RegExp(options.ipParsePattern).exec(value);
  292. if (matches) {
  293. negation = matches[1] || null;
  294. value = matches[2];
  295. cidr = matches[4] || null;
  296. }
  297. if (options.subnet === true && cidr === null) {
  298. pub.addMessage(messages, options.messages.noSubnet, value);
  299. return;
  300. }
  301. if (options.subnet === false && cidr !== null) {
  302. pub.addMessage(messages, options.messages.hasSubnet, value);
  303. return;
  304. }
  305. if (options.negation === false && negation !== null) {
  306. pub.addMessage(messages, options.messages.message, value);
  307. return;
  308. }
  309. var ipVersion = value.indexOf(':') === -1 ? 4 : 6;
  310. if (ipVersion == 6) {
  311. if (!(new RegExp(options.ipv6Pattern)).test(value)) {
  312. pub.addMessage(messages, options.messages.message, value);
  313. }
  314. if (!options.ipv6) {
  315. pub.addMessage(messages, options.messages.ipv6NotAllowed, value);
  316. }
  317. } else {
  318. if (!(new RegExp(options.ipv4Pattern)).test(value)) {
  319. pub.addMessage(messages, options.messages.message, value);
  320. }
  321. if (!options.ipv4) {
  322. pub.addMessage(messages, options.messages.ipv4NotAllowed, value);
  323. }
  324. }
  325. }
  326. };
  327. function getUploadedFiles(attribute, messages, options) {
  328. // Skip validation if File API is not available
  329. if (typeof File === "undefined") {
  330. return [];
  331. }
  332. var fileInput = $(attribute.input, attribute.$form).get(0);
  333. // Skip validation if file input does not exist
  334. // (in case file inputs are added dynamically and no file input has been added to the form)
  335. if (typeof fileInput === "undefined") {
  336. return [];
  337. }
  338. var files = fileInput.files;
  339. if (!files) {
  340. messages.push(options.message);
  341. return [];
  342. }
  343. if (files.length === 0) {
  344. if (!options.skipOnEmpty) {
  345. messages.push(options.uploadRequired);
  346. }
  347. return [];
  348. }
  349. if (options.maxFiles && options.maxFiles < files.length) {
  350. messages.push(options.tooMany);
  351. return [];
  352. }
  353. return files;
  354. }
  355. function validateFile(file, messages, options) {
  356. if (options.extensions && options.extensions.length > 0) {
  357. var found = false;
  358. var filename = file.name.toLowerCase();
  359. for (var index=0; index < options.extensions.length; index++) {
  360. var ext = options.extensions[index].toLowerCase();
  361. if ((ext === '' && filename.indexOf('.') === -1) || (filename.substr(filename.length - options.extensions[index].length - 1) === ('.' + ext))) {
  362. found = true;
  363. break;
  364. }
  365. }
  366. if (!found) {
  367. messages.push(options.wrongExtension.replace(/\{file\}/g, file.name));
  368. }
  369. }
  370. if (options.mimeTypes && options.mimeTypes.length > 0) {
  371. if (!validateMimeType(options.mimeTypes, file.type)) {
  372. messages.push(options.wrongMimeType.replace(/\{file\}/g, file.name));
  373. }
  374. }
  375. if (options.maxSize && options.maxSize < file.size) {
  376. messages.push(options.tooBig.replace(/\{file\}/g, file.name));
  377. }
  378. if (options.minSize && options.minSize > file.size) {
  379. messages.push(options.tooSmall.replace(/\{file\}/g, file.name));
  380. }
  381. }
  382. function validateMimeType(mimeTypes, fileType) {
  383. for (var i = 0, len = mimeTypes.length; i < len; i++) {
  384. if (new RegExp(mimeTypes[i]).test(fileType)) {
  385. return true;
  386. }
  387. }
  388. return false;
  389. }
  390. function validateImageSize(file, image, messages, options) {
  391. if (options.minWidth && image.width < options.minWidth) {
  392. messages.push(options.underWidth.replace(/\{file\}/g, file.name));
  393. }
  394. if (options.maxWidth && image.width > options.maxWidth) {
  395. messages.push(options.overWidth.replace(/\{file\}/g, file.name));
  396. }
  397. if (options.minHeight && image.height < options.minHeight) {
  398. messages.push(options.underHeight.replace(/\{file\}/g, file.name));
  399. }
  400. if (options.maxHeight && image.height > options.maxHeight) {
  401. messages.push(options.overHeight.replace(/\{file\}/g, file.name));
  402. }
  403. }
  404. /**
  405. * PHP: `trim($path, ' /')`, JS: `yii.helpers.trim(path, {chars: ' /'})`
  406. */
  407. function trimString(value, options = {skipOnEmpty: true, chars: null}) {
  408. if (options.skipOnEmpty !== false && pub.isEmpty(value)) {
  409. return value;
  410. }
  411. value = new String(value);
  412. if (options.chars || !String.prototype.trim) {
  413. var chars = !options.chars
  414. ? ' \\s\xA0'
  415. : options.chars.replace(/([\[\]\(\)\.\?\/\*\{\}\+\$\^\:])/g, '\$1');
  416. return value.replace(new RegExp('^[' + chars + ']+|[' + chars + ']+$', 'g'), '');
  417. }
  418. return value.trim();
  419. }
  420. return pub;
  421. })(jQuery);