xxxxxxxxxx
1
<main>
2
<div class="container">
3
<ul class="stepper horizontal" id="horizontal-stepper">
4
<li class="step active">
5
<div class="step-title waves-effect waves-dark">Step 1</div>
6
<div class="step-new-content">
7
<div class="row">
8
<div class="md-form col-12 ml-auto">
9
<input id="email-horizontal" type="email" class="validate form-control" required>
10
<label for="email-horizontal">Email</label>
11
</div>
12
</div>
13
<div class="step-actions">
14
<button class="waves-effect waves-dark btn btn-sm btn-primary next-step" data-feedback="someFunction21">CONTINUE</button>
15
</div>
16
</div>
17
</li>
18
<li class="step">
19
<div class="step-title waves-effect waves-dark">Step 2</div>
20
<div class="step-new-content">
21
<div class="row">
22
<div class="md-form col-12 ml-auto">
23
<input id="text1-horizontal" type="text" class="validate form-control" required>
24
<label for="text1-horizontal">Input 1</label>
25
</div>
26
<div class="md-form col-12 ml-auto">
27
<input id="text2-horizontal" type="text" class="validate form-control" required>
28
<label for="text2-horizontal">Input 2</label>
29
</div>
30
<div class="md-form col-12 ml-auto">
31
<input id="text3-horizontal" type="text" class="validate form-control" required>
32
<label for="text3-horizontal">Input 3</label>
33
</div>
34
<div class="md-form col-12 ml-auto">
35
<input id="text4-horizontal" type="text" class="validate form-control" required>
36
<label for="text4-horizontal">Input 4</label>
37
</div>
38
<div class="md-form col-12 ml-auto">
39
<input id="text5-horizontal" type="text" class="validate form-control" required>
40
<label for="text5-horizontal">Input 5</label>
41
</div>
42
<div class="md-form col-12 ml-auto">
43
<input id="text6-horizontal" type="text" class="validate form-control" required>
44
<label for="text6-horizontal">Input 6</label>
45
</div>
46
</div>
47
<div class="step-actions">
48
<button class="waves-effect waves-dark btn btn-sm btn-primary next-step" data-feedback="someFunction21">CONTINUE</button>
49
<button class="waves-effect waves-dark btn btn-sm btn-secondary previous-step">BACK</button>
50
</div>
51
</div>
52
</li>
53
<li class="step">
54
<div class="step-title waves-effect waves-dark">Step 3</div>
55
<div class="step-new-content">
56
Finish!
57
<div class="step-actions">
58
<button class="waves-effect waves-dark btn-sm btn btn-primary m-0 mt-4" type="button">SUBMIT</button>
59
</div>
60
</div>
61
</li>
62
</ul>
63
</div>
64
</main>
65
<!-- Main Layout -->
xxxxxxxxxx
1
/* Materializecss Stepper - By Kinark 2016
2
// https://github.com/Kinark/Materialize-stepper
3
// CSS v2.1.3
4
*/
5
/*Validate.js FIX*/
6
label.invalid {
7
font-size: 12.8px;
8
font-size: 0.8rem;
9
font-weight: 500;
10
color: red !important;
11
top: 50px !important;
12
}
13
14
label.invalid.active {
15
-webkit-transform: translateY(0%) !important;
16
-ms-transform: translateY(0%) !important;
17
transform: translateY(0%) !important;
18
}
19
20
/*Validate.js FIX*/
21
ul.stepper .wait-feedback {
22
left: 0;
23
right: 0;
24
top: 0;
25
z-index: 2;
26
position: absolute;
27
width: 100%;
28
height: 100%;
29
text-align: center;
30
display: -webkit-box;
31
display: -webkit-flex;
32
display: -ms-flexbox;
33
display: flex;
34
-webkit-box-pack: center;
35
-webkit-justify-content: center;
36
-ms-flex-pack: center;
37
justify-content: center;
38
-webkit-box-align: center;
39
-webkit-align-items: center;
40
-ms-flex-align: center;
41
align-items: center;
42
}
43
44
ul.stepper .step {
45
position: relative;
46
list-style: none;
47
}
48
49
ul.stepper .step.feedbacking .step-new-content>*:not(.wait-feedback) {
50
opacity: 0.1;
51
-ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=10)";
52
}
53
54
ul.stepper .step:not(:last-of-type).active {
55
margin-bottom: 2.25rem;
56
}
57
58
ul.stepper .step:before {
59
position: absolute;
60
top: 0.75rem;
61
counter-increment: section;
62
content: counter(section);
63
height: 1.75rem;
64
width: 1.75rem;
65
color: white;
66
background-color: rgba(0, 0, 0, 0.3);
67
-webkit-border-radius: 100%;
68
border-radius: 100%;
69
text-align: center;
70
line-height: 1.75rem;
71
font-weight: 400;
72
}
73
74
ul.stepper .step.active:before {
75
background-color: #4285f4;
76
}
77
78
ul.stepper .step.done:before {
79
content: '\f00c';
80
font-family: 'Font Awesome 5 free';
81
font-size: 1rem;
82
font-weight: 900;
83
background-color: #00c851;
84
}
85
86
ul.stepper .step.wrong:before {
87
content: '\f071';
88
font-family: 'Font Awesome 5 free';
89
font-size: 1.1rem;
90
font-weight: 900;
91
background-color: #ff3547;
92
}
93
94
ul.stepper>li:not(:last-of-type) {
95
margin-bottom: 0.625rem;
96
-webkit-transition: margin-bottom 0.4s;
97
-o-transition: margin-bottom 0.4s;
98
transition: margin-bottom 0.4s;
99
}
100
101
ul.stepper .step-title {
102
margin: 0 -1.3rem;
103
cursor: pointer;
104
padding: 0.9688rem 2.75rem 1.5rem 4rem;
105
display: block;
106
}
107
108
ul.stepper .step-title:after {
109
content: attr(data-step-label);
110
display: block;
111
position: absolute;
112
font-size: 0.8rem;
113
color: #424242;
114
font-weight: 400;
115
}
116
117
ul.stepper .step-title:hover {
118
background-color: rgba(0, 0, 0, 0.06);
119
}
120
121
ul.stepper .step.active .step-title {
122
font-weight: 500;
123
}
124
125
ul.stepper .step-new-content {
126
position: relative;
127
display: none;
128
height: calc(100% - 132px);
129
width: inherit;
130
overflow: visible;
131
margin-left: 41px;
132
margin-right: 24px;
133
}
134
135
ul.stepper>.step:not(:last-of-type):after {
136
content: '';
137
position: absolute;
138
top: 3.125rem;
139
left: 0.8438rem;
140
width: 0.0625rem;
141
height: 40%;
142
height: calc(100% - 38px);
143
background-color: rgba(0, 0, 0, 0.1);
144
-webkit-transition: all 0.4s;
145
-o-transition: all 0.4s;
146
transition: all 0.4s;
147
}
148
149
ul.stepper>.step.active:not(:last-child):after {
150
height: 93%;
151
height: calc(100% - 12px);
152
}
153
154
ul.stepper>.step[data-last="true"] {
155
margin-bottom: 0;
156
}
157
158
ul.stepper>.step[data-last="true"]:after {
159
height: 0;
160
width: 0;
161
}
162
163
ul.stepper .step-actions {
164
display: -webkit-box;
165
-webkit-box-pack: start;
166
}
167
168
ul.stepper .step-actions .btn:not(:last-child),
169
ul.stepper .step-actions .btn-flat:not(:last-child),
170
ul.stepper .step-actions .btn-large:not(:last-child) {
171
margin-right: 0.3125rem;
172
}
173
174
ul.stepper .step-new-content .row {
175
margin-bottom: 0.4375rem;
176
}
177
178
ul.stepper .md-form label {
179
left: 0.875rem;
180
}
181
182
ul.stepper .md-form .validate {
183
margin-bottom: 0;
184
}
185
186
@media only screen and (min-width: 993px) {
187
ul.stepper.horizontal {
188
position: relative;
189
display: -webkit-box;
190
display: -webkit-flex;
191
display: -ms-flexbox;
192
display: flex;
193
-webkit-box-pack: justify;
194
-webkit-justify-content: space-between;
195
-ms-flex-pack: justify;
196
justify-content: space-between;
197
min-height: 20rem;
198
margin-left: -1.5rem;
199
margin-right: -1.5rem;
200
padding-left: 1.5rem;
201
padding-right: 1.5rem;
202
overflow: hidden;
203
}
204
205
ul.stepper.horizontal:before {
206
content: '';
207
background-color: transparent;
208
width: 100%;
209
min-height: 5.25rem;
210
position: absolute;
211
left: -3px;
212
-webkit-border-top-left-radius: 2px;
213
border-top-left-radius: 2px;
214
}
215
216
ul.stepper.horizontal:first-child {
217
margin-top: -2.7rem;
218
}
219
220
ul.stepper.horizontal .step {
221
position: static;
222
margin: 0;
223
width: 100%;
224
display: -webkit-box;
225
display: -webkit-flex;
226
display: -ms-flexbox;
227
display: flex;
228
-webkit-box-align: center;
229
-webkit-align-items: center;
230
-ms-flex-align: center;
231
align-items: center;
232
height: 5.25rem !important;
233
}
234
235
ul.stepper.horizontal .step:not(:last-of-type):after {
236
content: '';
237
position: static;
238
display: inline-block;
239
width: 100%;
240
height: 0.0625rem;
241
}
242
243
ul.stepper.horizontal>.step:last-of-type,
244
ul.stepper.horizontal>.step[data-last="true"] {
245
width: auto !important;
246
}
247
248
ul.stepper.horizontal>.step.active:not(:last-of-type):after {
249
content: '';
250
position: static;
251
display: inline-block;
252
width: 100%;
253
height: 0.0625rem;
254
}
255
256
ul.stepper.horizontal .step.active .step-title:before {
257
background-color: #4285f4;
258
}
259
260
ul.stepper.horizontal .step.done .step-title:before {
261
font-family: 'Font Awesome 5 Free';
262
font-weight: 900;
263
content: '\f00c';
264
font-size: 1rem;
265
background: #00c851;
266
}
267
268
ul.stepper.horizontal .step.wrong .step-title:before {
269
font-family: 'Font Awesome 5 Free';
270
font-weight: 900;
271
content: '\f071';
272
font-size: 1.1rem;
273
background-color: #ff3547;
274
}
275
276
ul.stepper.horizontal .step-title {
277
line-height: 5.25rem;
278
height: 5.25rem;
279
margin: 0;
280
padding: 0 1.5625rem 0 4.0625rem;
281
display: inline-block;
282
max-width: 13.75rem;
283
white-space: nowrap;
284
overflow: hidden;
285
-o-text-overflow: ellipsis;
286
text-overflow: ellipsis;
287
-ms-flex-negative: 0;
288
-webkit-flex-shrink: 0;
289
flex-shrink: 0;
290
}
291
292
ul.stepper.horizontal .step:before {
293
content: none;
294
}
295
296
ul.stepper.horizontal .step .step-title:before {
297
position: absolute;
298
top: 1.7813rem;
299
left: 1.1875rem;
300
counter-increment: section;
301
content: counter(section);
302
height: 1.75rem;
303
width: 1.75rem;
304
color: white;
305
background-color: rgba(0, 0, 0, 0.3);
306
-webkit-border-radius: 100%;
307
border-radius: 100%;
308
text-align: center;
309
line-height: 1.75rem;
310
font-weight: 400;
311
}
312
313
ul.stepper.horizontal .step-title:after {
314
top: 0.9375rem;
315
}
316
317
ul.stepper.horizontal .step-new-content {
318
position: absolute;
319
height: calc(100% - 84px);
320
top: 6rem;
321
left: 0;
322
width: 100%;
323
overflow-y: auto;
324
overflow-x: hidden;
325
margin: 0;
326
padding: 1.25rem 1.25rem 4.75rem 1.25rem;
327
}
328
329
ul.stepper.horizontal .step-actions {
330
position: absolute;
331
bottom: 0;
332
left: 0;
333
width: 100%;
334
padding: 20px;
335
-webkit-box-orient: horizontal;
336
-webkit-box-direction: reverse;
337
-webkit-flex-direction: row-reverse;
338
-ms-flex-direction: row-reverse;
339
flex-direction: row-reverse;
340
}
341
342
ul.stepper.horizontal .step-actions .btn:not(:last-child),
343
ul.stepper.horizontal .step-actions .btn-flat:not(:last-child),
344
ul.stepper.horizontal .step-actions .btn-large:not(:last-child) {
345
margin-left: 0.3125rem;
346
margin-right: 0;
347
}
348
349
ul.stepper.horizontal .step-new-content,
350
ul.stepper.horizontal .step-actions {
351
padding-left: 2.5rem;
352
padding-right: 2.5rem;
353
}
354
}
355
xxxxxxxxxx
1
$(document).ready(function () {
2
$('.stepper').mdbStepper();
3
})
4
5
function someFunction21() {
6
setTimeout(function () {
7
$('#horizontal-stepper').nextStep();
8
}, 2000);
9
}
10
11
/* !
12
* jQuery Validation Plugin v1.17.0
13
*
14
* https://jqueryvalidation.org/
15
*
16
* Copyright (c) 2017 Jörn Zaefferer
17
* Released under the MIT license
18
*/
19
(function (factory) {
20
if (typeof define === 'function' && define.amd) {
21
define(['jquery'], factory);
22
} else if (typeof module === 'object' && module.exports) {
23
module.exports = factory(require('jquery'));
24
} else {
25
factory(jQuery);
26
}
27
}(($) => {
28
29
$.extend($.fn, {
30
31
// https://jqueryvalidation.org/validate/
32
validate: function (options) {
33
34
// If nothing is selected, return nothing; can't chain anyway
35
if (!this.length) {
36
if (options && options.debug && window.console) {
37
console.warn("Nothing selected, can't validate, returning nothing.");
38
}
39
return;
40
}
41
42
// Check if a validator for this form was already created
43
var validator = $.data(this[0], "validator");
44
if (validator) {
45
return validator;
46
}
47
48
// Add novalidate tag if HTML5.
49
this.attr("novalidate", "novalidate");
50
51
validator = new $.validator(options, this[0]);
52
$.data(this[0], "validator", validator);
53
54
if (validator.settings.onsubmit) {
55
56
this.on("click.validate", ":submit", function (event) {
57
58
// Track the used submit button to properly handle scripted
59
// submits later.
60
validator.submitButton = event.currentTarget;
61
62
// Allow suppressing validation by adding a cancel class to the submit button
63
if ($(this).hasClass("cancel")) {
64
validator.cancelSubmit = true;
65
}
66
67
// Allow suppressing validation by adding the html5 formnovalidate attribute to the submit button
68
if ($(this).attr("formnovalidate") !== undefined) {
69
validator.cancelSubmit = true;
70
}
71
});
72
73
// Validate the form on submit
74
this.on("submit.validate", function (event) {
75
if (validator.settings.debug) {
76
77
// Prevent form submit to be able to see console output
78
event.preventDefault();
79
}
80
81
function handle() {
82
var hidden, result;
83
84
// Insert a hidden input as a replacement for the missing submit button
85
// The hidden input is inserted in two cases:
86
// - A user defined a `submitHandler`
87
// - There was a pending request due to `remote` method and `stopRequest()`
88
// was called to submit the form in case it's valid
89
if (validator.submitButton && (validator.settings.submitHandler || validator.formSubmitted)) {
90
hidden = $("<input type='hidden'/>")
91
.attr("name", validator.submitButton.name)
92
.val($(validator.submitButton).val())
93
.appendTo(validator.currentForm);
94
}
95
96
if (validator.settings.submitHandler) {
97
result = validator.settings.submitHandler.call(validator, validator.currentForm, event);
98
if (hidden) {
99
100
// And clean up afterwards; thanks to no-block-scope, hidden can be referenced
101
hidden.remove();
102
}
103
if (result !== undefined) {
104
return result;
105
}
106
return false;
107
}
108
return true;
109
}
110
111
// Prevent submit for invalid forms or custom submit handlers
112
if (validator.cancelSubmit) {
113
validator.cancelSubmit = false;
114
return handle();
115
}
116
if (validator.form()) {
117
if (validator.pendingRequest) {
118
validator.formSubmitted = true;
119
return false;
120
}
121
return handle();
122
} else {
123
validator.focusInvalid();
124
return false;
125
}
126
});
127
}
128
129
return validator;
130
},
131
132
// https://jqueryvalidation.org/valid/
133
valid: function () {
134
var valid, validator, errorList;
135
136
if ($(this[0]).is("form")) {
137
valid = this.validate().form();
138
} else {
139
errorList = [];
140
valid = true;
141
validator = $(this[0].form).validate();
142
this.each(function () {
143
valid = validator.element(this) && valid;
144
if (!valid) {
145
errorList = errorList.concat(validator.errorList);
146
}
147
});
148
validator.errorList = errorList;
149
}
150
return valid;
151
},
152
153
// https://jqueryvalidation.org/rules/
154
rules: function (command, argument) {
155
var element = this[0],
156
settings, staticRules, existingRules, data, param, filtered;
157
158
// If nothing is selected, return empty object; can't chain anyway
159
if (element == null) {
160
return;
161
}
162
163
if (!element.form && element.hasAttribute("contenteditable")) {
164
element.form = this.closest("form")[0];
165
element.name = this.attr("name");
166
}
167
168
if (element.form == null) {
169
return;
170
}
171
172
if (command) {
173
settings = $.data(element.form, "validator").settings;
174
staticRules = settings.rules;
175
existingRules = $.validator.staticRules(element);
176
switch (command) {
177
case "add":
178
$.extend(existingRules, $.validator.normalizeRule(argument));
179
180
// Remove messages from rules, but allow them to be set separately
181
delete existingRules.messages;
182
staticRules[element.name] = existingRules;
183
if (argument.messages) {
184
settings.messages[element.name] = $.extend(settings.messages[element.name], argument.messages);
185
}
186
break;
187
case "remove":
188
if (!argument) {
189
delete staticRules[element.name];
190
return existingRules;
191
}
192
filtered = {};
193
$.each(argument.split(/\s/), function (index, method) {
194
filtered[method] = existingRules[method];
195
delete existingRules[method];
196
});
197
return filtered;
198
}
199
}
200
201
data = $.validator.normalizeRules(
202
$.extend({},
203
$.validator.classRules(element),
204
$.validator.attributeRules(element),
205
$.validator.dataRules(element),
206
$.validator.staticRules(element)
207
), element);
208
209
// Make sure required is at front
210
if (data.required) {
211
param = data.required;
212
delete data.required;
213
data = $.extend({
214
required: param
215
}, data);
216
}
217
218
// Make sure remote is at back
219
if (data.remote) {
220
param = data.remote;
221
delete data.remote;
222
data = $.extend(data, {
223
remote: param
224
});
225
}
226
227
return data;
228
}
229
});
230
231
// Custom selectors
232
$.extend($.expr.pseudos || $.expr[":"], { // '|| $.expr[ ":" ]' here enables backwards compatibility to jQuery 1.7. Can be removed when dropping jQ 1.7.x support
233
234
// https://jqueryvalidation.org/blank-selector/
235
blank: function (a) {
236
return !$.trim("" + $(a).val());
237
},
238
239
// https://jqueryvalidation.org/filled-selector/
240
filled: function (a) {
241
var val = $(a).val();
242
return val !== null && !!$.trim("" + val);
243
},
244
245
// https://jqueryvalidation.org/unchecked-selector/
246
unchecked: function (a) {
247
return !$(a).prop("checked");
248
}
249
});
250
251
// Constructor for validator
252
$.validator = function (options, form) {
253
this.settings = $.extend(true, {}, $.validator.defaults, options);
254
this.currentForm = form;
255
this.init();
256
};
257
258
// https://jqueryvalidation.org/jQuery.validator.format/
259
$.validator.format = function (source, params) {
260
if (arguments.length === 1) {
261
return function () {
262
var args = $.makeArray(arguments);
263
args.unshift(source);
264
return $.validator.format.apply(this, args);
265
};
266
}
267
if (params === undefined) {
268
return source;
269
}
270
if (arguments.length > 2 && params.constructor !== Array) {
271
params = $.makeArray(arguments).slice(1);
272
}
273
if (params.constructor !== Array) {
274
params = [params];
275
}
276
$.each(params, function (i, n) {
277
source = source.replace(new RegExp("\\{" + i + "\\}", "g"), function () {
278
return n;
279
});
280
});
281
return source;
282
};
283
284
$.extend($.validator, {
285
286
defaults: {
287
messages: {},
288
groups: {},
289
rules: {},
290
errorClass: "error",
291
pendingClass: "pending",
292
validClass: "valid",
293
errorElement: "label",
294
focusCleanup: false,
295
focusInvalid: true,
296
errorContainer: $([]),
297
errorLabelContainer: $([]),
298
onsubmit: true,
299
ignore: ":hidden",
300
ignoreTitle: false,
301
onfocusin: function (element) {
302
this.lastActive = element;
303
304
// Hide error label and remove error class on focus if enabled
305
if (this.settings.focusCleanup) {
306
if (this.settings.unhighlight) {
307
this.settings.unhighlight.call(this, element, this.settings.errorClass, this.settings.validClass);
308
}
309
this.hideThese(this.errorsFor(element));
310
}
311
},
312
onfocusout: function (element) {
313
if (!this.checkable(element) && (element.name in this.submitted || !this.optional(element))) {
314
this.element(element);
315
}
316
},
317
onkeyup: function (element, event) {
318
319
// Avoid revalidate the field when pressing one of the following keys
320
// Shift => 16
321
// Ctrl => 17
322
// Alt => 18
323
// Caps lock => 20
324
// End => 35
325
// Home => 36
326
// Left arrow => 37
327
// Up arrow => 38
328
// Right arrow => 39
329
// Down arrow => 40
330
// Insert => 45
331
// Num lock => 144
332
// AltGr key => 225
333
var excludedKeys = [
334
16, 17, 18, 20, 35, 36, 37,
335
38, 39, 40, 45, 144, 225
336
];
337
338
if (event.which === 9 && this.elementValue(element) === "" || $.inArray(event.keyCode, excludedKeys) !== -1) {
339
return;
340
} else if (element.name in this.submitted || element.name in this.invalid) {
341
this.element(element);
342
}
343
},
344
onclick: function (element) {
345
346
// Click on selects, radiobuttons and checkboxes
347
if (element.name in this.submitted) {
348
this.element(element);
349
350
// Or option elements, check parent select in that case
351
} else if (element.parentNode.name in this.submitted) {
352
this.element(element.parentNode);
353
}
354
},
355
highlight: function (element, errorClass, validClass) {
356
if (element.type === "radio") {
357
this.findByName(element.name).addClass(errorClass).removeClass(validClass);
358
} else {
359
$(element).addClass(errorClass).removeClass(validClass);
360
}
361
},
362
unhighlight: function (element, errorClass, validClass) {
363
if (element.type === "radio") {
364
this.findByName(element.name).removeClass(errorClass).addClass(validClass);
365
} else {
366
$(element).removeClass(errorClass).addClass(validClass);
367
}
368
}
369
},
370
371
// https://jqueryvalidation.org/jQuery.validator.setDefaults/
372
setDefaults: function (settings) {
373
$.extend($.validator.defaults, settings);
374
},
375
376
messages: {
377
required: "This field is required.",
378
remote: "Please fix this field.",
379
email: "Please enter a valid email address.",
380
url: "Please enter a valid URL.",
381
date: "Please enter a valid date.",
382
dateISO: "Please enter a valid date (ISO).",
383
number: "Please enter a valid number.",
384
digits: "Please enter only digits.",
385
equalTo: "Please enter the same value again.",
386
maxlength: $.validator.format("Please enter no more than {0} characters."),
387
minlength: $.validator.format("Please enter at least {0} characters."),
388
rangelength: $.validator.format("Please enter a value between {0} and {1} characters long."),
389
range: $.validator.format("Please enter a value between {0} and {1}."),
390
max: $.validator.format("Please enter a value less than or equal to {0}."),
391
min: $.validator.format("Please enter a value greater than or equal to {0}."),
392
step: $.validator.format("Please enter a multiple of {0}.")
393
},
394
395
autoCreateRanges: false,
396
397
prototype: {
398
399
init: function () {
400
this.labelContainer = $(this.settings.errorLabelContainer);
401
this.errorContext = this.labelContainer.length && this.labelContainer || $(this.currentForm);
402
this.containers = $(this.settings.errorContainer).add(this.settings.errorLabelContainer);
403
this.submitted = {};
404
this.valueCache = {};
405
this.pendingRequest = 0;
406
this.pending = {};
407
this.invalid = {};
408
this.reset();
409
410
var groups = (this.groups = {}),
411
rules;
412
$.each(this.settings.groups, function (key, value) {
413
if (typeof value === "string") {
414
value = value.split(/\s/);
415
}
416
$.each(value, function (index, name) {
417
groups[name] = key;
418
});
419
});
420
rules = this.settings.rules;
421
$.each(rules, function (key, value) {
422
rules[key] = $.validator.normalizeRule(value);
423
});
424
425
function delegate(event) {
426
427
// Set form expando on contenteditable
428
if (!this.form && this.hasAttribute("contenteditable")) {
429
this.form = $(this).closest("form")[0];
430
this.name = $(this).attr("name");
431
}
432
433
var validator = $.data(this.form, "validator"),
434
eventType = "on" + event.type.replace(/^validate/, ""),
435
settings = validator.settings;
436
if (settings[eventType] && !$(this).is(settings.ignore)) {
437
settings[eventType].call(validator, this, event);
438
}
439
}
440
441
$(this.currentForm)
442
.on("focusin.validate focusout.validate keyup.validate",
443
":text, [type='password'], [type='file'], select, textarea, [type='number'], [type='search'], " +
444
"[type='tel'], [type='url'], [type='email'], [type='datetime'], [type='date'], [type='month'], " +
445
"[type='week'], [type='time'], [type='datetime-local'], [type='range'], [type='color'], " +
446
"[type='radio'], [type='checkbox'], [contenteditable], [type='button']", delegate)
447
448
// Support: Chrome, oldIE
449
// "select" is provided as event.target when clicking a option
450
.on("click.validate", "select, option, [type='radio'], [type='checkbox']", delegate);
451
452
if (this.settings.invalidHandler) {
453
$(this.currentForm).on("invalid-form.validate", this.settings.invalidHandler);
454
}
455
},
456
457
// https://jqueryvalidation.org/Validator.form/
458
form: function () {
459
this.checkForm();
460
$.extend(this.submitted, this.errorMap);
461
this.invalid = $.extend({}, this.errorMap);
462
if (!this.valid()) {
463
$(this.currentForm).triggerHandler("invalid-form", [this]);
464
}
465
this.showErrors();
466
return this.valid();
467
},
468
469
checkForm: function () {
470
this.prepareForm();
471
for (var i = 0, elements = (this.currentElements = this.elements()); elements[i]; i++) {
472
this.check(elements[i]);
473
}
474
return this.valid();
475
},
476
477
// https://jqueryvalidation.org/Validator.element/
478
element: function (element) {
479
var cleanElement = this.clean(element),
480
checkElement = this.validationTargetFor(cleanElement),
481
v = this,
482
result = true,
483
rs, group;
484
485
if (checkElement === undefined) {
486
delete this.invalid[cleanElement.name];
487
} else {
488
this.prepareElement(checkElement);
489
this.currentElements = $(checkElement);
490
491
// If this element is grouped, then validate all group elements already
492
// containing a value
493
group = this.groups[checkElement.name];
494
if (group) {
495
$.each(this.groups, function (name, testgroup) {
496
if (testgroup === group && name !== checkElement.name) {
497
cleanElement = v.validationTargetFor(v.clean(v.findByName(name)));
498
if (cleanElement && cleanElement.name in v.invalid) {
499
v.currentElements.push(cleanElement);
500
result = v.check(cleanElement) && result;
501
}
502
}
503
});
504
}
505
506
rs = this.check(checkElement) !== false;
507
result = result && rs;
508
if (rs) {
509
this.invalid[checkElement.name] = false;
510
} else {
511
this.invalid[checkElement.name] = true;
512
}
513
514
if (!this.numberOfInvalids()) {
515
516
// Hide error containers on last error
517
this.toHide = this.toHide.add(this.containers);
518
}
519
this.showErrors();
520
521
// Add aria-invalid status for screen readers
522
$(element).attr("aria-invalid", !rs);
523
}
524
525
return result;
526
},
527
528
// https://jqueryvalidation.org/Validator.showErrors/
529
showErrors: function (errors) {
530
if (errors) {
531
var validator = this;
532
533
// Add items to error list and map
534
$.extend(this.errorMap, errors);
535
this.errorList = $.map(this.errorMap, function (message, name) {
536
return {
537
message: message,
538
element: validator.findByName(name)[0]
539
};
540
});
541
542
// Remove items from success list
543
this.successList = $.grep(this.successList, function (element) {
544
return !(element.name in errors);
545
});
546
}
547
if (this.settings.showErrors) {
548
this.settings.showErrors.call(this, this.errorMap, this.errorList);
549
} else {
550
this.defaultShowErrors();
551
}
552
},
553
554
// https://jqueryvalidation.org/Validator.resetForm/
555
resetForm: function () {
556
if ($.fn.resetForm) {
557
$(this.currentForm).resetForm();
558
}
559
this.invalid = {};
560
this.submitted = {};
561
this.prepareForm();
562
this.hideErrors();
563
var elements = this.elements()
564
.removeData("previousValue")
565
.removeAttr("aria-invalid");
566
567
this.resetElements(elements);
568
},
569
570
resetElements: function (elements) {
571
var i;
572
573
if (this.settings.unhighlight) {
574
for (i = 0; elements[i]; i++) {
575
this.settings.unhighlight.call(this, elements[i],
576
this.settings.errorClass, "");
577
this.findByName(elements[i].name).removeClass(this.settings.validClass);
578
}
579
} else {
580
elements
581
.removeClass(this.settings.errorClass)
582
.removeClass(this.settings.validClass);
583
}
584
},
585
586
numberOfInvalids: function () {
587
return this.objectLength(this.invalid);
588
},
589
590
objectLength: function (obj) {
591
/* jshint unused: false */
592
var count = 0,
593
i;
594
for (i in obj) {
595
596
// This check allows counting elements with empty error
597
// message as invalid elements
598
if (obj[i] !== undefined && obj[i] !== null && obj[i] !== false) {
599
count++;
600
}
601
}
602
return count;
603
},
604
605
hideErrors: function () {
606
this.hideThese(this.toHide);
607
},
608
609
hideThese: function (errors) {
610
errors.not(this.containers).text("");
611
this.addWrapper(errors).hide();
612
},
613
614
valid: function () {
615
return this.size() === 0;
616
},
617
618
size: function () {
619
return this.errorList.length;
620
},
621
622
focusInvalid: function () {
623
if (this.settings.focusInvalid) {
624
try {
625
$(this.findLastActive() || this.errorList.length && this.errorList[0].element || [])
626
.filter(":visible")
627
.focus()
628
629
// Manually trigger focusin event; without it, focusin handler isn't called, findLastActive won't have anything to find
630
.trigger("focusin");
631
} catch (e) {
632
633
// Ignore IE throwing errors when focusing hidden elements
634
}
635
}
636
},
637
638
findLastActive: function () {
639
var lastActive = this.lastActive;
640
return lastActive && $.grep(this.errorList, function (n) {
641
return n.element.name === lastActive.name;
642
}).length === 1 && lastActive;
643
},
644
645
elements: function () {
646
var validator = this,
647
rulesCache = {};
648
649
// Select all valid inputs inside the form (no submit or reset buttons)
650
return $(this.currentForm)
651
.find("input, select, textarea, [contenteditable]")
652
.not(":submit, :reset, :image, :disabled")
653
.not(this.settings.ignore)
654
.filter(function () {
655
var name = this.name || $(this).attr("name"); // For contenteditable
656
if (!name && validator.settings.debug && window.console) {
657
console.error("%o has no name assigned", this);
658
}
659
660
// Set form expando on contenteditable
661
if (this.hasAttribute("contenteditable")) {
662
this.form = $(this).closest("form")[0];
663
this.name = name;
664
}
665
666
// Select only the first element for each name, and only those with rules specified
667
if (name in rulesCache || !validator.objectLength($(this).rules())) {
668
return false;
669
}
670
671
rulesCache[name] = true;
672
return true;
673
});
674
},
675
676
clean: function (selector) {
677
return $(selector)[0];
678
},
679
680
errors: function () {
681
var errorClass = this.settings.errorClass.split(" ").join(".");
682
return $(this.settings.errorElement + "." + errorClass, this.errorContext);
683
},
684
685
resetInternals: function () {
686
this.successList = [];
687
this.errorList = [];
688
this.errorMap = {};
689
this.toShow = $([]);
690
this.toHide = $([]);
691
},
692
693
reset: function () {
694
this.resetInternals();
695
this.currentElements = $([]);
696
},
697
698
prepareForm: function () {
699
this.reset();
700
this.toHide = this.errors().add(this.containers);
701
},
702
703
prepareElement: function (element) {
704
this.reset();
705
this.toHide = this.errorsFor(element);
706
},
707
708
elementValue: function (element) {
709
var $element = $(element),
710
type = element.type,
711
val, idx;
712
713
if (type === "radio" || type === "checkbox") {
714
return this.findByName(element.name).filter(":checked").val();
715
} else if (type === "number" && typeof element.validity !== "undefined") {
716
return element.validity.badInput ? "NaN" : $element.val();
717
}
718
719
if (element.hasAttribute("contenteditable")) {
720
val = $element.text();
721
} else {
722
val = $element.val();
723
}
724
725
if (type === "file") {
726
727
// Modern browser (chrome & safari)
728
if (val.substr(0, 12) === "C:\\fakepath\\") {
729
return val.substr(12);
730
}
731
732
// Legacy browsers
733
// Unix-based path
734
idx = val.lastIndexOf("/");
735
if (idx >= 0) {
736
return val.substr(idx + 1);
737
}
738
739
// Windows-based path
740
idx = val.lastIndexOf("\\");
741
if (idx >= 0) {
742
return val.substr(idx + 1);
743
}
744
745
// Just the file name
746
return val;
747
}
748
749
if (typeof val === "string") {
750
return val.replace(/\r/g, "");
751
}
752
return val;
753
},
754
755
check: function (element) {
756
element = this.validationTargetFor(this.clean(element));
757
758
var rules = $(element).rules(),
759
rulesCount = $.map(rules, function (n, i) {
760
return i;
761
}).length,
762
dependencyMismatch = false,
763
val = this.elementValue(element),
764
result, method, rule, normalizer;
765
766
// Prioritize the local normalizer defined for this element over the global one
767
// if the former exists, otherwise user the global one in case it exists.
768
if (typeof rules.normalizer === "function") {
769
normalizer = rules.normalizer;
770
} else if (typeof this.settings.normalizer === "function") {
771
normalizer = this.settings.normalizer;
772
}
773
774
// If normalizer is defined, then call it to retreive the changed value instead
775
// of using the real one.
776
// Note that `this` in the normalizer is `element`.
777
if (normalizer) {
778
val = normalizer.call(element, val);
779
780
if (typeof val !== "string") {
781
throw new TypeError("The normalizer should return a string value.");
782
}
783
784
// Delete the normalizer from rules to avoid treating it as a pre-defined method.
785
delete rules.normalizer;
786
}
787
788
for (method in rules) {
789
rule = {
790
method: method,
791
parameters: rules[method]
792
};
793
try {
794
result = $.validator.methods[method].call(this, val, element, rule.parameters);
795
796
// If a method indicates that the field is optional and therefore valid,
797
// don't mark it as valid when there are no other rules
798
if (result === "dependency-mismatch" && rulesCount === 1) {
799
dependencyMismatch = true;
800
continue;
801
}
802
dependencyMismatch = false;
803
804
if (result === "pending") {
805
this.toHide = this.toHide.not(this.errorsFor(element));
806
return;
807
}
808
809
if (!result) {
810
this.formatAndAdd(element, rule);
811
return false;
812
}
813
} catch (e) {
814
if (this.settings.debug && window.console) {
815
console.log("Exception occurred when checking element " + element.id + ", check the '" + rule.method + "' method.", e);
816
}
817
if (e instanceof TypeError) {
818
e.message += ". Exception occurred when checking element " + element.id + ", check the '" + rule.method + "' method.";
819
}
820
821
throw e;
822
}
823
}
824
if (dependencyMismatch) {
825
return;
826
}
827
if (this.objectLength(rules)) {
828
this.successList.push(element);
829
}
830
return true;
831
},
832
833
// Return the custom message for the given element and validation method
834
// specified in the element's HTML5 data attribute
835
// return the generic message if present and no method specific message is present
836
customDataMessage: function (element, method) {
837
return $(element).data("msg" + method.charAt(0).toUpperCase() +
838
method.substring(1).toLowerCase()) || $(element).data("msg");
839
},
840
841
// Return the custom message for the given element name and validation method
842
customMessage: function (name, method) {
843
var m = this.settings.messages[name];
844
return m && (m.constructor === String ? m : m[method]);
845
},
846
847
// Return the first defined argument, allowing empty strings
848
findDefined: function () {
849
for (var i = 0; i < arguments.length; i++) {
850
if (arguments[i] !== undefined) {
851
return arguments[i];
852
}
853
}
854
return undefined;
855
},
856
857
// The second parameter 'rule' used to be a string, and extended to an object literal
858
// of the following form:
859
// rule = {
860
// method: "method name",
861
// parameters: "the given method parameters"
862
// }
863
//
864
// The old behavior still supported, kept to maintain backward compatibility with
865
// old code, and will be removed in the next major release.
866
defaultMessage: function (element, rule) {
867
if (typeof rule === "string") {
868
rule = {
869
method: rule
870
};
871
}
872
873
var message = this.findDefined(
874
this.customMessage(element.name, rule.method),
875
this.customDataMessage(element, rule.method),
876
877
// 'title' is never undefined, so handle empty string as undefined
878
!this.settings.ignoreTitle && element.title || undefined,
879
$.validator.messages[rule.method],
880
"<strong>Warning: No message defined for " + element.name + "</strong>"
881
),
882
theregex = /\$?\{(\d+)\}/g;
883
if (typeof message === "function") {
884
message = message.call(this, rule.parameters, element);
885
} else if (theregex.test(message)) {
886
message = $.validator.format(message.replace(theregex, "{$1}"), rule.parameters);
887
}
888
889
return message;
890
},
891
892
formatAndAdd: function (element, rule) {
893
var message = this.defaultMessage(element, rule);
894
895
this.errorList.push({
896
message: message,
897
element: element,
898
method: rule.method
899
});
900
901
this.errorMap[element.name] = message;
902
this.submitted[element.name] = message;
903
},
904
905
addWrapper: function (toToggle) {
906
if (this.settings.wrapper) {
907
toToggle = toToggle.add(toToggle.parent(this.settings.wrapper));
908
}
909
return toToggle;
910
},
911
912
defaultShowErrors: function () {
913
var i, elements, error;
914
for (i = 0; this.errorList[i]; i++) {
915
error = this.errorList[i];
916
if (this.settings.highlight) {
917
this.settings.highlight.call(this, error.element, this.settings.errorClass, this.settings.validClass);
918
}
919
this.showLabel(error.element, error.message);
920
}
921
if (this.errorList.length) {
922
this.toShow = this.toShow.add(this.containers);
923
}
924
if (this.settings.success) {
925
for (i = 0; this.successList[i]; i++) {
926
this.showLabel(this.successList[i]);
927
}
928
}
929
if (this.settings.unhighlight) {
930
for (i = 0, elements = this.validElements(); elements[i]; i++) {
931
this.settings.unhighlight.call(this, elements[i], this.settings.errorClass, this.settings.validClass);
932
}
933
}
934
this.toHide = this.toHide.not(this.toShow);
935
this.hideErrors();
936
this.addWrapper(this.toShow).show();
937
},
938
939
validElements: function () {
940
return this.currentElements.not(this.invalidElements());
941
},
942
943
invalidElements: function () {
944
return $(this.errorList).map(function () {
945
return this.element;
946
});
947
},
948
949
showLabel: function (element, message) {
950
var place, group, errorID, v,
951
error = this.errorsFor(element),
952
elementID = this.idOrName(element),
953
describedBy = $(element).attr("aria-describedby");
954
955
if (error.length) {
956
957
// Refresh error/success class
958
error.removeClass(this.settings.validClass).addClass(this.settings.errorClass);
959
960
// Replace message on existing label
961
error.html(message);
962
} else {
963
964
// Create error element
965
error = $("<" + this.settings.errorElement + ">")
966
.attr("id", elementID + "-error")
967
.addClass(this.settings.errorClass)
968
.html(message || "");
969
970
// Maintain reference to the element to be placed into the DOM
971
place = error;
972
if (this.settings.wrapper) {
973
974
// Make sure the element is visible, even in IE
975
// actually showing the wrapped element is handled elsewhere
976
place = error.hide().show().wrap("<" + this.settings.wrapper + "/>").parent();
977
}
978
if (this.labelContainer.length) {
979
this.labelContainer.append(place);
980
} else if (this.settings.errorPlacement) {
981
this.settings.errorPlacement.call(this, place, $(element));
982
} else {
983
place.insertAfter(element);
984
}
985
986
// Link error back to the element
987
if (error.is("label")) {
988
989
// If the error is a label, then associate using 'for'
990
error.attr("for", elementID);
991
992
// If the element is not a child of an associated label, then it's necessary
993
// to explicitly apply aria-describedby
994
} else if (error.parents("label[for='" + this.escapeCssMeta(elementID) + "']").length === 0) {
995
errorID = error.attr("id");
996
997
// Respect existing non-error aria-describedby
998
if (!describedBy) {
999
describedBy = errorID;
1000
} else if (!describedBy.match(new RegExp("\\b" + this.escapeCssMeta(errorID) + "\\b"))) {
1001
1002
// Add to end of list if not already present
1003
describedBy += " " + errorID;
1004
}
1005
$(element).attr("aria-describedby", describedBy);
1006
1007
// If this element is grouped, then assign to all elements in the same group
1008
group = this.groups[element.name];
1009
if (group) {
1010
v = this;
1011
$.each(v.groups, function (name, testgroup) {
1012
if (testgroup === group) {
1013
$("[name='" + v.escapeCssMeta(name) + "']", v.currentForm)
1014
.attr("aria-describedby", error.attr("id"));
1015
}
1016
});
1017
}
1018
}
1019
}
1020
if (!message && this.settings.success) {
1021
error.text("");
1022
if (typeof this.settings.success === "string") {
1023
error.addClass(this.settings.success);
1024
} else {
1025
this.settings.success(error, element);
1026
}
1027
}
1028
this.toShow = this.toShow.add(error);
1029
},
1030
1031
errorsFor: function (element) {
1032
var name = this.escapeCssMeta(this.idOrName(element)),
1033
describer = $(element).attr("aria-describedby"),
1034
selector = "label[for='" + name + "'], label[for='" + name + "'] *";
1035
1036
// 'aria-describedby' should directly reference the error element
1037
if (describer) {
1038
selector = selector + ", #" + this.escapeCssMeta(describer)
1039
.replace(/\s+/g, ", #");
1040
}
1041
1042
return this
1043
.errors()
1044
.filter(selector);
1045
},
1046
1047
// See https://api.jquery.com/category/selectors/, for CSS
1048
// meta-characters that should be escaped in order to be used with JQuery
1049
// as a literal part of a name/id or any selector.
1050
escapeCssMeta: function (string) {
1051
return string.replace(/([\\!"#$%&'()*+,./:;<=>?@\[\]^`{|}~])/g, "\\$1");
1052
},
1053
1054
idOrName: function (element) {
1055
return this.groups[element.name] || (this.checkable(element) ? element.name : element.id || element.name);
1056
},
1057
1058
validationTargetFor: function (element) {
1059
1060
// If radio/checkbox, validate first element in group instead
1061
if (this.checkable(element)) {
1062
element = this.findByName(element.name);
1063
}
1064
1065
// Always apply ignore filter
1066
return $(element).not(this.settings.ignore)[0];
1067
},
1068
1069
checkable: function (element) {
1070
return (/radio|checkbox/i).test(element.type);
1071
},
1072
1073
findByName: function (name) {
1074
return $(this.currentForm).find("[name='" + this.escapeCssMeta(name) + "']");
1075
},
1076
1077
getLength: function (value, element) {
1078
switch (element.nodeName.toLowerCase()) {
1079
case "select":
1080
return $("option:selected", element).length;
1081
case "input":
1082
if (this.checkable(element)) {
1083
return this.findByName(element.name).filter(":checked").length;
1084
}
1085
}
1086
return value.length;
1087
},
1088
1089
depend: function (param, element) {
1090
return this.dependTypes[typeof param] ? this.dependTypes[typeof param](param, element) : true;
1091
},
1092
1093
dependTypes: {
1094
"boolean": function (param) {
1095
return param;
1096
},
1097
"string": function (param, element) {
1098
return !!$(param, element.form).length;
1099
},
1100
"function": function (param, element) {
1101
return param(element);
1102
}
1103
},
1104
1105
optional: function (element) {
1106
var val = this.elementValue(element);
1107
return !$.validator.methods.required.call(this, val, element) && "dependency-mismatch";
1108
},
1109
1110
startRequest: function (element) {
1111
if (!this.pending[element.name]) {
1112
this.pendingRequest++;
1113
$(element).addClass(this.settings.pendingClass);
1114
this.pending[element.name] = true;
1115
}
1116
},
1117
1118
stopRequest: function (element, valid) {
1119
this.pendingRequest--;
1120
1121
// Sometimes synchronization fails, make sure pendingRequest is never < 0
1122
if (this.pendingRequest < 0) {
1123
this.pendingRequest = 0;
1124
}
1125
delete this.pending[element.name];
1126
$(element).removeClass(this.settings.pendingClass);
1127
if (valid && this.pendingRequest === 0 && this.formSubmitted && this.form()) {
1128
$(this.currentForm).submit();
1129
1130
// Remove the hidden input that was used as a replacement for the
1131
// missing submit button. The hidden input is added by `handle()`
1132
// to ensure that the value of the used submit button is passed on
1133
// for scripted submits triggered by this method
1134
if (this.submitButton) {
1135
$("input:hidden[name='" + this.submitButton.name + "']", this.currentForm).remove();
1136
}
1137
1138
this.formSubmitted = false;
1139
} else if (!valid && this.pendingRequest === 0 && this.formSubmitted) {
1140
$(this.currentForm).triggerHandler("invalid-form", [this]);
1141
this.formSubmitted = false;
1142
}
1143
},
1144
1145
previousValue: function (element, method) {
1146
method = typeof method === "string" && method || "remote";
1147
1148
return $.data(element, "previousValue") || $.data(element, "previousValue", {
1149
old: null,
1150
valid: true,
1151
message: this.defaultMessage(element, {
1152
method: method
1153
})
1154
});
1155
},
1156
1157
// Cleans up all forms and elements, removes validator-specific events
1158
destroy: function () {
1159
this.resetForm();
1160
1161
$(this.currentForm)
1162
.off(".validate")
1163
.removeData("validator")
1164
.find(".validate-equalTo-blur")
1165
.off(".validate-equalTo")
1166
.removeClass("validate-equalTo-blur");
1167
}
1168
1169
},
1170
1171
classRuleSettings: {
1172
required: {
1173
required: true
1174
},
1175
email: {
1176
email: true
1177
},
1178
url: {
1179
url: true
1180
},
1181
date: {
1182
date: true
1183
},
1184
dateISO: {
1185
dateISO: true
1186
},
1187
number: {
1188
number: true
1189
},
1190
digits: {
1191
digits: true
1192
},
1193
creditcard: {
1194
creditcard: true
1195
}
1196
},
1197
1198
addClassRules: function (className, rules) {
1199
if (className.constructor === String) {
1200
this.classRuleSettings[className] = rules;
1201
} else {
1202
$.extend(this.classRuleSettings, className);
1203
}
1204
},
1205
1206
classRules: function (element) {
1207
var rules = {},
1208
classes = $(element).attr("class");
1209
1210
if (classes) {
1211
$.each(classes.split(" "), function () {
1212
if (this in $.validator.classRuleSettings) {
1213
$.extend(rules, $.validator.classRuleSettings[this]);
1214
}
1215
});
1216
}
1217
return rules;
1218
},
1219
1220
normalizeAttributeRule: function (rules, type, method, value) {
1221
1222
// Convert the value to a number for number inputs, and for text for backwards compability
1223
// allows type="date" and others to be compared as strings
1224
if (/min|max|step/.test(method) && (type === null || /number|range|text/.test(type))) {
1225
value = Number(value);
1226
1227
// Support Opera Mini, which returns NaN for undefined minlength
1228
if (isNaN(value)) {
1229
value = undefined;
1230
}
1231
}
1232
1233
if (value || value === 0) {
1234
rules[method] = value;
1235
} else if (type === method && type !== "range") {
1236
1237
// Exception: the jquery validate 'range' method
1238
// does not test for the html5 'range' type
1239
rules[method] = true;
1240
}
1241
},
1242
1243
attributeRules: function (element) {
1244
var rules = {},
1245
$element = $(element),
1246
type = element.getAttribute("type"),
1247
method, value;
1248
1249
for (method in $.validator.methods) {
1250
1251
// Support for <input required> in both html5 and older browsers
1252
if (method === "required") {
1253
value = element.getAttribute(method);
1254
1255
// Some browsers return an empty string for the required attribute
1256
// and non-HTML5 browsers might have required="" markup
1257
if (value === "") {
1258
value = true;
1259
}
1260
1261
// Force non-HTML5 browsers to return bool
1262
value = !!value;
1263
} else {
1264
value = $element.attr(method);
1265
}
1266
1267
this.normalizeAttributeRule(rules, type, method, value);
1268
}
1269
1270
// 'maxlength' may be returned as -1, 2147483647 ( IE ) and 524288 ( safari ) for text inputs
1271
if (rules.maxlength && /-1|2147483647|524288/.test(rules.maxlength)) {
1272
delete rules.maxlength;
1273
}
1274
1275
return rules;
1276
},
1277
1278
dataRules: function (element) {
1279
var rules = {},
1280
$element = $(element),
1281
type = element.getAttribute("type"),
1282
method, value;
1283
1284
for (method in $.validator.methods) {
1285
value = $element.data("rule" + method.charAt(0).toUpperCase() + method.substring(1).toLowerCase());
1286
this.normalizeAttributeRule(rules, type, method, value);
1287
}
1288
return rules;
1289
},
1290
1291
staticRules: function (element) {
1292
var rules = {},
1293
validator = $.data(element.form, "validator");
1294
1295
if (validator.settings.rules) {
1296
rules = $.validator.normalizeRule(validator.settings.rules[element.name]) || {};
1297
}
1298
return rules;
1299
},
1300
1301
normalizeRules: function (rules, element) {
1302
1303
// Handle dependency check
1304
$.each(rules, function (prop, val) {
1305
1306
// Ignore rule when param is explicitly false, eg. required:false
1307
if (val === false) {
1308
delete rules[prop];
1309
return;
1310
}
1311
if (val.param || val.depends) {
1312
var keepRule = true;
1313
switch (typeof val.depends) {
1314
case "string":
1315
keepRule = !!$(val.depends, element.form).length;
1316
break;
1317
case "function":
1318
keepRule = val.depends.call(element, element);
1319
break;
1320
}
1321
if (keepRule) {
1322
rules[prop] = val.param !== undefined ? val.param : true;
1323
} else {
1324
$.data(element.form, "validator").resetElements($(element));
1325
delete rules[prop];
1326
}
1327
}
1328
});
1329
1330
// Evaluate parameters
1331
$.each(rules, function (rule, parameter) {
1332
rules[rule] = $.isFunction(parameter) && rule !== "normalizer" ? parameter(element) : parameter;
1333
});
1334
1335
// Clean number parameters
1336
$.each(["minlength", "maxlength"], function () {
1337
if (rules[this]) {
1338
rules[this] = Number(rules[this]);
1339
}
1340
});
1341
$.each(["rangelength", "range"], function () {
1342
var parts;
1343
if (rules[this]) {
1344
if ($.isArray(rules[this])) {
1345
rules[this] = [Number(rules[this][0]), Number(rules[this][1])];
1346
} else if (typeof rules[this] === "string") {
1347
parts = rules[this].replace(/[\[\]]/g, "").split(/[\s,]+/);
1348
rules[this] = [Number(parts[0]), Number(parts[1])];
1349
}
1350
}
1351
});
1352
1353
if ($.validator.autoCreateRanges) {
1354
1355
// Auto-create ranges
1356
if (rules.min != null && rules.max != null) {
1357
rules.range = [rules.min, rules.max];
1358
delete rules.min;
1359
delete rules.max;
1360
}
1361
if (rules.minlength != null && rules.maxlength != null) {
1362
rules.rangelength = [rules.minlength, rules.maxlength];
1363
delete rules.minlength;
1364
delete rules.maxlength;
1365
}
1366
}
1367
1368
return rules;
1369
},
1370
1371
// Converts a simple string to a {string: true} rule, e.g., "required" to {required:true}
1372
normalizeRule: function (data) {
1373
if (typeof data === "string") {
1374
var transformed = {};
1375
$.each(data.split(/\s/), function () {
1376
transformed[this] = true;
1377
});
1378
data = transformed;
1379
}
1380
return data;
1381
},
1382
1383
// https://jqueryvalidation.org/jQuery.validator.addMethod/
1384
addMethod: function (name, method, message) {
1385
$.validator.methods[name] = method;
1386
$.validator.messages[name] = message !== undefined ? message : $.validator.messages[name];
1387
if (method.length < 3) {
1388
$.validator.addClassRules(name, $.validator.normalizeRule(name));
1389
}
1390
},
1391
1392
// https://jqueryvalidation.org/jQuery.validator.methods/
1393
methods: {
1394
1395
// https://jqueryvalidation.org/required-method/
1396
required: function (value, element, param) {
1397
1398
// Check if dependency is met
1399
if (!this.depend(param, element)) {
1400
return "dependency-mismatch";
1401
}
1402
if (element.nodeName.toLowerCase() === "select") {
1403
1404
// Could be an array for select-multiple or a string, both are fine this way
1405
var val = $(element).val();
1406
return val && val.length > 0;
1407
}
1408
if (this.checkable(element)) {
1409
return this.getLength(value, element) > 0;
1410
}
1411
return value.length > 0;
1412
},
1413
1414
// https://jqueryvalidation.org/email-method/
1415
email: function (value, element) {
1416
1417
// From https://html.spec.whatwg.org/multipage/forms.html#valid-e-mail-address
1418
// Retrieved 2014-01-14
1419
// If you have a problem with this implementation, report a bug against the above spec
1420
// Or use custom methods to implement your own email validation
1421
return this.optional(element) || /^[a-zA-Z0-9.!#$%&'*+\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$/.test(value);
1422
},
1423
1424
// https://jqueryvalidation.org/url-method/
1425
url: function (value, element) {
1426
1427
// Copyright (c) 2010-2013 Diego Perini, MIT licensed
1428
// https://gist.github.com/dperini/729294
1429
// see also https://mathiasbynens.be/demo/url-regex
1430
// modified to allow protocol-relative URLs
1431
return this.optional(element) || /^(?:(?:(?:https?|ftp):)?\/\/)(?:\S+(?::\S*)?@)?(?:(?!(?:10|127)(?:\.\d{1,3}){3})(?!(?:169\.254|192\.168)(?:\.\d{1,3}){2})(?!172\.(?:1[6-9]|2\d|3[0-1])(?:\.\d{1,3}){2})(?:[1-9]\d?|1\d\d|2[01]\d|22[0-3])(?:\.(?:1?\d{1,2}|2[0-4]\d|25[0-5])){2}(?:\.(?:[1-9]\d?|1\d\d|2[0-4]\d|25[0-4]))|(?:(?:[a-z\u00a1-\uffff0-9]-*)*[a-z\u00a1-\uffff0-9]+)(?:\.(?:[a-z\u00a1-\uffff0-9]-*)*[a-z\u00a1-\uffff0-9]+)*(?:\.(?:[a-z\u00a1-\uffff]{2,})).?)(?::\d{2,5})?(?:[/?#]\S*)?$/i.test(value);
1432
},
1433
1434
// https://jqueryvalidation.org/date-method/
1435
date: function (value, element) {
1436
return this.optional(element) || !/Invalid|NaN/.test(new Date(value).toString());
1437
},
1438
1439
// https://jqueryvalidation.org/dateISO-method/
1440
dateISO: function (value, element) {
1441
return this.optional(element) || /^\d{4}[\/\-](0?[1-9]|1[012])[\/\-](0?[1-9]|[12][0-9]|3[01])$/.test(value);
1442
},
1443
1444
// https://jqueryvalidation.org/number-method/
1445
number: function (value, element) {
1446
return this.optional(element) || /^(?:-?\d+|-?\d{1,3}(?:,\d{3})+)?(?:\.\d+)?$/.test(value);
1447
},
1448
1449
// https://jqueryvalidation.org/digits-method/
1450
digits: function (value, element) {
1451
return this.optional(element) || /^\d+$/.test(value);
1452
},
1453
1454
// https://jqueryvalidation.org/minlength-method/
1455
minlength: function (value, element, param) {
1456
var length = $.isArray(value) ? value.length : this.getLength(value, element);
1457
return this.optional(element) || length >= param;
1458
},
1459
1460
// https://jqueryvalidation.org/maxlength-method/
1461
maxlength: function (value, element, param) {
1462
var length = $.isArray(value) ? value.length : this.getLength(value, element);
1463
return this.optional(element) || length <= param;
1464
},
1465
1466
// https://jqueryvalidation.org/rangelength-method/
1467
rangelength: function (value, element, param) {
1468
var length = $.isArray(value) ? value.length : this.getLength(value, element);
1469
return this.optional(element) || (length >= param[0] && length <= param[1]);
1470
},
1471
1472
// https://jqueryvalidation.org/min-method/
1473
min: function (value, element, param) {
1474
return this.optional(element) || value >= param;
1475
},
1476
1477
// https://jqueryvalidation.org/max-method/
1478
max: function (value, element, param) {
1479
return this.optional(element) || value <= param;
1480
},
1481
1482
// https://jqueryvalidation.org/range-method/
1483
range: function (value, element, param) {
1484
return this.optional(element) || (value >= param[0] && value <= param[1]);
1485
},
1486
1487
// https://jqueryvalidation.org/step-method/
1488
step: function (value, element, param) {
1489
var type = $(element).attr("type"),
1490
errorMessage = "Step attribute on input type " + type + " is not supported.",
1491
supportedTypes = ["text", "number", "range"],
1492
re = new RegExp("\\b" + type + "\\b"),
1493
notSupported = type && !re.test(supportedTypes.join()),
1494
decimalPlaces = function (num) {
1495
var match = ("" + num).match(/(?:\.(\d+))?$/);
1496
if (!match) {
1497
return 0;
1498
}
1499
1500
// Number of digits right of decimal point.
1501
return match[1] ? match[1].length : 0;
1502
},
1503
toInt = function (num) {
1504
return Math.round(num * Math.pow(10, decimals));
1505
},
1506
valid = true,
1507
decimals;
1508
1509
// Works only for text, number and range input types
1510
// TODO find a way to support input types date, datetime, datetime-local, month, time and week
1511
if (notSupported) {
1512
throw new Error(errorMessage);
1513
}
1514
1515
decimals = decimalPlaces(param);
1516
1517
// Value can't have too many decimals
1518
if (decimalPlaces(value) > decimals || toInt(value) % toInt(param) !== 0) {
1519
valid = false;
1520
}
1521
1522
return this.optional(element) || valid;
1523
},
1524
1525
// https://jqueryvalidation.org/equalTo-method/
1526
equalTo: function (value, element, param) {
1527
1528
// Bind to the blur event of the target in order to revalidate whenever the target field is updated
1529
var target = $(param);
1530
if (this.settings.onfocusout && target.not(".validate-equalTo-blur").length) {
1531
target.addClass("validate-equalTo-blur").on("blur.validate-equalTo", function () {
1532
$(element).valid();
1533
});
1534
}
1535
return value === target.val();
1536
},
1537
1538
// https://jqueryvalidation.org/remote-method/
1539
remote: function (value, element, param, method) {
1540
if (this.optional(element)) {
1541
return "dependency-mismatch";
1542
}
1543
1544
method = typeof method === "string" && method || "remote";
1545
1546
var previous = this.previousValue(element, method),
1547
validator, data, optionDataString;
1548
1549
if (!this.settings.messages[element.name]) {
1550
this.settings.messages[element.name] = {};
1551
}
1552
previous.originalMessage = previous.originalMessage || this.settings.messages[element.name][method];
1553
this.settings.messages[element.name][method] = previous.message;
1554
1555
param = typeof param === "string" && {
1556
url: param
1557
} || param;
1558
optionDataString = $.param($.extend({
1559
data: value
1560
}, param.data));
1561
if (previous.old === optionDataString) {
1562
return previous.valid;
1563
}
1564
1565
previous.old = optionDataString;
1566
validator = this;
1567
this.startRequest(element);
1568
data = {};
1569
data[element.name] = value;
1570
$.ajax($.extend(true, {
1571
mode: "abort",
1572
port: "validate" + element.name,
1573
dataType: "json",
1574
data: data,
1575
context: validator.currentForm,
1576
success: function (response) {
1577
var valid = response === true || response === "true",
1578
errors, message, submitted;
1579
1580
validator.settings.messages[element.name][method] = previous.originalMessage;
1581
if (valid) {
1582
submitted = validator.formSubmitted;
1583
validator.resetInternals();
1584
validator.toHide = validator.errorsFor(element);
1585
validator.formSubmitted = submitted;
1586
validator.successList.push(element);
1587
validator.invalid[element.name] = false;
1588
validator.showErrors();
1589
} else {
1590
errors = {};
1591
message = response || validator.defaultMessage(element, {
1592
method: method,
1593
parameters: value
1594
});
1595
errors[element.name] = previous.message = message;
1596
validator.invalid[element.name] = true;
1597
validator.showErrors(errors);
1598
}
1599
previous.valid = valid;
1600
validator.stopRequest(element, valid);
1601
}
1602
}, param));
1603
return "pending";
1604
}
1605
}
1606
1607
});
1608
1609
// Ajax mode: abort
1610
// usage: $.ajax({ mode: "abort"[, port: "uniqueport"]});
1611
// if mode:"abort" is used, the previous request on that port (port can be undefined) is aborted via XMLHttpRequest.abort()
1612
1613
var pendingRequests = {},
1614
ajax;
1615
1616
// Use a prefilter if available (1.5+)
1617
if ($.ajaxPrefilter) {
1618
$.ajaxPrefilter(function (settings, _, xhr) {
1619
var port = settings.port;
1620
if (settings.mode === "abort") {
1621
if (pendingRequests[port]) {
1622
pendingRequests[port].abort();
1623
}
1624
pendingRequests[port] = xhr;
1625
}
1626
});
1627
} else {
1628
1629
// Proxy ajax
1630
ajax = $.ajax;
1631
$.ajax = function (settings) {
1632
var mode = ("mode" in settings ? settings : $.ajaxSettings).mode,
1633
port = ("port" in settings ? settings : $.ajaxSettings).port;
1634
if (mode === "abort") {
1635
if (pendingRequests[port]) {
1636
pendingRequests[port].abort();
1637
}
1638
pendingRequests[port] = ajax.apply(this, arguments);
1639
return pendingRequests[port];
1640
}
1641
return ajax.apply(this, arguments);
1642
};
1643
}
1644
return $;
1645
}));
1646
1647
/* Materializecss Stepper - By Kinark 2016
1648
// https://github.com/Kinark/Materialize-stepper
1649
// JS v2.1.3
1650
*/
1651
1652
let validation = $.isFunction($.fn.valid) ? 1 : 0;
1653
1654
$.fn.isValid = function () {
1655
if (validation) {
1656
return this.valid();
1657
}
1658
return true;
1659
1660
};
1661
1662
if (validation) {
1663
$.validator.setDefaults({
1664
errorClass: 'invalid',
1665
validClass: 'valid',
1666
errorPlacement (error, element) {
1667
if (element.is(':radio') || element.is(':checkbox')) {
1668
error.insertBefore($(element).parent());
1669
} else {
1670
error.insertAfter(element); // default error placement.
1671
// element.closest('label').data('error', error);
1672
// element.next().attr('data-error', error);
1673
}
1674
},
1675
success (element) {
1676
if (!$(element).closest('li').find('label.invalid:not(:empty)').length) {
1677
$(element).closest('li').removeClass('wrong');
1678
}
1679
}
1680
});
1681
1682
// When parallel stepper is defined we need to consider invisible and
1683
// hidden fields
1684
if ($('.stepper.parallel').length) {$.validator.setDefaults({
1685
ignore: ''
1686
});}
1687
}
1688
1689
$.fn.getActiveStep = function () {
1690
let active = this.find('.step.active');
1691
return $(this.children('.step:visible')).index($(active)) + 1;
1692
};
1693
1694
$.fn.activateStep = function (callback) {
1695
if ($(this).hasClass('step')) {return;}
1696
let stepper = $(this).closest('ul.stepper');
1697
stepper.find('>li').removeAttr('data-last');
1698
if (window.innerWidth < 993 || !stepper.hasClass('horizontal')) {
1699
$(this).addClass('step').stop().slideDown(400, function () {
1700
$(this).css({
1701
height: 'auto',
1702
'margin-bottom': '',
1703
display: 'inherit'
1704
});
1705
if (callback) {callback();}
1706
stepper.find('>li.step').last().attr('data-last', 'true');
1707
1708
});
1709
} else {
1710
$(this).addClass('step').stop().css({
1711
width: '0%',
1712
display: 'inherit'
1713
}).animate({
1714
width: '100%'
1715
}, 400, function () {
1716
$(this).css({
1717
height: 'auto',
1718
'margin-bottom': '',
1719
display: 'inherit'
1720
});
1721
if (callback) {callback();}
1722
stepper.find('>li.step').last().attr('data-last', 'true');
1723
});
1724
}
1725
};
1726
1727
$.fn.deactivateStep = function (callback) {
1728
if (!$(this).hasClass('step')) {return;}
1729
let stepper = $(this).closest('ul.stepper');
1730
stepper.find('>li').removeAttr('data-last');
1731
if (window.innerWidth < 993 || !stepper.hasClass('horizontal')) {
1732
$(this).stop().css({
1733
transition: 'none',
1734
'-webkit-transition': 'margin-bottom none'
1735
}).slideUp(400, function () {
1736
$(this).removeClass('step').css({
1737
height: 'auto',
1738
'margin-bottom': '',
1739
transition: 'margin-bottom .4s',
1740
'-webkit-transition': 'margin-bottom .4s'
1741
});
1742
if (callback) {callback();}
1743
stepper.find('>li').removeAttr('data-last');
1744
stepper.find('>li.step').last().attr('data-last', 'true');
1745
});
1746
} else {
1747
$(this).stop().animate({
1748
width: '0%'
1749
}, 400, function () {
1750
$(this).removeClass('step').hide().css({
1751
height: 'auto',
1752
'margin-bottom': '',
1753
display: 'none',
1754
width: ''
1755
});
1756
if (callback) {callback();}
1757
stepper.find('>li.step').last().attr('data-last', 'true');
1758
});
1759
}
1760
};
1761
1762
$.fn.showError = function (error) {
1763
if (validation) {
1764
let name = this.attr('name');
1765
let form = this.closest('form');
1766
let obj = {};
1767
obj[name] = error;
1768
form.validate().showErrors(obj);
1769
this.closest('li').addClass('wrong');
1770
} else {
1771
this.removeClass('valid').addClass('invalid');
1772
this.next().attr('data-error', error);
1773
}
1774
};
1775
1776
$.fn.activateFeedback = function () {
1777
let active = this.find('.step.active:not(.feedbacking)').addClass('feedbacking').find('.step-new-content');
1778
active.prepend('<div class="wait-feedback"> <div class="preloader-wrapper active"> <div class="spinner-layer spinner-blue"> <div class="circle-clipper left"> <div class="circle"></div></div><div class="gap-patch"> <div class="circle"></div></div><div class="circle-clipper right"> <div class="circle"></div></div></div><div class="spinner-layer spinner-red"> <div class="circle-clipper left"> <div class="circle"></div></div><div class="gap-patch"> <div class="circle"></div></div><div class="circle-clipper right"> <div class="circle"></div></div></div><div class="spinner-layer spinner-yellow"> <div class="circle-clipper left"> <div class="circle"></div></div><div class="gap-patch"> <div class="circle"></div></div><div class="circle-clipper right"> <div class="circle"></div></div></div><div class="spinner-layer spinner-green"> <div class="circle-clipper left"> <div class="circle"></div></div><div class="gap-patch"> <div class="circle"></div></div><div class="circle-clipper right"> <div class="circle"></div></div></div></div></div>');
1779
};
1780
1781
$.fn.destroyFeedback = function () {
1782
let active = this.find('.step.active.feedbacking');
1783
if (active) {
1784
active.removeClass('feedbacking');
1785
active.find('.wait-feedback').remove();
1786
}
1787
return true;
1788
};
1789
1790
$.fn.resetStepper = function (step) {
1791
if (!step) {step = 1;}
1792
let form = $(this).closest('form');
1793
$(form)[0].reset();
1794
Materialize.updateTextFields();
1795
1796
if (validation) {
1797
form.validate().resetForm();
1798
$(this).find('li.step').removeClass('wrong');
1799
}
1800
1801
return $(this).openStep(step);
1802
};
1803
1804
$.fn.submitStepper = function (step) {
1805
let form = this.closest('form');
1806
if (form.isValid()) {
1807
form.submit();
1808
}
1809
};
1810
1811
$.fn.nextStep = function (callback, activefb, e) {
1812
var stepper = this;
1813
var settings = $(stepper).data('settings');
1814
var form = this.closest('form');
1815
var active = this.find('.step.active');
1816
var next = $(this.children('.step:visible')).index($(active)) + 2;
1817
var feedback = active.find('.next-step').length > 1 ? (e ? $(e.target).data("feedback") : undefined) : active.find('.next-step').data("feedback");
1818
// If the stepper is parallel, we want to validate the input of the current active step. Not all elements.
1819
if ((settings.parallel && $(active).validateStep()) || (!settings.parallel && form.isValid())) {
1820
if (feedback && activefb) {
1821
if (settings.showFeedbackLoader) stepper.activateFeedback();
1822
return window[feedback].call();
1823
}
1824
active.removeClass('wrong').addClass('done');
1825
this.openStep(next, callback);
1826
return this.trigger('nextstep');
1827
}
1828
return active.removeClass('done').addClass('wrong');
1829
1830
};
1831
1832
$.fn.prevStep = function (callback) {
1833
let active = this.find('.step.active');
1834
if (active.hasClass('feedbacking')) {return;}
1835
let prev = $(this.children('.step:visible')).index($(active));
1836
active.removeClass('wrong');
1837
this.openStep(prev, callback);
1838
return this.trigger('prevstep');
1839
};
1840
1841
$.fn.openStep = function (step, callback) {
1842
let settings = $(this).closest('ul.stepper').data('settings');
1843
let $this = this;
1844
let step_num = step - 1;
1845
step = this.find(`.step:visible:eq(${ step_num })`);
1846
if (step.hasClass('active')) {return;}
1847
let active = this.find('.step.active');
1848
let next;
1849
let prev_active = next = $(this.children('.step:visible')).index($(active));
1850
let order = step_num > prev_active ? 1 : 0;
1851
if (active.hasClass('feedbacking')) {$this.destroyFeedback();}
1852
active.closeAction(order);
1853
step.openAction(order, () => {
1854
if (settings.autoFocusInput) step.find('input:enabled:visible:first').focus();
1855
$this.trigger('stepchange').trigger('step' + (step_num + 1));
1856
if (step.data('event')) $this.trigger(step.data('event'));
1857
if (callback) callback();
1858
});
1859
};
1860
1861
$.fn.closeAction = function (order, callback) {
1862
let closable = this.removeClass('active').find('.step-new-content');
1863
if (window.innerWidth < 993 || !this.closest('ul').hasClass('horizontal')) {
1864
closable.stop().slideUp(300, 'easeOutQuad', callback);
1865
} else if (order == 1) {
1866
closable.animate({
1867
left: '-100%'
1868
}, function () {
1869
closable.css({
1870
display: 'none',
1871
left: '0%'
1872
}, callback);
1873
});
1874
} else {
1875
closable.animate({
1876
left: '100%'
1877
}, function () {
1878
closable.css({
1879
display: 'none',
1880
left: '0%'
1881
}, callback);
1882
});
1883
}
1884
1885
if (this.closest('ul').hasClass('horizontal') && this.closest('ul').hasClass('horizontal-fix')) {
1886
let bodyHeight = $('.step-new-content').children().height() + 250;
1887
$(this).parent().animate({
1888
height: bodyHeight
1889
}, 'slow');
1890
}
1891
};
1892
1893
$.fn.openAction = function (order, callback) {
1894
let openable = this.removeClass('done').addClass('active').find('.step-new-content');
1895
if (window.innerWidth < 993 || !this.closest('ul').hasClass('horizontal')) {
1896
openable.slideDown(300, 'easeOutQuad', callback);
1897
} else if (order == 1) {
1898
openable.css({
1899
left: '100%',
1900
display: 'block'
1901
}).animate({
1902
left: '0%'
1903
}, callback);
1904
} else {
1905
openable.css({
1906
left: '-100%',
1907
display: 'block'
1908
}).animate({
1909
left: '0%'
1910
}, callback);
1911
}
1912
1913
if (this.closest('ul').hasClass('horizontal') && this.closest('ul').hasClass('horizontal-fix')) {
1914
let bodyHeight = $(this).find('.step-new-content').children().height() + 250;
1915
$(this).parent().animate({
1916
height: bodyHeight
1917
}, 'fast');
1918
}
1919
};
1920
1921
$.fn.mdbStepper = function (options) {
1922
let settings = $.extend({
1923
linearStepsNavigation: true,
1924
autoFocusInput: true,
1925
showFeedbackLoader: true,
1926
autoFormCreation: true,
1927
parallel: false // By default we don't assume the stepper is parallel
1928
}, options);
1929
$(document).on('click', (e) => {
1930
if (!$(e.target).parents(".stepper").length) {
1931
$('.stepper.focused').removeClass('focused');
1932
}
1933
});
1934
1935
$(this).each(function () {
1936
let $stepper = $(this);
1937
if (!$stepper.parents('form').length && settings.autoFormCreation) {
1938
var method = $stepper.data('method');
1939
let action = $stepper.data('action');
1940
var method = method ? method : "GET";
1941
action = action ? action : "?";
1942
$stepper.wrap(`<form action="${ action }" method="${ method }"></form>`);
1943
}
1944
1945
$stepper.data('settings', {
1946
linearStepsNavigation: settings.linearStepsNavigation,
1947
autoFocusInput: settings.autoFocusInput,
1948
showFeedbackLoader: settings.showFeedbackLoader,
1949
parallel: $stepper.hasClass('parallel')
1950
});
1951
$stepper.find('li.step.active').openAction(1);
1952
$stepper.find('>li').removeAttr('data-last');
1953
$stepper.find('>li.step').last().attr('data-last', 'true');
1954
1955
$stepper.on('click', '.step:not(.active)', function () {
1956
let object = $($stepper.children('.step:visible')).index($(this));
1957
if ($stepper.data('settings').parallel && validation) { // Invoke parallel stepper behaviour
1958
$(this).addClass('temp-active');
1959
$stepper.validatePreviousSteps();
1960
$stepper.openStep(object + 1);
1961
$(this).removeClass('temp-active');
1962
} else if (!$stepper.hasClass('linear')) {
1963
$stepper.openStep(object + 1);
1964
} else if (settings.linearStepsNavigation) {
1965
let active = $stepper.find('.step.active');
1966
if ($($stepper.children('.step:visible')).index($(active)) + 1 == object) {
1967
$stepper.nextStep(undefined, true, undefined);
1968
} else if ($($stepper.children('.step:visible')).index($(active)) - 1 == object) {
1969
$stepper.prevStep(undefined);
1970
}
1971
}
1972
}).on('click', '.next-step', (e) => {
1973
e.preventDefault();
1974
$stepper.nextStep(undefined, true, e);
1975
}).on('click', '.previous-step', (e) => {
1976
e.preventDefault();
1977
$stepper.prevStep(undefined);
1978
}).on('click', 'button:submit:not(.next-step, .previous-step)', (e) => {
1979
e.preventDefault();
1980
feedback = e ? $(e.target).data("feedback") : undefined;
1981
var form = $stepper.closest('form');
1982
if (form.isValid()) {
1983
if (feedback) {
1984
stepper.activateFeedback();
1985
return window[feedback].call();
1986
}
1987
form.submit();
1988
}
1989
}).on('click', function () {
1990
$('.stepper.focused').removeClass('focused');
1991
$(this).addClass('focused');
1992
});
1993
});
1994
};
1995
1996
/**
1997
* Return the step element on given index.
1998
*
1999
* @param step, index of the step to be returned
2000
* @returns {*}, the step requested
2001
*/
2002
$.fn.getStep = function (step) {
2003
let settings = $(this).closest('ul.stepper').data('settings');
2004
let $this = this;
2005
let step_num = step - 1;
2006
step = this.find(`.step:visible:eq(${ step_num })`);
2007
return step;
2008
};
2009
2010
/**
2011
* Run validation over all previous steps from the steps this
2012
* function is called on.
2013
*/
2014
$.fn.validatePreviousSteps = function () {
2015
let active = $(this).find('.step.temp-active');
2016
let index = $(this.children('.step')).index($(active));
2017
// We assume that the validator is set to ignore nothing.
2018
$(this.children('.step')).each(function (i) {
2019
if (i >= index) {
2020
$(this).removeClass('wrong done');
2021
} else {
2022
$(this).validateStep();
2023
}
2024
});
2025
};
2026
2027
/**
2028
* Validate the step that this function is called on.
2029
*/
2030
$.fn.validateStep = function () {
2031
var stepper = this.closest('ul.stepper');
2032
var form = this.closest('form');
2033
var step = $(this);
2034
// Retrieve the custom validator for that step if exists.
2035
var validator = step.find('.next-step').data("validator");
2036
if (this.validateStepInput()) { // If initial base validation succeeded go on
2037
if (validator) { // If a custom validator is given also call that validator
2038
if (window[validator].call()) {
2039
step.removeClass('wrong').addClass('done');
2040
return true;
2041
} else {
2042
step.removeClass('done').addClass('wrong');
2043
return false;
2044
}
2045
}
2046
step.removeClass('wrong').addClass('done');
2047
return true;
2048
}
2049
step.removeClass('done').addClass('wrong');
2050
return false;
2051
2052
};
2053
2054
/**
2055
* Uses the validation variable set by the stepper constructor
2056
* to run standard validation on the current step.
2057
* @returns {boolean}
2058
*/
2059
$.fn.validateStepInput = function () {
2060
let valid = true;
2061
if (validation) {
2062
// Find all input fields dat need validation in current step.
2063
$(this).find('input.validate').each(function () {
2064
if (!$(this).valid()) {
2065
valid = false;
2066
return false;
2067
}
2068
});
2069
}
2070
return valid;
2071
};
2072
2073
Console errors: 0