lib.validate.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482
  1. /**
  2. * 파일명: lib.validate.js
  3. * 설 명: 폼 체크, 값 표준화
  4. * 작성자: jstoy project
  5. * 날 짜: 2003-10-28
  6. * lainTT (2003-11-20) : FormChecker Class의 함수 prototype화 & 전역변수를 클래스 안으로...-_-;
  7. ***********************************************
  8. */
  9. /**
  10. * <pre>
  11. * 폼 체크 trigger 함수
  12. * </pre>
  13. *
  14. * @param Form Object
  15. * @return boolean
  16. */
  17. function validate(form) {
  18. var result;
  19. var checker = new FormChecker(form);
  20. result = checker.go();
  21. checker.destroy();
  22. return result;
  23. }
  24. function validate_init() {
  25. for (var i=0; i<document.forms.length; i++) {
  26. var formObj = document.forms[i];
  27. if (document.forms[i].getAttribute('VALIDATE') != null) {
  28. // pre_validate를 사용하지 않는다면 이 아랫줄을 주석처리합니다.
  29. new FormLoader(formObj);
  30. formObj.submitAction = formObj.onsubmit;
  31. formObj.onsubmit = function() {
  32. formObj.submitAction;
  33. return validate(this);
  34. }
  35. }
  36. }
  37. }
  38. FormChecker = function(form) {
  39. /**
  40. * <pre>
  41. * 미리 정의된 에러 메시지들
  42. * </pre>
  43. */
  44. /*
  45. this.FORM_ERROR_MSG = {
  46. //common : "입력하신 내용이 규칙에 어긋납니다.\n규칙에 어긋나는 내용을 바로잡아주세요.",
  47. common : "",
  48. required : "반드시 입력하셔야 합니다.",
  49. notequal : "서로 일치하지 않습니다.",
  50. invalid : "입력 형식에 어긋납니다.",
  51. denied : "업로드가 제한된 파일입니다.",
  52. minbyte : "길이가 {minbyte}Byte 이상이어야 합니다.",
  53. maxbyte : "길이가 {maxbyte}Byte를 초과할 수 없습니다."
  54. }
  55. */
  56. this.FORM_ERROR_MSG = {
  57. //common : "입력하신 내용이 규칙에 어긋납니다.\n규칙에 어긋나는 내용을 바로잡아주세요.",
  58. common : "",
  59. required : "입력하세요.",
  60. notequal : "서로 일치하지 않습니다.",
  61. invalid : "잘못 입력하였습니다.",
  62. denied : "업로드가 제한된 파일입니다.",
  63. minbyte : "{minbyte}자 이상 입력하세요.",
  64. maxbyte : "{maxbyte}자 이내로 입력하세요.",
  65. fixbyte : "{fixbyte}자 로 입력하세요."
  66. }
  67. this.FORM_ERROR_MSG_POSTPOSITION = {
  68. //common : "입력하신 내용이 규칙에 어긋납니다.\n규칙에 어긋나는 내용을 바로잡아주세요.",
  69. common : "",
  70. required : "을",
  71. notequal : "이",
  72. invalid : "을",
  73. denied : "은",
  74. minbyte : "은",
  75. maxbyte : "은",
  76. fixbyte : "은"
  77. }
  78. /**
  79. * <pre>
  80. * 폼 체크 함수 매핑
  81. * </pre>
  82. */
  83. this.VALIDATE_FUNCTION = {
  84. email : this.func_isValidEmail,
  85. phone : this.func_isValidPhone,
  86. userid : this.func_isValidUserid,
  87. hangul : this.func_hasHangul,
  88. number : this.func_isNumeric,
  89. money : this.func_isMoney,
  90. engonly : this.func_alphaOnly,
  91. jumin : this.func_isValidJumin,
  92. bizno : this.func_isValidBizNo
  93. }
  94. /**
  95. * <pre>
  96. * 에러 출력 플래그
  97. * </pre>
  98. */
  99. this.ERROR_MODE_FLAG = {
  100. all : 1, // 전체 에러를 표시
  101. one : 2, // 처음에 걸린 에러 하나만 표시
  102. one_per_obj : 3 // 한 object당 처음의 에러 표시
  103. }
  104. this.form = form;
  105. this.isErr = false;
  106. this.errMsg = (this.FORM_ERROR_MSG["common"] != "") ? this.FORM_ERROR_MSG["common"] + "\n\n" : "";
  107. this.errObj = "";
  108. this.curObj = "";
  109. this.errMode = this.ERROR_MODE_FLAG["one"]; // 에러메시지 출력모드
  110. }
  111. FormChecker.prototype.go = function() {
  112. for (var i = 0; i < this.form.elements.length; i++) {
  113. var el = this.form.elements[i];
  114. if (el.tagName.toLowerCase() == "fieldset" || el.tagName.toLowerCase() == "object")
  115. continue;
  116. /*웹표준대응 PDF계약관련*/
  117. SetElementValue(el);
  118. /*IE 10 대응 domhtml 상에 required의 문자 있으면 Y로 set해준다.*/
  119. var sTestHtml = el.outerHTML.toUpperCase();
  120. var nPos = sTestHtml.indexOf("REQUIRED");
  121. var nChar = sTestHtml.charAt(nPos-1);
  122. if(nPos>0 && nChar==' ')
  123. el.setAttribute("required","Y");
  124. if (el.getAttribute("HNAME") == null || el.getAttribute("HNAME") == "")
  125. el.setAttribute("HNAME", el.getAttribute("DESC"));
  126. var trim = el.getAttribute("TRIM");
  127. var minbyte = el.getAttribute("MINBYTE");
  128. var maxbyte = el.getAttribute("MAXBYTE");
  129. var fixbyte = el.getAttribute("FIXBYTE");
  130. var option = el.getAttribute("OPTION");
  131. var match = el.getAttribute("MATCH");
  132. var delim = el.getAttribute("DELIM");
  133. var glue = el.getAttribute("GLUE");
  134. var pattern = el.getAttribute("PATTERN");
  135. var allow = el.getAttribute("ALLOW");
  136. var deny = el.getAttribute("DENY");
  137. var func = el.getAttribute("FUNC");
  138. if (el.type == "text") {
  139. switch (trim) {
  140. case "ltrim": el.value = el.value.ltrim(); break;
  141. case "rtrim": el.value = el.value.rtrim(); break;
  142. case "notrim": break;
  143. default: el.value = el.value.trim(); break;
  144. }
  145. }
  146. if (el.getAttribute("REQUIRED") != null || el.getAttribute("CHK_REQUIRED") != null) {
  147. var bChkRequired = true;
  148. if(el.getAttribute("CHK_REQUIRED") != null) // 체크박스 선택에 따라 필수 체크하는 경우. 만약 체크 안되어 있으면 체크하지 않는다.
  149. bChkRequired = this.form.elements[el.getAttribute("CHK_REQUIRED")].checked;
  150. if(bChkRequired){
  151. switch (el.type) {
  152. case "file": case "text": case "textarea": case "password": case "hidden":
  153. if (el.value == null || el.value == "") this.addError(el,"required");
  154. break;
  155. case "select-one":
  156. if (el.options.length == 0 || el[el.selectedIndex].value == null || el[el.selectedIndex].value == "") this.addError(el,"required");
  157. break;
  158. case "radio":
  159. case "checkbox":
  160. // var elCheck = this.form.elements[el.name];
  161. var elCheck = document.getElementsByName(el.name);
  162. for (var j = 0, isChecked = false; j < elCheck.length; j++) {
  163. if (elCheck[j].checked == true) isChecked = true;
  164. }
  165. if (isChecked == false) this.addError(el,"required");
  166. break;
  167. //case "checkbox":
  168. // if (el.checked == false) this.addError(el,"required");
  169. // break;
  170. }
  171. }
  172. }
  173. if (el.type == "text" || el.type == "password") {
  174. if (match && (el.value != this.form.elements[match].value)) {
  175. this.addError(el,"notequal");
  176. }
  177. if (el.value && option != null) {
  178. if (glue != null) {
  179. var _value = new Array(el.value);
  180. var glue_arr = glue.split(",");
  181. for (var j = 0; j < glue_arr.length; j++) {
  182. _value[j+1] = this.form.elements[glue_arr[j]].value;
  183. }
  184. var value = _value.join(delim == null ? "" : delim);
  185. var tmp_msg = this.VALIDATE_FUNCTION[option](el, value);
  186. if (tmp_msg != true) this.addError(el,tmp_msg);
  187. } else {
  188. var tmp_msg = this.VALIDATE_FUNCTION[option](el);
  189. if (tmp_msg != true) this.addError(el,tmp_msg);
  190. }
  191. }
  192. if (el.value && minbyte != null && el.getAttribute("REQUIRED") != null) {
  193. if (el.value.bytes() < parseInt(minbyte)) this.addError(el,"minbyte");
  194. }
  195. if (el.value && maxbyte != null) {
  196. if (el.value.bytes() > parseInt(maxbyte)) this.addError(el,"maxbyte");
  197. }
  198. if (el.value && fixbyte != null && el.getAttribute("REQUIRED")) {
  199. if (el.value.bytes() != parseInt(fixbyte)) this.addError(el,"fixbyte");
  200. }
  201. if (pattern != null) {
  202. pattern = new RegExp(pattern);
  203. if (!pattern.test(el.value)) this.addError(el,'invalid');
  204. }
  205. }
  206. if (el.type == "file") {
  207. if (el.value && allow != null && allow != "") {
  208. pattern = new RegExp("(" + allow.replace(",", "|").toLowerCase() + ")$");
  209. if (!pattern.test(el.value.toLowerCase())) this.addError(el,'denied');
  210. }
  211. if (el.value && deny != null && deny != "") {
  212. pattern = new RegExp("(" + deny.replace(",", "|").toLowerCase() + ")$");
  213. if (pattern.test(el.value.toLowerCase())) this.addError(el,'denied');
  214. }
  215. }
  216. if(func) {
  217. if(this.isErr) continue;
  218. var result = eval(func + "()");
  219. if(result + "" == "undefined") {
  220. this.addError(el, "개발:[FUNC]속성 사용시 반드시 결과값(true/false)을 리턴해야 합니다.");
  221. }
  222. if(result === false) {
  223. return false;
  224. this.addError(el, "invalid");
  225. } else if(result !== true && result != "") {
  226. //this.errMsg = result;
  227. el.setAttribute("func_errmsg", result);
  228. this.addError(el, result);
  229. }
  230. }
  231. }
  232. return !this.isErr;
  233. }
  234. FormChecker.prototype.destroy = function() {
  235. if (this.isErr == true) {
  236. alert(this.errMsg);
  237. if (this.errObj.getAttribute("delete") != null)
  238. this.errObj.value = "";
  239. if (this.errObj.getAttribute("select") != null)
  240. this.errObj.select();
  241. if (this.errObj.getAttribute("errfunc") != null) // 물류 현대택배에서 사용. 필수입력 사항 없는 경우 탭이동하는 함수 호출.
  242. eval(this.errObj.getAttribute("errfunc"));
  243. if (this.errObj.getAttribute("nofocus") == null && this.errObj.type != "hidden")
  244. this.errObj.focus();
  245. }
  246. this.errMsg = "";
  247. this.errObj = "";
  248. }
  249. FormChecker.prototype.addError = function(el, type) {
  250. var pattern = /\{([a-zA-Z0-9_]+)\}/i;
  251. var msg = (this.FORM_ERROR_MSG[type]) ? this.FORM_ERROR_MSG[type] : type;
  252. var pp = this.FORM_ERROR_MSG_POSTPOSITION[type] ? this.FORM_ERROR_MSG_POSTPOSITION[type] : "은";
  253. if(type == "required") {
  254. if(el.type == "checkbox" || el.type == "radio" || el.type == "file" || el.type == "select-one") {
  255. msg = "선택해 주세요.";
  256. }
  257. }
  258. if (el.getAttribute("errmsg") != null) msg = el.getAttribute("errmsg");
  259. if (pattern.test(msg) == true) {
  260. while (pattern.exec(msg)) msg = msg.replace(pattern, el.getAttribute(RegExp.$1));
  261. }
  262. if (!this.errObj || this.errMode != this.ERROR_MODE_FLAG["one"]) {
  263. if (this.curObj == el.name && el.getAttribute("errmsg") == null) {
  264. if (this.errMode == this.ERROR_MODE_FLAG["all"]) {
  265. this.errMsg += " - "+ msg +"\n";
  266. }
  267. } else if (this.curObj != el.name) {
  268. if (this.curObj) {
  269. this.errMsg += "\n";
  270. }
  271. if(el.getAttribute("func_errmsg") != null) {
  272. this.errMsg += type;
  273. } else {
  274. if (el.getAttribute("errmsg") != null) {
  275. this.errMsg += el.getAttribute("errmsg");
  276. } else {
  277. // this.errMsg += "["+ el.getAttribute("hname") +"] 항목은 "+ msg +"\n";
  278. this.errMsg += postposition(el.getAttribute("hname"), pp) + " " + msg +"\n";
  279. }
  280. }
  281. //el.style.backgroundColor = "yellow";
  282. }
  283. }
  284. if (!this.errObj) this.errObj = el;
  285. this.curObj = el.name;
  286. this.isErr = true;
  287. return;
  288. }
  289. /// 패턴 검사 함수들 ///
  290. FormChecker.prototype.func_isValidEmail = function(el,value) {
  291. var value = value ? value : el.value;
  292. var pattern = /^[_a-zA-Z0-9-\.]+@[\.a-zA-Z0-9-]+\.[a-zA-Z]+$/;
  293. return (pattern.test(value)) ? true : "invalid";
  294. }
  295. FormChecker.prototype.func_isValidUserid = function(el) {
  296. var pattern = /^[a-zA-Z]{1}[a-zA-Z0-9_]{3,14}$/;
  297. return (pattern.test(el.value)) ? true : "4자이상 15자 미만,\n 영문,숫자, _ 문자 조합만 사용할 수 있습니다";
  298. }
  299. FormChecker.prototype.func_hasHangul = function(el) {
  300. var pattern = /[가-힝]/;
  301. // return (pattern.test(el.value)) ? true : "반드시 한글을 포함해야 합니다";
  302. return (pattern.test(el.value)) ? true : "한글을 포함해야 합니다";
  303. }
  304. FormChecker.prototype.func_alphaOnly = function(el) {
  305. var pattern = /^[a-zA-Z]+$/;
  306. return (pattern.test(el.value)) ? true : "invalid";
  307. }
  308. FormChecker.prototype.func_isNumeric = function(el) {
  309. var pattern = /^[0-9]+$/;
  310. // return (pattern.test(el.value)) ? true : "반드시 숫자로만 입력해야 합니다";
  311. return (pattern.test(el.value)) ? true : "숫자로만 입력해야 합니다";
  312. }
  313. FormChecker.prototype.func_isMoney = function(el) {
  314. var pattern = /^[0-9\,]+$/;
  315. // return (pattern.test(el.value)) ? true : "반드시 숫자로만 입력해야 합니다";
  316. return (pattern.test(el.value)) ? true : "숫자로만 입력해야 합니다";
  317. }
  318. FormChecker.prototype.func_isValidJumin = function(el,value) {
  319. var pattern = /^([0-9]{6})-?([0-9]{7})$/;
  320. var num = value ? value : el.value;
  321. if (!pattern.test(num)) return "invalid";
  322. num = RegExp.$1 + RegExp.$2;
  323. var sum = 0;
  324. var last = num.charCodeAt(12) - 0x30;
  325. var bases = "234567892345";
  326. for (var i=0; i<12; i++) {
  327. if (isNaN(num.substring(i,i+1))) return "invalid";
  328. sum += (num.charCodeAt(i) - 0x30) * (bases.charCodeAt(i) - 0x30);
  329. }
  330. var mod = sum % 11;
  331. return ((11 - mod) % 10 == last) ? true : "invalid";
  332. }
  333. FormChecker.prototype.func_isValidBizNo = function(el,value) {
  334. var pattern = /([0-9]{3})-?([0-9]{2})-?([0-9]{5})/;
  335. var num = value ? value : el.value;
  336. if (!pattern.test(num)) return "invalid";
  337. num = RegExp.$1 + RegExp.$2 + RegExp.$3;
  338. var cVal = 0;
  339. for (var i=0; i<8; i++) {
  340. var cKeyNum = parseInt(((_tmp = i % 3) == 0) ? 1 : ( _tmp == 1 ) ? 3 : 7);
  341. cVal += (parseFloat(num.substring(i,i+1)) * cKeyNum) % 10;
  342. }
  343. var li_temp = parseFloat(num.substring(i,i+1)) * 5 + "0";
  344. cVal += parseFloat(li_temp.substring(0,1)) + parseFloat(li_temp.substring(1,2));
  345. return (parseInt(num.substring(9,10)) == 10-(cVal % 10)%10) ? true : "invalid";
  346. }
  347. FormChecker.prototype.func_isValidPhone = function(el,value) {
  348. var pattern = /^([0]{1}[0-9]{1,2})-?([1-9]{1}[0-9]{2,3})-?([0-9]{4})$/;
  349. var num = value ? value : el.value;
  350. if (pattern.exec(num)) {
  351. if(RegExp.$1 == "010" || RegExp.$1 == "011" || RegExp.$1 == "016" || RegExp.$1 == "017" || RegExp.$1 == "018" || RegExp.$1 == "019") {
  352. if(!el.getAttribute("span"))
  353. el.value = RegExp.$1 + "-" + RegExp.$2 + "-" + RegExp.$3;
  354. }
  355. return true;
  356. } else {
  357. return "invalid";
  358. }
  359. }
  360. /**
  361. * common prototype functions
  362. */
  363. String.prototype.trim = function(str) {
  364. str = this != window ? this : str;
  365. return str.ltrim().rtrim();
  366. }
  367. String.prototype.ltrim = function(str) {
  368. str = this != window ? this : str;
  369. return str.replace(/^\s+/g,"");
  370. }
  371. String.prototype.rtrim = function(str) {
  372. str = this != window ? this : str;
  373. return str.replace(/\s+$/g,"");
  374. }
  375. String.prototype.bytes = function(str) {
  376. var len = 0;
  377. str = this != window ? this : str;
  378. for (j=0; j<str.length; j++) {
  379. var chr = str.charAt(j);
  380. len += (chr.charCodeAt() > 128) ? 2 : 1;
  381. }
  382. return len;
  383. }
  384. String.prototype.bytesCut = function(bytes) {
  385. var str = this;
  386. var len = 0;
  387. for (j=0; j<str.length; j++) {
  388. var chr = str.charAt(j);
  389. len += (chr.charCodeAt() > 128) ? 2 : 1;
  390. if (len > bytes) {
  391. str = str.substring(0, j);
  392. break;
  393. }
  394. }
  395. return str;
  396. }
  397. function autoNext(el, limit, next_el) {
  398. if(el.value.bytes() == 6) next_el.focus();
  399. }
  400. //출처:어떤놈의 블로그
  401. function postposition(txt, josa)
  402. {
  403. if(!txt) return "";
  404. var code = txt.charCodeAt(txt.length-1) - 44032;
  405. if (txt.length == 0) return '';
  406. if (code < 0 || code > 11171) return txt;
  407. if (code % 28 == 0) return txt + postposition.get(josa, false);
  408. else return txt + postposition.get(josa, true);
  409. }
  410. postposition.get = function (josa, jong) {
  411. // jong : true면 받침있음, false면 받침없음
  412. if (josa == '을' || josa == '를') return (jong?'을':'를');
  413. if (josa == '이' || josa == '가') return (jong?'이':'가');
  414. if (josa == '은' || josa == '는') return (jong?'은':'는');
  415. if (josa == '와' || josa == '과') return (jong?'와':'과');
  416. // 알 수 없는 조사
  417. return '**';
  418. }
  419. function SetElementValue(element) {
  420. if(!element) return false;
  421. switch(element.type) {
  422. case 'text':
  423. case 'password':
  424. case 'hidden':
  425. element.setAttribute("value",element.value);
  426. break;
  427. case 'textarea':
  428. //element.innerHTML = element.value;
  429. break;
  430. case 'checkbox':
  431. if(element.checked){ element.setAttribute("checked",true);}else{ element.removeAttribute("checked");}
  432. break;
  433. default:
  434. break;
  435. }
  436. }