-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathjquery.form-dependencies.js
167 lines (111 loc) · 4.41 KB
/
jquery.form-dependencies.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
/*
jQuery Form Dependencies v2.0
http://github.com/digitalnature/Form-Dependencies
*/
;(function($, window, document, undefined){
$.fn.FormDependencies = function(opts){
var defaults = {
// the attribute which contains the rules
ruleAttr : 'data-depends-on',
// if given, this class will be applied to disabled elements
inactiveClass : false,
// clears input values from disabled fields
clearValues : false,
// attribute used to identify dependencies
identifyBy : 'name'
},
opts = $.extend(defaults, opts),
disable = function(e){
if(!$(e).is(':input') && !$(e).hasClass('disabled'))
$(e).addClass('disabled');
if(!e.disabled){
e.disabled = true;
$('label[for="' + e.id + '"]').addClass('disabled');
if(opts.inactiveClass)
$(e, 'label[for="' + e.id + '"]').addClass(opts.inactiveClass);
// we don't want to "clear" submit buttons
if(opts.clearValues && !$(e).is(':submit'))
if($(e).is(':checkbox, :radio')) e.checked = false; else if(!$(e).is('select')) $(e).val('');
}
},
enable = function(e){
if(!$(e).is(':input') && $(e).hasClass('disabled'))
$(e).removeClass('disabled');
if(e.disabled){
e.disabled = false;
$('label[for="' + e.id + '"]').removeClass('disabled');
if(opts.inactiveClass || !$(e).is(':visible'))
$(e, 'label[for="' + e.id + '"]').removeClass(opts.inactiveClass);
}
},
// verifies if conditions are met
matches = function(key, values, block){
var i, v, invert = false, e = $('[' + opts.identifyBy + '="' + key + '"]', block);
e = e.is(':radio') ? e.filter(':checked') : e.filter('[type!="hidden"]')
for(i = 0; i < values.length; i++){
v = values[i];
invert = false;
if(v[0] === '!'){
invert = true;
v = v.substr(1);
}
if((e.val() == v) || (!v && e.is(':checked')) || ((e.is(':submit') || e.is(':button')) && !e.is(':disabled')))
return !invert;
}
return invert;
},
split = function(str, chr){
return $.map(str.split(chr), $.trim);
};
return this.each(function(){
var block = this, rules = [], keys = [];
// parse rules
$('[' + opts.ruleAttr + ']', this).each(function(){
var deps = $(this).attr(opts.ruleAttr), dep, values, parsedDeps = {}, i, invert;
if(!deps)
return this;
deps = split(deps, '+');
for(i = 0; i < deps.length; i++){
dep = deps[i];
invert = false;
// reverse conditional check if the name starts with '!'
// the rules should have any values specified in this case
if(dep[0] === '!'){
dep = dep.substr(1);
invert = true;
}
dep = split(dep, ':');
values = dep[1] || '';
if(!values && invert)
values = '!';
parsedDeps[dep[0]] = split(values, '|');
// store dep. elements in a separate array
$('[' + opts.identifyBy + '="' + dep[0] + '"]', block).filter('[type!="hidden"]').each(function(){
($.inArray(this, keys) !== -1) || keys.push(this);
parsedDeps[dep[0]].target = this;
});
}
rules.push({target: this, deps: parsedDeps});
});
if(!keys.length)
return this;
// attach our state checking function on keys (ie. elements on which other elements depend on)
$(keys).on('change.FormDependencies keyup', function(event){
// iterate trough all rules
$.each(rules, function(input, inputRules){
var hideIt = false;
$.each(inputRules.deps, function(key, values){
// we check only if a condition fails,
// in which case we know we need to disable the hooked element
if(!matches(key, values, block)){
hideIt = true;
return false;
}
});
hideIt ? disable(inputRules.target) : enable(inputRules.target);
});
}).trigger('change.FormDependencies');
return this;
});
};
})(jQuery, window, document);