jquery.idealforms.js 90 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368
  1. (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
  2. /**
  3. * Errors
  4. */
  5. module.exports = {
  6. required: 'This field is required',
  7. digits: 'Must be only digits',
  8. name: 'Must be at least 3 characters long and must only contain letters',
  9. email: 'Must be a valid email',
  10. username: 'Must be at between 4 and 32 characters long and start with a letter. You may use letters, numbers, underscores, and one dot',
  11. pass: 'Must be at least 6 characters long, and contain at least one number, one uppercase and one lowercase letter',
  12. strongpass: 'Must be at least 8 characters long and contain at least one uppercase and one lowercase letter and one number or special character',
  13. phone: 'Must be a valid phone number',
  14. zip: 'Must be a valid zip code',
  15. url: 'Must be a valid URL',
  16. number: 'Must be a number',
  17. range: 'Must be a number between {0} and {1}',
  18. min: 'Must be at least {0} characters long',
  19. max: 'Must be under {0} characters',
  20. minoption: 'Select at least {0} options',
  21. maxoption: 'Select no more than {0} options',
  22. minmax: 'Must be between {0} and {1} characters long',
  23. select: 'Select an option',
  24. extension: 'File(s) must have a valid extension ({*})',
  25. equalto: 'Must have the same value as the "{0}" field',
  26. date: 'Must be a valid date {0}'
  27. };
  28. },{}],2:[function(require,module,exports){
  29. /**
  30. * Adaptive
  31. */
  32. module.exports = {
  33. name: 'adaptive',
  34. methods: {
  35. // @extend
  36. _init: function () {
  37. var self = this;
  38. var $dummy = $('<p class="idealforms-field-width"/>').appendTo('body');
  39. this.opts.adaptiveWidth = $dummy.css('width').replace('px','');
  40. function adapt() {
  41. var formWidth = self.$form.outerWidth()
  42. , isAdaptive = self.opts.adaptiveWidth > formWidth;
  43. self.$form.toggleClass('adaptive', isAdaptive);
  44. if (self._hasExtension('steps')) {
  45. self.$stepsContainer.toggleClass('adaptive', isAdaptive);
  46. }
  47. $('#ui-datepicker-div').hide();
  48. }
  49. $(window).resize(adapt);
  50. adapt();
  51. this.$form.find('select, .datepicker').each(function() {
  52. self._getField(this).find(self.opts.error).addClass('hidden');
  53. });
  54. $dummy.remove();
  55. }
  56. }
  57. };
  58. },{}],3:[function(require,module,exports){
  59. module.exports = {
  60. name: 'ajax',
  61. methods: {
  62. // @extend
  63. _init: function() {
  64. $.extend($.idealforms, { _requests: {} });
  65. $.idealforms.errors.ajax = $.idealforms.errors.ajax || 'Loading...';
  66. $.extend($.idealforms.rules, {
  67. ajax: function(input) {
  68. var self = this
  69. , $field = this._getField(input)
  70. , url = $(input).data('idealforms-ajax')
  71. , userError = $.idealforms._getKey('errors.'+ input.name +'.ajaxError', self.opts)
  72. , requests = $.idealforms._requests
  73. , data = {};
  74. data[input.name] = input.value;
  75. $field.addClass('ajax');
  76. if (requests[input.name]) requests[input.name].abort();
  77. requests[input.name] = $.post(url, data, function(resp) {
  78. if (resp === true) {
  79. $field.data('idealforms-valid', true);
  80. self._handleError(input);
  81. self._handleStyle(input);
  82. } else {
  83. self._handleError(input, userError);
  84. }
  85. self.opts.onValidate.call(self, input, 'ajax', resp);
  86. $field.removeClass('ajax');
  87. }, 'json');
  88. return false;
  89. }
  90. });
  91. },
  92. // @extend
  93. _validate: function(input, rule) {
  94. if (rule != 'ajax' && $.idealforms._requests[input.name]) {
  95. $.idealforms._requests[input.name].abort();
  96. this._getField(input).removeClass('ajax');
  97. }
  98. }
  99. }
  100. };
  101. },{}],4:[function(require,module,exports){
  102. require('./idealfile');
  103. require('./idealradiocheck');
  104. module.exports = {
  105. name: 'customInputs',
  106. options: {
  107. customInputs: {
  108. i18n: {
  109. open: 'Open'
  110. }
  111. }
  112. },
  113. methods: {
  114. // @extend
  115. _init: function() {
  116. this._buildCustomInputs();
  117. },
  118. // @extend
  119. 'addFields:after': function() {
  120. this._buildCustomInputs();
  121. },
  122. _buildCustomInputs: function() {
  123. this.$form.find(':file').idealfile(this.opts.customInputs.i18n);
  124. this.$form.find(':checkbox, :radio').idealradiocheck();
  125. }
  126. }
  127. };
  128. },{"./idealfile":5,"./idealradiocheck":6}],5:[function(require,module,exports){
  129. /**
  130. * Ideal File
  131. */
  132. (function($, win, doc, undefined) {
  133. // Browser supports HTML5 multiple file?
  134. var multipleSupport = typeof $('<input/>')[0].multiple !== 'undefined'
  135. , isIE = /msie/i.test(navigator.userAgent)
  136. , plugin = {};
  137. plugin.name = 'idealfile';
  138. plugin.defaults = {
  139. open: 'Open'
  140. };
  141. plugin.methods = {
  142. _init: function() {
  143. var $file = $(this.el).addClass('ideal-file') // the original file input
  144. , $wrap = $('<div class="ideal-file-wrap">')
  145. , $input = $('<input type="text" class="ideal-file-filename" />')
  146. // Button that will be used in non-IE browsers
  147. , $button = $('<button type="button" class="ideal-file-upload">'+ this.opts.open +'</button>')
  148. // Hack for IE
  149. , $label = $('<label class="ideal-file-upload" for="' + $file[0].id + '">'+ this.opts.open +'</label>');
  150. if (isIE) $label.add($button).addClass('ie');
  151. // Hide by shifting to the left so we
  152. // can still trigger events
  153. $file.css({
  154. position: 'absolute',
  155. left: '-9999px'
  156. });
  157. $wrap.append($input, (isIE ? $label : $button)).insertAfter($file);
  158. // Prevent focus
  159. $file.attr('tabIndex', -1);
  160. $button.attr('tabIndex', -1);
  161. $button.click(function () {
  162. $file.focus().click(); // Open dialog
  163. });
  164. $file.change(function () {
  165. var files = []
  166. , fileArr, filename;
  167. // If multiple is supported then extract
  168. // all filenames from the file array
  169. if (multipleSupport) {
  170. fileArr = $file[0].files;
  171. for (var i = 0, len = fileArr.length; i < len; i++) {
  172. files.push(fileArr[i].name);
  173. }
  174. filename = files.join(', ');
  175. // If not supported then just take the value
  176. // and remove the path to just show the filename
  177. } else {
  178. filename = $file.val().split('\\').pop();
  179. }
  180. $input .val(filename).attr('title', filename);
  181. });
  182. $input.on({
  183. blur: function () {
  184. $file.trigger('blur');
  185. },
  186. keydown: function (e) {
  187. if (e.which === 13) { // Enter
  188. if (!isIE) $file.trigger('click');
  189. $(this).closest('form').one('keydown', function(e) {
  190. if (e.which === 13) e.preventDefault();
  191. });
  192. } else if (e.which === 8 || e.which === 46) { // Backspace & Del
  193. // In IE the value is read-only
  194. // with this trick we remove the old input and add
  195. // a clean clone with all the original events attached
  196. if (isIE) $file.replaceWith($file = $file.clone(true));
  197. $file.val('').trigger('change');
  198. $input.val('');
  199. } else if (e.which === 9) { // TAB
  200. return;
  201. } else { // All other keys
  202. return false;
  203. }
  204. }
  205. });
  206. }
  207. };
  208. require('../../plugin')(plugin);
  209. }(jQuery, window, document));
  210. },{"../../plugin":12}],6:[function(require,module,exports){
  211. /*
  212. * idealRadioCheck: jQuery plguin for checkbox and radio replacement
  213. * Usage: $('input[type=checkbox], input[type=radio]').idealRadioCheck()
  214. */
  215. (function($, win, doc, undefined) {
  216. var plugin = {};
  217. plugin.name = 'idealradiocheck';
  218. plugin.methods = {
  219. _init: function() {
  220. var $input = $(this.el);
  221. var $span = $('<span/>');
  222. $span.addClass('ideal-'+ ($input.is(':checkbox') ? 'check' : 'radio'));
  223. $input.is(':checked') && $span.addClass('checked'); // init
  224. $span.insertAfter($input);
  225. $input.parent('label')
  226. .addClass('ideal-radiocheck-label')
  227. .attr('onclick', ''); // Fix clicking label in iOS
  228. $input.css({ position: 'absolute', left: '-9999px' }); // hide by shifting left
  229. // Events
  230. $input.on({
  231. change: function() {
  232. var $input = $(this);
  233. if ( $input.is('input[type="radio"]') ) {
  234. $input.parent().siblings('label').find('.ideal-radio').removeClass('checked');
  235. }
  236. $span.toggleClass('checked', $input.is(':checked'));
  237. },
  238. focus: function() { $span.addClass('focus') },
  239. blur: function() { $span.removeClass('focus') },
  240. click: function() { $(this).trigger('focus') }
  241. });
  242. }
  243. };
  244. require('../../plugin')(plugin);
  245. }(jQuery, window, document));
  246. },{"../../plugin":12}],7:[function(require,module,exports){
  247. module.exports = {
  248. name: 'datepicker',
  249. methods: {
  250. // @extend
  251. _init: function() {
  252. this._buildDatepicker();
  253. },
  254. _buildDatepicker: function() {
  255. var $datepicker = this.$form.find('input.datepicker');
  256. // Always show datepicker below the input
  257. if (jQuery.ui) {
  258. $.datepicker._checkOffset = function(a,b,c){ return b };
  259. }
  260. if (jQuery.ui && $datepicker.length) {
  261. $datepicker.each(function() {
  262. $(this).datepicker({
  263. beforeShow: function(input) {
  264. $(input).addClass('open');
  265. },
  266. onChangeMonthYear: function() {
  267. // Hack to fix IE9 not resizing
  268. var $this = $(this)
  269. , width = $this.outerWidth(); // cache first!
  270. setTimeout(function() {
  271. $this.datepicker('widget').css('width', width);
  272. }, 1);
  273. },
  274. onClose: function() {
  275. $(this).removeClass('open');
  276. }
  277. });
  278. });
  279. // Adjust width
  280. $datepicker.on('focus keyup', function() {
  281. var t = $(this), w = t.outerWidth();
  282. t.datepicker('widget').css('width', w);
  283. });
  284. }
  285. }
  286. }
  287. };
  288. },{}],8:[function(require,module,exports){
  289. function template(html, data) {
  290. var loop = /\{@([^}]+)\}(.+?)\{\/\1\}/g
  291. , loopVariable = /\{#([^}]+)\}/g
  292. , variable = /\{([^}]+)\}/g;
  293. return html
  294. .replace(loop, function(_, key, list) {
  295. return $.map(data[key], function(item) {
  296. return list.replace(loopVariable, function(_, k) {
  297. return item[k];
  298. });
  299. }).join('');
  300. })
  301. .replace(variable, function(_, key) {
  302. return data[key] || '';
  303. });
  304. }
  305. module.exports = {
  306. name: 'dynamicFields',
  307. options: {
  308. templates: {
  309. base:'\
  310. <div class="field">\
  311. <label class="main">{label}</label>\
  312. {field}\
  313. <span class="error"></span>\
  314. </div>\
  315. ',
  316. text: '<input name="{name}" type="{subtype}" value="{value}" {attrs}>',
  317. file: '<input id="{name}" name="{name}" type="file" {attrs}>',
  318. textarea: '<textarea name="{name}" {attrs}>{text}</textarea>',
  319. group: '\
  320. <p class="group">\
  321. {@list}\
  322. <label><input name="{name}" type="{subtype}" value="{#value}" {#attrs}>{#text}</label>\
  323. {/list}\
  324. </p>\
  325. ',
  326. select: '\
  327. <select name={name}>\
  328. {@list}\
  329. <option value="{#value}">{#text}</option>\
  330. {/list}\
  331. </select>\
  332. '
  333. }
  334. },
  335. methods: {
  336. addFields: function(fields) {
  337. var self = this;
  338. $.each(fields, function(name, field) {
  339. var typeArray = field.type.split(':')
  340. , rules = {}
  341. , $last = self.$form.find(self.opts.field).last();
  342. field.name = name;
  343. field.type = typeArray[0];
  344. if (typeArray[1]) field.subtype = typeArray[1];
  345. field.html = template(self.opts.templates.base, {
  346. label: field.label,
  347. field: template(self.opts.templates[field.type], field)
  348. });
  349. self._inject('addFields:before', field);
  350. if (field.after || field.before) {
  351. self.$form.find('[name="'+ (field.after || field.before) +'"]').first().each(function() {
  352. self._getField(this)[field.after ? 'after' : 'before'](field.html);
  353. });
  354. } else {
  355. // Form has at least one field
  356. if ($last.length) $last.after(field.html);
  357. // Form has no fields
  358. else self.$form.append(field.html);
  359. }
  360. if (field.rules) {
  361. rules[name] = field.rules;
  362. self.addRules(rules);
  363. }
  364. self._inject('addFields:after', field);
  365. });
  366. },
  367. removeFields: function(names) {
  368. var self = this;
  369. $.each(names.split(' '), function(i, name) {
  370. var $field = self._getField($('[name="'+ name +'"]'));
  371. self.$fields = self.$fields.filter(function() {
  372. return ! $(this).is($field);
  373. });
  374. $field.remove();
  375. });
  376. this._inject('removeFields');
  377. },
  378. toggleFields: function(names) {
  379. var self = this;
  380. $.each(names.split(' '), function(i, name) {
  381. var $field = self._getField($('[name="'+ name +'"]'));
  382. $field.data('idealforms-valid', $field.is(':visible')).toggle();
  383. });
  384. this._inject('toggleFields');
  385. }
  386. }
  387. };
  388. },{}],9:[function(require,module,exports){
  389. /*!
  390. * Ideal Steps
  391. */
  392. (function($, win, doc, undefined) {
  393. var plugin = {};
  394. plugin.name = 'idealsteps';
  395. plugin.defaults = {
  396. nav: '.idealsteps-nav',
  397. navItems: 'li',
  398. buildNavItems: true,
  399. wrap: '.idealsteps-wrap',
  400. step: '.idealsteps-step',
  401. activeClass: 'idealsteps-step-active',
  402. before: $.noop,
  403. after: $.noop,
  404. fadeSpeed: 0
  405. };
  406. plugin.methods = {
  407. _init: function() {
  408. var self = this,
  409. active = this.opts.activeClass;
  410. this.$el = $(this.el);
  411. this.$nav = this.$el.find(this.opts.nav);
  412. this.$navItems = this.$nav.find(this.opts.navItems);
  413. this.$wrap = this.$el.find(this.opts.wrap);
  414. this.$steps = this.$wrap.find(this.opts.step);
  415. if (this.opts.buildNavItems) this._buildNavItems();
  416. this.$steps.hide().first().show();
  417. this.$navItems.removeClass(active).first().addClass(active);
  418. this.$navItems.click(function(e) {
  419. e.preventDefault();
  420. if (! $(this).is('.'+ self.opts.activeClass)) {
  421. self.go(self.$navItems.index(this));
  422. }
  423. });
  424. },
  425. _buildNavItems: function() {
  426. var self = this,
  427. isCustom = typeof this.opts.buildNavItems == 'function',
  428. item = function(val){ return '<li><a href="#" tabindex="-1">'+ val +'</a></li>'; },
  429. items;
  430. items = isCustom ?
  431. this.$steps.map(function(i){ return item(self.opts.buildNavItems.call(self, i)) }).get() :
  432. this.$steps.map(function(i){ return item(++i); }).get();
  433. this.$navItems = $(items.join(''));
  434. this.$nav.append($('<ul/>').append(this.$navItems));
  435. },
  436. _getCurIdx: function() {
  437. return this.$steps.index(this.$steps.filter(':visible'));
  438. },
  439. go: function(idx) {
  440. var active = this.opts.activeClass,
  441. fadeSpeed = this.opts.fadeSpeed;
  442. if (typeof idx == 'function') idx = idx.call(this, this._getCurIdx());
  443. if (idx >= this.$steps.length) idx = 0;
  444. if (idx < 0) idx = this.$steps.length-1;
  445. this.opts.before.call(this, idx);
  446. this.$navItems.removeClass(active).eq(idx).addClass(active);
  447. this.$steps.hide().eq(idx).fadeIn(fadeSpeed);
  448. this.opts.after.call(this, idx);
  449. },
  450. prev: function() {
  451. this.go(this._getCurIdx() - 1);
  452. },
  453. next: function() {
  454. this.go(this._getCurIdx() + 1);
  455. },
  456. first: function() {
  457. this.go(0);
  458. },
  459. last: function() {
  460. this.go(this.$steps.length-1);
  461. }
  462. };
  463. require('../../plugin')(plugin);
  464. }(jQuery, window, document));
  465. },{"../../plugin":12}],10:[function(require,module,exports){
  466. require('./idealsteps');
  467. module.exports = {
  468. name: 'steps',
  469. options: {
  470. steps: {
  471. container: '.idealsteps-container',
  472. nav: '.idealsteps-nav',
  473. navItems: 'li',
  474. buildNavItems: function(i) {
  475. return this.opts.steps.i18n.step +' '+ (i+1);
  476. },
  477. wrap: '.idealsteps-wrap',
  478. step: '.idealsteps-step',
  479. activeClass: 'idealsteps-step-active',
  480. before: $.noop,
  481. after: $.noop,
  482. fadeSpeed: 0,
  483. i18n: {
  484. step: 'Step'
  485. }
  486. }
  487. },
  488. methods: {
  489. // @extend
  490. _init: function() {
  491. this._buildSteps();
  492. },
  493. // @extend
  494. _validate: function() {
  495. var self = this;
  496. this._updateSteps();
  497. if (this._hasExtension('ajax')) {
  498. $.each($.idealforms._requests, function(key, request) {
  499. request.done(function(){ self._updateSteps() });
  500. });
  501. }
  502. },
  503. // @extend
  504. focusFirstInvalid: function(firstInvalid) {
  505. var self = this;
  506. this.$stepsContainer.idealsteps('go', function() {
  507. return this.$steps.filter(function() {
  508. return $(this).find(firstInvalid).length;
  509. }).index();
  510. });
  511. setTimeout(function(){ $(firstInvalid).focus() }, this.opts.steps.fadeSpeed);
  512. },
  513. // @extend
  514. addRules: function() {
  515. this.firstStep();
  516. },
  517. // @extend
  518. 'addFields:before': function(field) {
  519. if (field.after || field.before) return;
  520. var $steps = this.$stepsContainer.find(this.opts.steps.step);
  521. if (! ('appendToStep' in field)) {
  522. field.appendToStep = $steps.length-1;
  523. }
  524. field.after = $steps
  525. .eq(field.appendToStep)
  526. .find('input, select, textarea')
  527. .last()[0].name;
  528. },
  529. // @extend
  530. toggleFields: function() {
  531. this._updateSteps();
  532. },
  533. // @extend
  534. removeFields: function() {
  535. this._updateSteps();
  536. },
  537. _buildSteps: function() {
  538. var self = this, options
  539. , hasRules = ! $.isEmptyObject(this.opts.rules)
  540. , buildNavItems = this.opts.steps.buildNavItems
  541. , counter = hasRules
  542. ? '<span class="counter"/>'
  543. : '<span class="counter zero">0</span>';
  544. if (this.opts.steps.buildNavItems) {
  545. this.opts.steps.buildNavItems = function(i) {
  546. return buildNavItems.call(self, i) + counter;
  547. };
  548. }
  549. this.$stepsContainer = this.$form
  550. .closest(this.opts.steps.container)
  551. .idealsteps(this.opts.steps);
  552. },
  553. _updateSteps: function() {
  554. var self = this;
  555. this.$stepsContainer.idealsteps('_inject', function() {
  556. var idealsteps = this;
  557. this.$navItems.each(function(i) {
  558. var invalid = idealsteps.$steps.eq(i).find(self.getInvalid()).length;
  559. $(this).find('span').text(invalid).toggleClass('zero', ! invalid);
  560. });
  561. });
  562. },
  563. goToStep: function(idx) {
  564. this.$stepsContainer.idealsteps('go', idx);
  565. },
  566. prevStep: function() {
  567. this.$stepsContainer.idealsteps('prev');
  568. },
  569. nextStep: function() {
  570. this.$stepsContainer.idealsteps('next');
  571. },
  572. firstStep: function() {
  573. this.$stepsContainer.idealsteps('first');
  574. },
  575. lastStep: function() {
  576. this.$stepsContainer.idealsteps('last');
  577. }
  578. }
  579. };
  580. },{"./idealsteps":9}],11:[function(require,module,exports){
  581. /*!
  582. * jQuery Ideal Forms
  583. * @author: Cedric Ruiz
  584. * @version: 3.0
  585. * @license GPL or MIT
  586. */
  587. (function($, win, doc, undefined) {
  588. var plugin = {};
  589. plugin.name = 'idealforms';
  590. plugin.defaults = {
  591. field: '.field',
  592. error: '.error',
  593. iconHtml: '<i/>',
  594. iconClass: 'icon',
  595. invalidClass: 'invalid',
  596. validClass: 'valid',
  597. silentLoad: true,
  598. onValidate: $.noop,
  599. onSubmit: $.noop,
  600. rules: {},
  601. errors: {}
  602. };
  603. plugin.global = {
  604. _format: function(str) {
  605. var args = [].slice.call(arguments, 1);
  606. return str.replace(/\{(\d)\}/g, function(_, match) {
  607. return args[+match] || '';
  608. }).replace(/\{\*([^*}]*)\}/g, function(_, sep) {
  609. return args.join(sep || ', ');
  610. });
  611. },
  612. _getKey: function(key, obj) {
  613. return key.split('.').reduce(function(a,b) {
  614. return a && a[b];
  615. }, obj);
  616. },
  617. i18n: {},
  618. ruleSeparator: ' ',
  619. argSeparator: ':',
  620. rules: require('./rules'),
  621. errors: require('./errors'),
  622. extensions: [
  623. require('./extensions/dynamic-fields/dynamic-fields.ext'),
  624. require('./extensions/ajax/ajax.ext'),
  625. require('./extensions/steps/steps.ext'),
  626. require('./extensions/custom-inputs/custom-inputs.ext'),
  627. require('./extensions/datepicker/datepicker.ext'),
  628. require('./extensions/adaptive/adaptive.ext')
  629. ]
  630. };
  631. plugin.methods = $.extend({}, require('./private'), require('./public'));
  632. require('./plugin')(plugin);
  633. }(jQuery, window, document));
  634. },{"./errors":1,"./extensions/adaptive/adaptive.ext":2,"./extensions/ajax/ajax.ext":3,"./extensions/custom-inputs/custom-inputs.ext":4,"./extensions/datepicker/datepicker.ext":7,"./extensions/dynamic-fields/dynamic-fields.ext":8,"./extensions/steps/steps.ext":10,"./plugin":12,"./private":13,"./public":14,"./rules":15}],12:[function(require,module,exports){
  635. /**
  636. * Plugin boilerplate
  637. */
  638. module.exports = (function() {
  639. var AP = Array.prototype;
  640. return function(plugin) {
  641. plugin = $.extend(true, {
  642. name: 'plugin',
  643. defaults: {
  644. disabledExtensions: 'none'
  645. },
  646. methods: {},
  647. global: {},
  648. }, plugin);
  649. $[plugin.name] = $.extend({
  650. addExtension: function(extension) {
  651. plugin.global.extensions.push(extension);
  652. }
  653. }, plugin.global);
  654. function Plugin(element, options) {
  655. this.opts = $.extend({}, plugin.defaults, options);
  656. this.el = element;
  657. this._name = plugin.name;
  658. this._init();
  659. }
  660. Plugin._extended = {};
  661. Plugin.prototype._hasExtension = function(extension) {
  662. var self = this;
  663. return plugin.global.extensions.filter(function(ext) {
  664. return ext.name == extension && self.opts.disabledExtensions.indexOf(ext.name) < 0;
  665. }).length;
  666. };
  667. Plugin.prototype._extend = function(extensions) {
  668. var self = this;
  669. $.each(extensions, function(i, extension) {
  670. $.extend(self.opts, $.extend(true, extension.options, self.opts));
  671. $.each(extension.methods, function(method, fn) {
  672. if (self.opts.disabledExtensions.indexOf(extension.name) > -1) {
  673. return;
  674. }
  675. if (Plugin.prototype[method.split(':')[0]]) {
  676. Plugin._extended[method] = Plugin._extended[method] || [];
  677. Plugin._extended[method].push({ name: extension.name, fn: fn });
  678. } else {
  679. Plugin.prototype[method] = fn;
  680. }
  681. });
  682. });
  683. };
  684. Plugin.prototype._inject = function(method) {
  685. var args = [].slice.call(arguments, 1);
  686. if (typeof method == 'function') return method.call(this);
  687. var self = this;
  688. if (Plugin._extended[method]) {
  689. $.each(Plugin._extended[method], function(i, plugin) {
  690. plugin.fn.apply(self, args);
  691. });
  692. }
  693. };
  694. Plugin.prototype._init = $.noop;
  695. Plugin.prototype[plugin.name] = function(method) {
  696. if (!method) return this;
  697. try { return this[method].apply(this, AP.slice.call(arguments, 1)); }
  698. catch(e) {}
  699. };
  700. $.extend(Plugin.prototype, plugin.methods);
  701. $.fn[plugin.name] = function() {
  702. var args = AP.slice.call(arguments)
  703. , methodArray = typeof args[0] == 'string' && args[0].split(':')
  704. , method = methodArray[methodArray.length > 1 ? 1 : 0]
  705. , prefix = methodArray.length > 1 && methodArray[0]
  706. , opts = typeof args[0] == 'object' && args[0]
  707. , params = args.slice(1)
  708. , ret;
  709. if (prefix) {
  710. method = prefix + method.substr(0,1).toUpperCase() + method.substr(1,method.length-1);
  711. }
  712. this.each(function() {
  713. var instance = $.data(this, plugin.name);
  714. // Method
  715. if (instance) {
  716. return ret = instance[plugin.name].apply(instance, [method].concat(params));
  717. }
  718. // Init
  719. return $.data(this, plugin.name, new Plugin(this, opts));
  720. });
  721. return prefix ? ret : this;
  722. };
  723. };
  724. }());
  725. },{}],13:[function(require,module,exports){
  726. /**
  727. * Private methods
  728. */
  729. module.exports = {
  730. _init: function() {
  731. var self = this;
  732. this.$form = $(this.el);
  733. this.$fields = $();
  734. this.$inputs = $();
  735. this._extend($.idealforms.extensions);
  736. this._i18n();
  737. this._inject('_init');
  738. this._addMarkupRules();
  739. this.addRules(this.opts.rules || {});
  740. this.$form.submit(function(e) {
  741. self._validateAll(true);
  742. self.focusFirstInvalid();
  743. self.opts.onSubmit.call(self, self.getInvalid().length, e);
  744. });
  745. if (! this.opts.silentLoad) {
  746. // 1ms timeout to make sure error shows up
  747. setTimeout($.proxy(this.focusFirstInvalid, this), 1);
  748. }
  749. },
  750. _addMarkupRules: function() {
  751. var rules = {};
  752. this.$form.find('input, select, textarea').each(function() {
  753. var rule = $(this).data('idealforms-rules');
  754. if (rule && ! rules[this.name]) rules[this.name] = rule;
  755. });
  756. this.addRules(rules);
  757. },
  758. _i18n: function() {
  759. var self = this;
  760. $.each($.idealforms.i18n, function(locale, lang) {
  761. var errors = lang.errors
  762. , options = {};
  763. delete lang.errors;
  764. for (var ext in lang) options[ext] = { i18n: lang[ext] };
  765. $.extend($.idealforms.errors, errors);
  766. $.extend(true, self.opts, options);
  767. });
  768. },
  769. _buildField: function(input) {
  770. var self = this
  771. , $field = this._getField(input)
  772. , $icon;
  773. $icon = $(this.opts.iconHtml, {
  774. class: this.opts.iconClass,
  775. click: function(){ $(input).focus() }
  776. });
  777. if (! this.$fields.filter($field).length) {
  778. this.$fields = this.$fields.add($field);
  779. if (this.opts.iconHtml) $field.append($icon);
  780. $field.addClass('idealforms-field idealforms-field-'+ input.type);
  781. }
  782. this._addEvents(input);
  783. this._inject('_buildField', input);
  784. },
  785. _addEvents: function(input) {
  786. var self = this
  787. , $field = this._getField(input);
  788. $(input)
  789. .on('change keyup', function(e) {
  790. if (e.which == 9 || e.which == 16) return;
  791. self._validate(this, true, true);
  792. })
  793. .focus(function() {
  794. if (! self.isValid(this.name)) {
  795. $field.find(self.opts.error).show();
  796. }
  797. })
  798. .blur(function() {
  799. $field.find(self.opts.error).hide();
  800. });
  801. },
  802. _isRequired: function(input) {
  803. // We assume non-text inputs with rules are required
  804. if ($(input).is(':checkbox, :radio, select')) return true;
  805. return this.opts.rules[input.name].indexOf('required') > -1;
  806. },
  807. _getRelated: function(input) {
  808. return this._getField(input).find('[name="'+ input.name +'"]');
  809. },
  810. _getField: function(input) {
  811. return $(input).closest(this.opts.field);
  812. },
  813. _getFirstInvalid: function() {
  814. return this.getInvalid().first().find('input:first, textarea, select');
  815. },
  816. _handleError: function(input, error, valid) {
  817. valid = valid || this.isValid(input.name);
  818. var $error = this._getField(input).find(this.opts.error);
  819. this.$form.find(this.opts.error).hide();
  820. if (error) $error.text(error);
  821. $error.toggle(!valid);
  822. },
  823. _handleStyle: function(input, valid) {
  824. valid = valid || this.isValid(input.name);
  825. this._getField(input)
  826. .removeClass(this.opts.validClass +' '+ this.opts.invalidClass)
  827. .addClass(valid ? this.opts.validClass : this.opts.invalidClass)
  828. .find('.'+ this.opts.iconClass).show();
  829. },
  830. _fresh: function(input) {
  831. this._getField(input)
  832. .removeClass(this.opts.validClass +' '+ this.opts.invalidClass)
  833. .find(this.opts.error).hide()
  834. .end()
  835. .find('.'+ this.opts.iconClass).toggle(this._isRequired(input));
  836. },
  837. _validate: function(input, handleError, handleStyle) {
  838. var self = this
  839. , $field = this._getField(input)
  840. , userRules = this.opts.rules[input.name].split($.idealforms.ruleSeparator)
  841. , oldValue = $field.data('idealforms-value')
  842. , valid = true
  843. , rule;
  844. // Don't validate input if value hasn't changed
  845. if (! $(input).is(':checkbox, :radio') && oldValue == input.value) {
  846. return $field.data('idealforms-valid');
  847. }
  848. $field.data('idealforms-value', input.value);
  849. // Non-required input with empty value must pass validation
  850. if (! input.value && ! this._isRequired(input)) {
  851. $field.removeData('idealforms-valid');
  852. this._fresh(input);
  853. // Inputs with value or required
  854. } else {
  855. $.each(userRules, function(i, userRule) {
  856. userRule = userRule.split($.idealforms.argSeparator);
  857. rule = userRule[0];
  858. var theRule = $.idealforms.rules[rule]
  859. , args = userRule.slice(1)
  860. , error;
  861. error = $.idealforms._format.apply(null, [
  862. $.idealforms._getKey('errors.'+ input.name +'.'+ rule, self.opts) ||
  863. $.idealforms.errors[rule]
  864. ].concat(args));
  865. valid = typeof theRule == 'function'
  866. ? theRule.apply(self, [input, input.value].concat(args))
  867. : theRule.test(input.value);
  868. $field.data('idealforms-valid', valid);
  869. if (handleError) self._handleError(input, error, valid);
  870. if (handleStyle) self._handleStyle(input, valid);
  871. self.opts.onValidate.call(self, input, rule, valid);
  872. return valid;
  873. });
  874. }
  875. this._inject('_validate', input, rule, valid);
  876. return valid;
  877. },
  878. _validateAll: function(handleError, handleStyle) {
  879. var self = this;
  880. this.$inputs.each(function(){ self._validate(this, handleError, handleStyle); });
  881. }
  882. };
  883. },{}],14:[function(require,module,exports){
  884. /**
  885. * Public methods
  886. */
  887. module.exports = {
  888. addRules: function(rules) {
  889. var self = this;
  890. var $inputs = this.$form.find($.map(rules, function(_, name) {
  891. return '[name="'+ name +'"]';
  892. }).join(','));
  893. $.extend(this.opts.rules, rules);
  894. $inputs.each(function(){ self._buildField(this) });
  895. this.$inputs = this.$inputs.add($inputs);
  896. this._validateAll(true);
  897. this.$fields.find(this.opts.error).hide();
  898. this._inject('addRules');
  899. },
  900. getInvalid: function() {
  901. return this.$fields.filter(function() {
  902. return $(this).data('idealforms-valid') === false;
  903. });
  904. },
  905. focusFirstInvalid: function() {
  906. var firstInvalid = this._getFirstInvalid()[0];
  907. if (firstInvalid) {
  908. this._handleError(firstInvalid);
  909. this._handleStyle(firstInvalid);
  910. this._inject('focusFirstInvalid', firstInvalid);
  911. $(firstInvalid).focus();
  912. }
  913. },
  914. isValid: function(name) {
  915. if (name) return ! this.getInvalid().find('[name="'+ name +'"]').length;
  916. return ! this.getInvalid().length;
  917. },
  918. reset: function(name) {
  919. var self = this
  920. , $inputs = this.$inputs;
  921. if (name) $inputs = $inputs.filter('[name="'+ name +'"]');
  922. $inputs.filter('input:not(:checkbox, :radio), textarea').val('');
  923. $inputs.filter(':checkbox, :radio').prop('checked', false);
  924. $inputs.filter('select').find('option').prop('selected', function() {
  925. return this.defaultSelected;
  926. });
  927. $inputs.change().each(function(){ self._fresh(this) });
  928. this._inject('reset', name);
  929. }
  930. };
  931. },{}],15:[function(require,module,exports){
  932. /**
  933. * Rules
  934. */
  935. module.exports = {
  936. required: /.+/,
  937. digits: /^\d+$/,
  938. email: /^[^@]+@[^@]+\..{2,6}$/,
  939. username: /^[a-z](?=[\w.]{3,31}$)\w*\.?\w*$/i,
  940. pass: /(?=.*\d)(?=.*[a-z])(?=.*[A-Z]).{6,}/,
  941. strongpass: /(?=^.{8,}$)((?=.*\d)|(?=.*\W+))(?![.\n])(?=.*[A-Z])(?=.*[a-z]).*$/,
  942. phone: /^[2-9]\d{2}-\d{3}-\d{4}$/,
  943. zip: /^\d{5}$|^\d{5}-\d{4}$/,
  944. url: /^(?:(ftp|http|https):\/\/)?(?:[\w\-]+\.)+[a-z]{2,6}([\:\/?#].*)?$/i,
  945. number: function(input, value) {
  946. return !isNaN(value);
  947. },
  948. range: function(input, value, min, max) {
  949. return Number(value) >= min && Number(value) <= max;
  950. },
  951. min: function(input, value, min) {
  952. return value.length >= min;
  953. },
  954. max: function(input, value, max) {
  955. return value.length <= max;
  956. },
  957. minoption: function(input, value, min) {
  958. return this._getRelated(input).filter(':checked').length >= min;
  959. },
  960. maxoption: function(input, value, max) {
  961. return this._getRelated(input).filter(':checked').length <= max;
  962. },
  963. minmax: function(input, value, min, max) {
  964. return value.length >= min && value.length <= max;
  965. },
  966. select: function(input, value, def) {
  967. return value != def;
  968. },
  969. extension: function(input) {
  970. var extensions = [].slice.call(arguments, 1)
  971. , valid = false;
  972. $.each(input.files || [{name: input.value}], function(i, file) {
  973. valid = $.inArray(file.name.split('.').pop().toLowerCase(), extensions) > -1;
  974. });
  975. return valid;
  976. },
  977. equalto: function(input, value, target) {
  978. var self = this
  979. , $target = $('[name="'+ target +'"]');
  980. if (this.getInvalid().find($target).length) return false;
  981. $target.off('keyup.equalto').on('keyup.equalto', function() {
  982. self._getField(input).removeData('idealforms-value');
  983. self._validate(input, false, true);
  984. });
  985. return input.value == $target.val();
  986. },
  987. date: function(input, value, format) {
  988. format = format || 'mm/dd/yyyy';
  989. var delimiter = /[^mdy]/.exec(format)[0]
  990. , theFormat = format.split(delimiter)
  991. , theDate = value.split(delimiter);
  992. function isDate(date, format) {
  993. var m, d, y;
  994. for (var i = 0, len = format.length; i < len; i++) {
  995. if (/m/.test(format[i])) m = date[i];
  996. if (/d/.test(format[i])) d = date[i];
  997. if (/y/.test(format[i])) y = date[i];
  998. }
  999. if (!m || !d || !y) return false;
  1000. return m > 0 && m < 13 &&
  1001. y && y.length == 4 &&
  1002. d > 0 && d <= (new Date(y, m, 0)).getDate();
  1003. }
  1004. return isDate(theDate, theFormat);
  1005. }
  1006. };
  1007. },{}]},{},[11])
  1008. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"generated.js","sources":["/usr/lib/node_modules/browserify/node_modules/browser-pack/_prelude.js","/home/elclanrs/Documents/jq-idealforms/js/errors.js","/home/elclanrs/Documents/jq-idealforms/js/extensions/adaptive/adaptive.ext.js","/home/elclanrs/Documents/jq-idealforms/js/extensions/ajax/ajax.ext.js","/home/elclanrs/Documents/jq-idealforms/js/extensions/custom-inputs/custom-inputs.ext.js","/home/elclanrs/Documents/jq-idealforms/js/extensions/custom-inputs/idealfile.js","/home/elclanrs/Documents/jq-idealforms/js/extensions/custom-inputs/idealradiocheck.js","/home/elclanrs/Documents/jq-idealforms/js/extensions/datepicker/datepicker.ext.js","/home/elclanrs/Documents/jq-idealforms/js/extensions/dynamic-fields/dynamic-fields.ext.js","/home/elclanrs/Documents/jq-idealforms/js/extensions/steps/idealsteps.js","/home/elclanrs/Documents/jq-idealforms/js/extensions/steps/steps.ext.js","/home/elclanrs/Documents/jq-idealforms/js/main.js","/home/elclanrs/Documents/jq-idealforms/js/plugin.js","/home/elclanrs/Documents/jq-idealforms/js/private.js","/home/elclanrs/Documents/jq-idealforms/js/public.js","/home/elclanrs/Documents/jq-idealforms/js/rules.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error(\"Cannot find module '\"+o+\"'\")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","/**\n * Errors\n */\nmodule.exports = {\n\n  required: 'This field is required',\n  digits: 'Must be only digits',\n  name: 'Must be at least 3 characters long and must only contain letters',\n  email: 'Must be a valid email',\n  username: 'Must be at between 4 and 32 characters long and start with a letter. You may use letters, numbers, underscores, and one dot',\n  pass: 'Must be at least 6 characters long, and contain at least one number, one uppercase and one lowercase letter',\n  strongpass: 'Must be at least 8 characters long and contain at least one uppercase and one lowercase letter and one number or special character',\n  phone: 'Must be a valid phone number',\n  zip: 'Must be a valid zip code',\n  url: 'Must be a valid URL',\n  number: 'Must be a number',\n  range: 'Must be a number between {0} and {1}',\n  min: 'Must be at least {0} characters long',\n  max: 'Must be under {0} characters',\n  minoption: 'Select at least {0} options',\n  maxoption: 'Select no more than {0} options',\n  minmax: 'Must be between {0} and {1} characters long',\n  select: 'Select an option',\n  extension: 'File(s) must have a valid extension ({*})',\n  equalto: 'Must have the same value as the \"{0}\" field',\n  date: 'Must be a valid date {0}'\n\n};\n","/**\n * Adaptive\n */\nmodule.exports = {\n\n  name: 'adaptive',\n\n  methods: {\n\n    // @extend\n    _init: function () {\n\n      var self = this;\n\n      var $dummy = $('<p class=\"idealforms-field-width\"/>').appendTo('body');\n\n      this.opts.adaptiveWidth = $dummy.css('width').replace('px','');\n\n      function adapt() {\n\n        var formWidth = self.$form.outerWidth()\n          , isAdaptive = self.opts.adaptiveWidth > formWidth;\n\n        self.$form.toggleClass('adaptive', isAdaptive);\n\n        if (self._hasExtension('steps')) {\n          self.$stepsContainer.toggleClass('adaptive', isAdaptive);\n        }\n\n        $('#ui-datepicker-div').hide();\n      }\n\n      $(window).resize(adapt);\n      adapt();\n\n      this.$form.find('select, .datepicker').each(function() {\n        self._getField(this).find(self.opts.error).addClass('hidden');\n      });\n\n      $dummy.remove();\n    }\n\n  }\n};\n","module.exports = {\n\n  name: 'ajax',\n\n  methods: {\n\n    // @extend\n    _init: function() {\n\n      $.extend($.idealforms, { _requests: {} });\n\n      $.idealforms.errors.ajax = $.idealforms.errors.ajax || 'Loading...';\n\n      $.extend($.idealforms.rules, {\n\n        ajax: function(input) {\n\n          var self = this\n            , $field = this._getField(input)\n            , url = $(input).data('idealforms-ajax')\n            , userError = $.idealforms._getKey('errors.'+ input.name +'.ajaxError', self.opts)\n            , requests = $.idealforms._requests\n            , data = {};\n\n          data[input.name] = input.value;\n\n          $field.addClass('ajax');\n\n          if (requests[input.name]) requests[input.name].abort();\n\n          requests[input.name] = $.post(url, data, function(resp) {\n\n            if (resp === true) {\n              $field.data('idealforms-valid', true);\n              self._handleError(input);\n              self._handleStyle(input);\n            } else {\n              self._handleError(input, userError);\n            }\n\n            self.opts.onValidate.call(self, input, 'ajax', resp);\n\n            $field.removeClass('ajax');\n\n          }, 'json');\n\n          return false;\n        }\n      });\n    },\n\n    // @extend\n    _validate: function(input, rule) {\n      if (rule != 'ajax' && $.idealforms._requests[input.name]) {\n        $.idealforms._requests[input.name].abort();\n        this._getField(input).removeClass('ajax');\n      }\n    }\n\n  }\n};\n","require('./idealfile');\nrequire('./idealradiocheck');\n\nmodule.exports = {\n\n  name: 'customInputs',\n\n  options: {\n    customInputs: {\n      i18n: {\n        open: 'Open'\n      }\n    }\n  },\n\n  methods: {\n\n    // @extend\n    _init: function() {\n      this._buildCustomInputs();\n    },\n\n    // @extend\n    'addFields:after': function() {\n      this._buildCustomInputs();\n    },\n\n    _buildCustomInputs: function() {\n      this.$form.find(':file').idealfile(this.opts.customInputs.i18n);\n      this.$form.find(':checkbox, :radio').idealradiocheck();\n    }\n\n  }\n};\n","/**\n * Ideal File\n */\n(function($, win, doc, undefined) {\n\n  // Browser supports HTML5 multiple file?\n  var multipleSupport = typeof $('<input/>')[0].multiple !== 'undefined'\n    , isIE = /msie/i.test(navigator.userAgent)\n    , plugin = {};\n\n  plugin.name = 'idealfile';\n\n  plugin.defaults = {\n    open: 'Open'\n  };\n\n  plugin.methods = {\n\n    _init: function() {\n\n      var $file = $(this.el).addClass('ideal-file') // the original file input\n        , $wrap = $('<div class=\"ideal-file-wrap\">')\n        , $input = $('<input type=\"text\" class=\"ideal-file-filename\" />')\n          // Button that will be used in non-IE browsers\n        , $button = $('<button type=\"button\" class=\"ideal-file-upload\">'+ this.opts.open +'</button>')\n          // Hack for IE\n        , $label = $('<label class=\"ideal-file-upload\" for=\"' + $file[0].id + '\">'+ this.opts.open +'</label>');\n\n      if (isIE) $label.add($button).addClass('ie');\n\n      // Hide by shifting to the left so we\n      // can still trigger events\n      $file.css({\n        position: 'absolute',\n        left: '-9999px'\n      });\n\n      $wrap.append($input, (isIE ? $label : $button)).insertAfter($file);\n\n      // Prevent focus\n      $file.attr('tabIndex', -1);\n      $button.attr('tabIndex', -1);\n\n      $button.click(function () {\n        $file.focus().click(); // Open dialog\n      });\n\n      $file.change(function () {\n\n        var files = []\n          , fileArr, filename;\n\n          // If multiple is supported then extract\n          // all filenames from the file array\n        if (multipleSupport) {\n          fileArr = $file[0].files;\n          for (var i = 0, len = fileArr.length; i < len; i++) {\n            files.push(fileArr[i].name);\n          }\n          filename = files.join(', ');\n\n          // If not supported then just take the value\n          // and remove the path to just show the filename\n        } else {\n          filename = $file.val().split('\\\\').pop();\n        }\n\n        $input .val(filename).attr('title', filename);\n\n      });\n\n      $input.on({\n        blur: function () {\n          $file.trigger('blur');\n        },\n        keydown: function (e) {\n          if (e.which === 13) { // Enter\n            if (!isIE) $file.trigger('click');\n            $(this).closest('form').one('keydown', function(e) {\n              if (e.which === 13) e.preventDefault();\n            });\n          } else if (e.which === 8 || e.which === 46) { // Backspace & Del\n            // In IE the value is read-only\n            // with this trick we remove the old input and add\n            // a clean clone with all the original events attached\n            if (isIE) $file.replaceWith($file = $file.clone(true));\n            $file.val('').trigger('change');\n            $input.val('');\n          } else if (e.which === 9) { // TAB\n            return;\n          } else { // All other keys\n            return false;\n          }\n        }\n      });\n\n    }\n\n  };\n\n  require('../../plugin')(plugin);\n\n}(jQuery, window, document));\n","/*\n * idealRadioCheck: jQuery plguin for checkbox and radio replacement\n * Usage: $('input[type=checkbox], input[type=radio]').idealRadioCheck()\n */\n(function($, win, doc, undefined) {\n\n  var plugin = {};\n\n  plugin.name = 'idealradiocheck';\n\n  plugin.methods = {\n\n    _init: function() {\n\n      var $input = $(this.el);\n      var $span = $('<span/>');\n\n      $span.addClass('ideal-'+ ($input.is(':checkbox') ? 'check' : 'radio'));\n      $input.is(':checked') && $span.addClass('checked'); // init\n      $span.insertAfter($input);\n\n      $input.parent('label')\n        .addClass('ideal-radiocheck-label')\n        .attr('onclick', ''); // Fix clicking label in iOS\n\n      $input.css({ position: 'absolute', left: '-9999px' }); // hide by shifting left\n\n      // Events\n      $input.on({\n        change: function() {\n          var $input = $(this);\n          if ( $input.is('input[type=\"radio\"]') ) {\n            $input.parent().siblings('label').find('.ideal-radio').removeClass('checked');\n          }\n          $span.toggleClass('checked', $input.is(':checked'));\n        },\n        focus: function() { $span.addClass('focus') },\n        blur: function() { $span.removeClass('focus') },\n        click: function() { $(this).trigger('focus') }\n      });\n    }\n\n  };\n\n  require('../../plugin')(plugin);\n\n}(jQuery, window, document));\n\n","module.exports = {\n\n  name: 'datepicker',\n\n  methods: {\n\n    // @extend\n    _init: function() {\n      this._buildDatepicker();\n    },\n\n   _buildDatepicker: function() {\n\n      var $datepicker = this.$form.find('input.datepicker');\n\n      // Always show datepicker below the input\n      if (jQuery.ui) {\n        $.datepicker._checkOffset = function(a,b,c){ return b };\n      }\n\n      if (jQuery.ui && $datepicker.length) {\n\n        $datepicker.each(function() {\n\n          $(this).datepicker({\n            beforeShow: function(input) {\n              $(input).addClass('open');\n            },\n            onChangeMonthYear: function() {\n              // Hack to fix IE9 not resizing\n              var $this = $(this)\n                , width = $this.outerWidth(); // cache first!\n              setTimeout(function() {\n                $this.datepicker('widget').css('width', width);\n              }, 1);\n            },\n            onClose: function() {\n              $(this).removeClass('open');\n            }\n          });\n        });\n\n        // Adjust width\n        $datepicker.on('focus keyup', function() {\n          var t = $(this), w = t.outerWidth();\n          t.datepicker('widget').css('width', w);\n        });\n      }\n    }\n\n  }\n};\n","function template(html, data) {\n\n  var loop = /\\{@([^}]+)\\}(.+?)\\{\\/\\1\\}/g\n    , loopVariable = /\\{#([^}]+)\\}/g\n    , variable = /\\{([^}]+)\\}/g;\n\n  return html\n    .replace(loop, function(_, key, list) {\n      return $.map(data[key], function(item) {\n        return list.replace(loopVariable, function(_, k) {\n          return item[k];\n        });\n      }).join('');\n    })\n    .replace(variable, function(_, key) {\n      return data[key] || '';\n    });\n}\n\nmodule.exports = {\n\n  name: 'dynamicFields',\n\n  options: {\n\n    templates: {\n\n      base:'\\\n        <div class=\"field\">\\\n          <label class=\"main\">{label}</label>\\\n          {field}\\\n          <span class=\"error\"></span>\\\n        </div>\\\n      ',\n\n      text: '<input name=\"{name}\" type=\"{subtype}\" value=\"{value}\" {attrs}>',\n\n      file: '<input id=\"{name}\" name=\"{name}\" type=\"file\" {attrs}>',\n\n      textarea: '<textarea name=\"{name}\" {attrs}>{text}</textarea>',\n\n      group: '\\\n        <p class=\"group\">\\\n          {@list}\\\n          <label><input name=\"{name}\" type=\"{subtype}\" value=\"{#value}\" {#attrs}>{#text}</label>\\\n          {/list}\\\n        </p>\\\n      ',\n\n      select: '\\\n        <select name={name}>\\\n          {@list}\\\n          <option value=\"{#value}\">{#text}</option>\\\n          {/list}\\\n        </select>\\\n      '\n    }\n  },\n\n  methods: {\n\n    addFields: function(fields) {\n\n      var self = this;\n\n      $.each(fields, function(name, field) {\n\n        var typeArray = field.type.split(':')\n          , rules = {}\n          , $last = self.$form.find(self.opts.field).last();\n\n        field.name = name;\n        field.type = typeArray[0];\n        if (typeArray[1]) field.subtype = typeArray[1];\n\n        field.html = template(self.opts.templates.base, {\n          label: field.label,\n          field: template(self.opts.templates[field.type], field)\n        });\n\n        self._inject('addFields:before', field);\n\n        if (field.after || field.before) {\n          self.$form.find('[name=\"'+ (field.after || field.before) +'\"]').first().each(function() {\n            self._getField(this)[field.after ? 'after' : 'before'](field.html);\n          });\n        } else {\n          // Form has at least one field\n          if ($last.length) $last.after(field.html);\n          // Form has no fields\n          else self.$form.append(field.html);\n        }\n\n        if (field.rules) {\n          rules[name] = field.rules;\n          self.addRules(rules);\n        }\n\n        self._inject('addFields:after', field);\n      });\n\n    },\n\n    removeFields: function(names) {\n\n      var self = this;\n\n      $.each(names.split(' '), function(i, name) {\n        var $field = self._getField($('[name=\"'+ name +'\"]'));\n        self.$fields = self.$fields.filter(function() {\n          return ! $(this).is($field);\n        });\n        $field.remove();\n      });\n\n      this._inject('removeFields');\n    },\n\n    toggleFields: function(names) {\n\n      var self = this;\n\n      $.each(names.split(' '), function(i, name) {\n        var $field = self._getField($('[name=\"'+ name +'\"]'));\n        $field.data('idealforms-valid', $field.is(':visible')).toggle();\n      });\n\n      this._inject('toggleFields');\n    }\n\n  }\n};\n","/*!\n * Ideal Steps\n*/\n(function($, win, doc, undefined) {\n\n  var plugin = {};\n\n  plugin.name = 'idealsteps';\n\n  plugin.defaults = {\n    nav: '.idealsteps-nav',\n    navItems: 'li',\n    buildNavItems: true,\n    wrap: '.idealsteps-wrap',\n    step: '.idealsteps-step',\n    activeClass: 'idealsteps-step-active',\n    before: $.noop,\n    after: $.noop,\n    fadeSpeed: 0\n  };\n\n  plugin.methods = {\n\n    _init: function() {\n\n      var self = this,\n          active = this.opts.activeClass;\n\n      this.$el = $(this.el);\n\n      this.$nav = this.$el.find(this.opts.nav);\n      this.$navItems = this.$nav.find(this.opts.navItems);\n\n      this.$wrap = this.$el.find(this.opts.wrap);\n      this.$steps = this.$wrap.find(this.opts.step);\n\n      if (this.opts.buildNavItems) this._buildNavItems();\n\n      this.$steps.hide().first().show();\n      this.$navItems.removeClass(active).first().addClass(active);\n\n      this.$navItems.click(function(e) {\n        e.preventDefault();\n        if (! $(this).is('.'+ self.opts.activeClass)) {\n          self.go(self.$navItems.index(this));\n        }\n      });\n    },\n\n    _buildNavItems: function() {\n\n      var self = this,\n          isCustom = typeof this.opts.buildNavItems == 'function',\n          item = function(val){ return '<li><a href=\"#\" tabindex=\"-1\">'+ val +'</a></li>'; },\n          items;\n\n      items = isCustom ?\n        this.$steps.map(function(i){ return item(self.opts.buildNavItems.call(self, i)) }).get() :\n        this.$steps.map(function(i){ return item(++i); }).get();\n\n      this.$navItems = $(items.join(''));\n\n      this.$nav.append($('<ul/>').append(this.$navItems));\n    },\n\n    _getCurIdx: function() {\n      return this.$steps.index(this.$steps.filter(':visible'));\n    },\n\n    go: function(idx) {\n\n      var active = this.opts.activeClass,\n          fadeSpeed = this.opts.fadeSpeed;\n\n      if (typeof idx == 'function') idx = idx.call(this, this._getCurIdx());\n\n      if (idx >= this.$steps.length) idx = 0;\n      if (idx < 0) idx = this.$steps.length-1;\n\n      this.opts.before.call(this, idx);\n\n      this.$navItems.removeClass(active).eq(idx).addClass(active);\n      this.$steps.hide().eq(idx).fadeIn(fadeSpeed);\n\n      this.opts.after.call(this, idx);\n    },\n\n    prev: function() {\n      this.go(this._getCurIdx() - 1);\n    },\n\n    next: function() {\n      this.go(this._getCurIdx() + 1);\n    },\n\n    first: function() {\n      this.go(0);\n    },\n\n    last: function() {\n      this.go(this.$steps.length-1);\n    }\n  };\n\n  require('../../plugin')(plugin);\n\n}(jQuery, window, document));\n","require('./idealsteps');\n\nmodule.exports = {\n\n  name: 'steps',\n\n  options: {\n\n    steps: {\n\n      container: '.idealsteps-container',\n      nav: '.idealsteps-nav',\n      navItems: 'li',\n      buildNavItems: function(i) {\n        return this.opts.steps.i18n.step +' '+ (i+1);\n      },\n      wrap: '.idealsteps-wrap',\n      step: '.idealsteps-step',\n      activeClass: 'idealsteps-step-active',\n      before: $.noop,\n      after: $.noop,\n      fadeSpeed: 0,\n\n      i18n: {\n        step: 'Step'\n      }\n    }\n  },\n\n  methods: {\n\n    // @extend\n    _init: function() {\n      this._buildSteps();\n    },\n\n    // @extend\n    _validate: function() {\n\n      var self = this;\n\n      this._updateSteps();\n\n      if (this._hasExtension('ajax')) {\n        $.each($.idealforms._requests, function(key, request) {\n          request.done(function(){ self._updateSteps() });\n        });\n      }\n    },\n\n    // @extend\n    focusFirstInvalid: function(firstInvalid) {\n\n      var self = this;\n\n      this.$stepsContainer.idealsteps('go', function() {\n        return this.$steps.filter(function() {\n          return $(this).find(firstInvalid).length;\n        }).index();\n      });\n\n      setTimeout(function(){ $(firstInvalid).focus() }, this.opts.steps.fadeSpeed);\n    },\n\n    // @extend\n    addRules: function() {\n      this.firstStep();\n    },\n\n    // @extend\n    'addFields:before': function(field) {\n\n      if (field.after || field.before) return;\n\n      var $steps = this.$stepsContainer.find(this.opts.steps.step);\n\n      if (! ('appendToStep' in field)) {\n        field.appendToStep = $steps.length-1;\n      }\n\n      field.after = $steps\n        .eq(field.appendToStep)\n        .find('input, select, textarea')\n        .last()[0].name;\n    },\n\n    // @extend\n    toggleFields: function() {\n      this._updateSteps();\n    },\n\n    // @extend\n    removeFields: function() {\n      this._updateSteps();\n    },\n\n    _buildSteps: function() {\n\n      var self = this, options\n        , hasRules = ! $.isEmptyObject(this.opts.rules)\n        , buildNavItems = this.opts.steps.buildNavItems\n        , counter = hasRules\n          ? '<span class=\"counter\"/>'\n          : '<span class=\"counter zero\">0</span>';\n\n      if (this.opts.steps.buildNavItems) {\n        this.opts.steps.buildNavItems = function(i) {\n          return buildNavItems.call(self, i) + counter;\n        };\n      }\n\n      this.$stepsContainer = this.$form\n        .closest(this.opts.steps.container)\n        .idealsteps(this.opts.steps);\n    },\n\n    _updateSteps: function() {\n\n      var self = this;\n\n      this.$stepsContainer.idealsteps('_inject', function() {\n\n        var idealsteps = this;\n\n        this.$navItems.each(function(i) {\n          var invalid = idealsteps.$steps.eq(i).find(self.getInvalid()).length;\n          $(this).find('span').text(invalid).toggleClass('zero', ! invalid);\n        });\n      });\n    },\n\n    goToStep: function(idx) {\n      this.$stepsContainer.idealsteps('go', idx);\n    },\n\n    prevStep: function() {\n      this.$stepsContainer.idealsteps('prev');\n    },\n\n    nextStep: function() {\n      this.$stepsContainer.idealsteps('next');\n    },\n\n    firstStep: function() {\n      this.$stepsContainer.idealsteps('first');\n    },\n\n    lastStep: function() {\n      this.$stepsContainer.idealsteps('last');\n    }\n  }\n\n};\n","/*!\n * jQuery Ideal Forms\n * @author: Cedric Ruiz\n * @version: 3.0\n * @license GPL or MIT\n */\n(function($, win, doc, undefined) {\n\n  var plugin = {};\n\n  plugin.name = 'idealforms';\n\n  plugin.defaults = {\n    field: '.field',\n    error: '.error',\n    iconHtml: '<i/>',\n    iconClass: 'icon',\n    invalidClass: 'invalid',\n    validClass: 'valid',\n    silentLoad: true,\n    onValidate: $.noop,\n    onSubmit: $.noop,\n    rules: {},\n    errors: {}\n  };\n\n  plugin.global = {\n\n    _format: function(str) {\n      var args = [].slice.call(arguments, 1);\n      return str.replace(/\\{(\\d)\\}/g, function(_, match) {\n        return args[+match] || '';\n      }).replace(/\\{\\*([^*}]*)\\}/g, function(_, sep) {\n        return args.join(sep || ', ');\n      });\n    },\n\n    _getKey: function(key, obj) {\n      return key.split('.').reduce(function(a,b) {\n        return a && a[b];\n      }, obj);\n    },\n\n    i18n: {},\n\n    ruleSeparator: ' ',\n    argSeparator: ':',\n\n    rules: require('./rules'),\n    errors: require('./errors'),\n\n    extensions: [\n      require('./extensions/dynamic-fields/dynamic-fields.ext'),\n      require('./extensions/ajax/ajax.ext'),\n      require('./extensions/steps/steps.ext'),\n      require('./extensions/custom-inputs/custom-inputs.ext'),\n      require('./extensions/datepicker/datepicker.ext'),\n      require('./extensions/adaptive/adaptive.ext')\n    ]\n  };\n\n  plugin.methods = $.extend({}, require('./private'), require('./public'));\n\n  require('./plugin')(plugin);\n\n}(jQuery, window, document));\n","/**\n * Plugin boilerplate\n */\nmodule.exports = (function() {\n\n  var AP = Array.prototype;\n\n  return function(plugin) {\n\n    plugin = $.extend(true, {\n      name: 'plugin',\n      defaults: {\n        disabledExtensions: 'none'\n      },\n      methods: {},\n      global: {},\n    }, plugin);\n\n    $[plugin.name] = $.extend({\n\n      addExtension: function(extension) {\n        plugin.global.extensions.push(extension);\n      }\n    }, plugin.global);\n\n    function Plugin(element, options) {\n\n      this.opts = $.extend({}, plugin.defaults, options);\n      this.el = element;\n\n      this._name = plugin.name;\n\n      this._init();\n    }\n\n    Plugin._extended = {};\n\n    Plugin.prototype._hasExtension = function(extension) {\n\n      var self = this;\n\n      return plugin.global.extensions.filter(function(ext) {\n        return ext.name == extension && self.opts.disabledExtensions.indexOf(ext.name) < 0;\n      }).length;\n    };\n\n    Plugin.prototype._extend = function(extensions) {\n\n      var self = this;\n\n      $.each(extensions, function(i, extension) {\n\n        $.extend(self.opts, $.extend(true, extension.options, self.opts));\n\n        $.each(extension.methods, function(method, fn) {\n\n          if (self.opts.disabledExtensions.indexOf(extension.name) > -1) {\n            return;\n          }\n\n          if (Plugin.prototype[method.split(':')[0]]) {\n            Plugin._extended[method] = Plugin._extended[method] || [];\n            Plugin._extended[method].push({ name: extension.name, fn: fn });\n          } else {\n            Plugin.prototype[method] = fn;\n          }\n        });\n\n      });\n    };\n\n    Plugin.prototype._inject = function(method) {\n\n      var args = [].slice.call(arguments, 1);\n\n      if (typeof method == 'function') return method.call(this);\n\n      var self = this;\n\n      if (Plugin._extended[method]) {\n        $.each(Plugin._extended[method], function(i, plugin) {\n          plugin.fn.apply(self, args);\n        });\n      }\n    };\n\n    Plugin.prototype._init = $.noop;\n\n    Plugin.prototype[plugin.name] = function(method) {\n      if (!method) return this;\n      try { return this[method].apply(this, AP.slice.call(arguments, 1)); }\n      catch(e) {}\n    };\n\n    $.extend(Plugin.prototype, plugin.methods);\n\n    $.fn[plugin.name] = function() {\n\n      var args = AP.slice.call(arguments)\n        , methodArray = typeof args[0] == 'string' && args[0].split(':')\n        , method = methodArray[methodArray.length > 1 ? 1 : 0]\n        , prefix = methodArray.length > 1 && methodArray[0]\n        , opts = typeof args[0] == 'object' && args[0]\n        , params = args.slice(1)\n        , ret;\n\n      if (prefix) {\n        method = prefix + method.substr(0,1).toUpperCase() + method.substr(1,method.length-1);\n      }\n\n      this.each(function() {\n\n        var instance = $.data(this, plugin.name);\n\n        // Method\n        if (instance) {\n          return ret = instance[plugin.name].apply(instance, [method].concat(params));\n        }\n\n        // Init\n        return $.data(this, plugin.name, new Plugin(this, opts));\n      });\n\n      return prefix ? ret : this;\n    };\n  };\n\n}());\n","/**\n * Private methods\n */\nmodule.exports = {\n\n  _init: function() {\n\n    var self = this;\n\n    this.$form = $(this.el);\n    this.$fields = $();\n    this.$inputs = $();\n\n    this._extend($.idealforms.extensions);\n    this._i18n();\n\n    this._inject('_init');\n\n    this._addMarkupRules();\n    this.addRules(this.opts.rules || {});\n\n    this.$form.submit(function(e) {\n      self._validateAll(true);\n      self.focusFirstInvalid();\n      self.opts.onSubmit.call(self, self.getInvalid().length, e);\n    });\n\n    if (! this.opts.silentLoad) {\n      // 1ms timeout to make sure error shows up\n      setTimeout($.proxy(this.focusFirstInvalid, this), 1);\n    }\n  },\n\n  _addMarkupRules: function() {\n\n    var rules = {};\n\n    this.$form.find('input, select, textarea').each(function() {\n      var rule = $(this).data('idealforms-rules');\n      if (rule && ! rules[this.name]) rules[this.name] = rule;\n    });\n\n    this.addRules(rules);\n  },\n\n  _i18n: function() {\n\n    var self = this;\n\n    $.each($.idealforms.i18n, function(locale, lang) {\n\n      var errors = lang.errors\n        , options = {};\n\n      delete lang.errors;\n\n      for (var ext in lang) options[ext] = { i18n: lang[ext] };\n\n      $.extend($.idealforms.errors, errors);\n      $.extend(true, self.opts, options);\n    });\n  },\n\n  _buildField: function(input) {\n\n    var self = this\n      , $field = this._getField(input)\n      , $icon;\n\n    $icon = $(this.opts.iconHtml, {\n      class: this.opts.iconClass,\n      click: function(){ $(input).focus() }\n    });\n\n    if (! this.$fields.filter($field).length) {\n      this.$fields = this.$fields.add($field);\n      if (this.opts.iconHtml) $field.append($icon);\n      $field.addClass('idealforms-field idealforms-field-'+ input.type);\n    }\n\n    this._addEvents(input);\n\n    this._inject('_buildField', input);\n  },\n\n  _addEvents: function(input) {\n\n    var self = this\n      , $field = this._getField(input);\n\n    $(input)\n      .on('change keyup', function(e) {\n        if (e.which == 9 || e.which == 16) return;\n        self._validate(this, true, true);\n      })\n      .focus(function() {\n        if (! self.isValid(this.name)) {\n          $field.find(self.opts.error).show();\n        }\n      })\n      .blur(function() {\n        $field.find(self.opts.error).hide();\n      });\n  },\n\n  _isRequired: function(input) {\n    // We assume non-text inputs with rules are required\n    if ($(input).is(':checkbox, :radio, select')) return true;\n    return this.opts.rules[input.name].indexOf('required') > -1;\n  },\n\n  _getRelated: function(input) {\n    return this._getField(input).find('[name=\"'+ input.name +'\"]');\n  },\n\n  _getField: function(input) {\n    return $(input).closest(this.opts.field);\n  },\n\n  _getFirstInvalid: function() {\n    return this.getInvalid().first().find('input:first, textarea, select');\n  },\n\n  _handleError: function(input, error, valid) {\n    valid = valid || this.isValid(input.name);\n    var $error = this._getField(input).find(this.opts.error);\n    this.$form.find(this.opts.error).hide();\n    if (error) $error.text(error);\n    $error.toggle(!valid);\n  },\n\n  _handleStyle: function(input, valid) {\n    valid = valid || this.isValid(input.name);\n    this._getField(input)\n      .removeClass(this.opts.validClass +' '+ this.opts.invalidClass)\n      .addClass(valid ? this.opts.validClass : this.opts.invalidClass)\n      .find('.'+ this.opts.iconClass).show();\n  },\n\n  _fresh: function(input) {\n    this._getField(input)\n      .removeClass(this.opts.validClass +' '+ this.opts.invalidClass)\n      .find(this.opts.error).hide()\n      .end()\n      .find('.'+ this.opts.iconClass).toggle(this._isRequired(input));\n  },\n\n  _validate: function(input, handleError, handleStyle) {\n\n    var self = this\n      , $field = this._getField(input)\n      , userRules = this.opts.rules[input.name].split($.idealforms.ruleSeparator)\n      , oldValue = $field.data('idealforms-value')\n      , valid = true\n      , rule;\n\n    // Don't validate input if value hasn't changed\n    if (! $(input).is(':checkbox, :radio') && oldValue == input.value) {\n      return $field.data('idealforms-valid');\n    }\n\n    $field.data('idealforms-value', input.value);\n\n    // Non-required input with empty value must pass validation\n    if (! input.value && ! this._isRequired(input)) {\n      $field.removeData('idealforms-valid');\n      this._fresh(input);\n\n    // Inputs with value or required\n    } else {\n\n      $.each(userRules, function(i, userRule) {\n\n        userRule = userRule.split($.idealforms.argSeparator);\n\n        rule = userRule[0];\n\n        var theRule = $.idealforms.rules[rule]\n          , args = userRule.slice(1)\n          , error;\n\n        error = $.idealforms._format.apply(null, [\n          $.idealforms._getKey('errors.'+ input.name +'.'+ rule, self.opts) ||\n          $.idealforms.errors[rule]\n        ].concat(args));\n\n        valid = typeof theRule == 'function'\n          ? theRule.apply(self, [input, input.value].concat(args))\n          : theRule.test(input.value);\n\n        $field.data('idealforms-valid', valid);\n\n        if (handleError) self._handleError(input, error, valid);\n        if (handleStyle) self._handleStyle(input, valid);\n\n        self.opts.onValidate.call(self, input, rule, valid);\n\n        return valid;\n      });\n    }\n\n    this._inject('_validate', input, rule, valid);\n\n    return valid;\n  },\n\n  _validateAll: function(handleError, handleStyle) {\n    var self = this;\n    this.$inputs.each(function(){ self._validate(this, handleError, handleStyle); });\n  }\n};\n","/**\n * Public methods\n */\nmodule.exports = {\n\n  addRules: function(rules) {\n\n    var self = this;\n\n    var $inputs = this.$form.find($.map(rules, function(_, name) {\n      return '[name=\"'+ name +'\"]';\n    }).join(','));\n\n    $.extend(this.opts.rules, rules);\n\n    $inputs.each(function(){ self._buildField(this) });\n    this.$inputs = this.$inputs.add($inputs);\n\n    this._validateAll(true);\n    this.$fields.find(this.opts.error).hide();\n\n    this._inject('addRules');\n  },\n\n  getInvalid: function() {\n    return this.$fields.filter(function() {\n      return $(this).data('idealforms-valid') === false;\n    });\n  },\n\n  focusFirstInvalid: function() {\n\n    var firstInvalid = this._getFirstInvalid()[0];\n\n    if (firstInvalid) {\n      this._handleError(firstInvalid);\n      this._handleStyle(firstInvalid);\n      this._inject('focusFirstInvalid', firstInvalid);\n      $(firstInvalid).focus();\n    }\n  },\n\n  isValid: function(name) {\n    if (name) return ! this.getInvalid().find('[name=\"'+ name +'\"]').length;\n    return ! this.getInvalid().length;\n  },\n\n  reset: function(name) {\n\n    var self = this\n      , $inputs = this.$inputs;\n\n    if (name) $inputs = $inputs.filter('[name=\"'+ name +'\"]');\n\n    $inputs.filter('input:not(:checkbox, :radio), textarea').val('');\n    $inputs.filter(':checkbox, :radio').prop('checked', false);\n    $inputs.filter('select').find('option').prop('selected', function() {\n      return this.defaultSelected;\n    });\n\n    $inputs.change().each(function(){ self._fresh(this) });\n\n    this._inject('reset', name);\n  }\n\n};\n","/**\n * Rules\n */\nmodule.exports = {\n\n  required: /.+/,\n  digits: /^\\d+$/,\n  email: /^[^@]+@[^@]+\\..{2,6}$/,\n  username: /^[a-z](?=[\\w.]{3,31}$)\\w*\\.?\\w*$/i,\n  pass: /(?=.*\\d)(?=.*[a-z])(?=.*[A-Z]).{6,}/,\n  strongpass: /(?=^.{8,}$)((?=.*\\d)|(?=.*\\W+))(?![.\\n])(?=.*[A-Z])(?=.*[a-z]).*$/,\n  phone: /^[2-9]\\d{2}-\\d{3}-\\d{4}$/,\n  zip: /^\\d{5}$|^\\d{5}-\\d{4}$/,\n  url: /^(?:(ftp|http|https):\\/\\/)?(?:[\\w\\-]+\\.)+[a-z]{2,6}([\\:\\/?#].*)?$/i,\n\n  number: function(input, value) {\n    return !isNaN(value);\n  },\n\n  range: function(input, value, min, max) {\n    return Number(value) >= min && Number(value) <= max;\n  },\n\n  min: function(input, value, min) {\n    return value.length >= min;\n  },\n\n  max: function(input, value, max) {\n    return value.length <= max;\n  },\n\n  minoption: function(input, value, min) {\n    return this._getRelated(input).filter(':checked').length >= min;\n  },\n\n  maxoption: function(input, value, max) {\n    return this._getRelated(input).filter(':checked').length <= max;\n  },\n\n  minmax: function(input, value, min, max) {\n    return value.length >= min && value.length <= max;\n  },\n\n  select: function(input, value, def) {\n    return value != def;\n  },\n\n  extension: function(input) {\n\n    var extensions = [].slice.call(arguments, 1)\n      , valid = false;\n\n    $.each(input.files || [{name: input.value}], function(i, file) {\n      valid = $.inArray(file.name.split('.').pop().toLowerCase(), extensions) > -1;\n    });\n\n    return valid;\n  },\n\n  equalto: function(input, value, target) {\n\n    var self = this\n      , $target = $('[name=\"'+ target +'\"]');\n\n    if (this.getInvalid().find($target).length) return false;\n\n    $target.off('keyup.equalto').on('keyup.equalto', function() {\n      self._getField(input).removeData('idealforms-value');\n      self._validate(input, false, true);\n    });\n\n    return input.value == $target.val();\n  },\n\n  date: function(input, value, format) {\n\n    format = format || 'mm/dd/yyyy';\n\n    var delimiter = /[^mdy]/.exec(format)[0]\n      , theFormat = format.split(delimiter)\n      , theDate = value.split(delimiter);\n\n    function isDate(date, format) {\n\n      var m, d, y;\n\n      for (var i = 0, len = format.length; i < len; i++) {\n        if (/m/.test(format[i])) m = date[i];\n        if (/d/.test(format[i])) d = date[i];\n        if (/y/.test(format[i])) y = date[i];\n      }\n\n      if (!m || !d || !y) return false;\n\n      return m > 0 && m < 13 &&\n        y && y.length == 4 &&\n        d > 0 && d <= (new Date(y, m, 0)).getDate();\n    }\n\n    return isDate(theDate, theFormat);\n  }\n\n};\n"]}