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