-
Notifications
You must be signed in to change notification settings - Fork 83
/
Filter.js
93 lines (93 loc) · 2.87 KB
/
Filter.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
define(['dojo/_base/declare'], function (declare) {
// a Filter builder
function filterCreator(type) {
// constructs a new filter based on type, used to create each comparison method
return function newFilter() {
var Filter = this.constructor;
var filter = new Filter();
filter.type = type;
// ensure args is array so we can concat, slice, unshift
filter.args = Array.prototype.slice.call(arguments);
if (this.type) {
// we are chaining, so combine with an and operator
return filterCreator('and').call(Filter.prototype, this, filter);
}
return filter;
};
}
function logicalOperatorCreator(type) {
// constructs a new logical operator 'filter', used to create each logical operation method
return function newLogicalOperator() {
var Filter = this.constructor;
var argsArray = [];
for (var i = 0; i < arguments.length; i++) {
var arg = arguments[i];
argsArray.push(arg instanceof Filter ? arg : new Filter(arg));
}
var filter = new Filter();
filter.type = type;
filter.args = argsArray;
if (this.type === type) {
// chaining, same type
// combine arguments
filter.args = this.args.concat(argsArray);
} else if (this.type) {
// chaining, different type
// add this filter to start of arguments
argsArray.unshift(this);
} else if (argsArray.length === 1) {
// not chaining and only one argument
// returned filter is the same as the single argument
if (argsArray[0].type) {
filter.type = argsArray[0].type;
filter.args = argsArray[0].args.slice();
}
}
return filter;
};
}
var Filter = declare(null, {
constructor: function (filterArg) {
var argType = typeof filterArg;
switch (argType) {
case 'object':
var filter = this;
// construct a filter based on the query object
for (var key in filterArg){
var value = filterArg[key];
if (value instanceof this.constructor) {
// fully construct the filter from the single arg
filter = filter[value.type](key, value.args[0]);
} else if (value && value.test) {
// support regex
filter = filter.match(key, value);
} else {
filter = filter.eq(key, value);
}
}
this.type = filter.type;
this.args = filter.args;
break;
case 'function': case 'string':
// allow string and function args as well
this.type = argType;
this.args = [filterArg];
}
},
// define our operators
and: logicalOperatorCreator('and'),
or: logicalOperatorCreator('or'),
eq: filterCreator('eq'),
ne: filterCreator('ne'),
lt: filterCreator('lt'),
lte: filterCreator('lte'),
gt: filterCreator('gt'),
gte: filterCreator('gte'),
contains: filterCreator('contains'),
'in': filterCreator('in'),
match: filterCreator('match')
});
Filter.filterCreator = filterCreator;
Filter.logicalOperatorCreator = logicalOperatorCreator;
return Filter;
});