22
22
* - `Object`: A pattern object can be used to filter specific properties on objects contained
23
23
* by `array`. For example `{name:"M", phone:"1"}` predicate will return an array of items
24
24
* which have property `name` containing "M" and property `phone` containing "1". A special
25
- * property name `$` can be used (as in `{$:"text"}`) to accept a match against any
26
- * property of the object or its nested object properties. That's equivalent to the simple
27
- * substring match with a `string` as described above. The predicate can be negated by prefixing
28
- * the string with `!`.
25
+ * property name (`$` by default) can be used (e.g. as in `{$: "text"}`) to accept a match
26
+ * against any property of the object or its nested object properties. That's equivalent to the
27
+ * simple substring match with a `string` as described above. The special property name can be
28
+ * overwritten, using the `anyPropertyKey` parameter.
29
+ * The predicate can be negated by prefixing the string with `!`.
29
30
* For example `{name: "!M"}` predicate will return an array of items which have property `name`
30
31
* not containing "M".
31
32
*
59
60
* Primitive values are converted to strings. Objects are not compared against primitives,
60
61
* unless they have a custom `toString` method (e.g. `Date` objects).
61
62
*
63
+ * @param {string= } anyPropertyKey The special property name that matches against any property.
64
+ * By default `$`.
65
+ *
62
66
* @example
63
67
<example>
64
68
<file name="index.html">
127
131
</file>
128
132
</example>
129
133
*/
134
+
130
135
function filterFilter ( ) {
131
- return function ( array , expression , comparator ) {
136
+ return function ( array , expression , comparator , anyPropertyKey ) {
132
137
if ( ! isArrayLike ( array ) ) {
133
138
if ( array == null ) {
134
139
return array ;
@@ -137,6 +142,7 @@ function filterFilter() {
137
142
}
138
143
}
139
144
145
+ anyPropertyKey = anyPropertyKey || '$' ;
140
146
var expressionType = getTypeForFilter ( expression ) ;
141
147
var predicateFn ;
142
148
var matchAgainstAnyProp ;
@@ -153,7 +159,7 @@ function filterFilter() {
153
159
//jshint -W086
154
160
case 'object' :
155
161
//jshint +W086
156
- predicateFn = createPredicateFn ( expression , comparator , matchAgainstAnyProp ) ;
162
+ predicateFn = createPredicateFn ( expression , comparator , anyPropertyKey , matchAgainstAnyProp ) ;
157
163
break ;
158
164
default :
159
165
return array ;
@@ -164,8 +170,8 @@ function filterFilter() {
164
170
}
165
171
166
172
// Helper functions for `filterFilter`
167
- function createPredicateFn ( expression , comparator , matchAgainstAnyProp ) {
168
- var shouldMatchPrimitives = isObject ( expression ) && ( '$' in expression ) ;
173
+ function createPredicateFn ( expression , comparator , anyPropertyKey , matchAgainstAnyProp ) {
174
+ var shouldMatchPrimitives = isObject ( expression ) && ( anyPropertyKey in expression ) ;
169
175
var predicateFn ;
170
176
171
177
if ( comparator === true ) {
@@ -193,25 +199,25 @@ function createPredicateFn(expression, comparator, matchAgainstAnyProp) {
193
199
194
200
predicateFn = function ( item ) {
195
201
if ( shouldMatchPrimitives && ! isObject ( item ) ) {
196
- return deepCompare ( item , expression . $ , comparator , false ) ;
202
+ return deepCompare ( item , expression [ anyPropertyKey ] , comparator , anyPropertyKey , false ) ;
197
203
}
198
- return deepCompare ( item , expression , comparator , matchAgainstAnyProp ) ;
204
+ return deepCompare ( item , expression , comparator , anyPropertyKey , matchAgainstAnyProp ) ;
199
205
} ;
200
206
201
207
return predicateFn ;
202
208
}
203
209
204
- function deepCompare ( actual , expected , comparator , matchAgainstAnyProp , dontMatchWholeObject ) {
210
+ function deepCompare ( actual , expected , comparator , anyPropertyKey , matchAgainstAnyProp , dontMatchWholeObject ) {
205
211
var actualType = getTypeForFilter ( actual ) ;
206
212
var expectedType = getTypeForFilter ( expected ) ;
207
213
208
214
if ( ( expectedType === 'string' ) && ( expected . charAt ( 0 ) === '!' ) ) {
209
- return ! deepCompare ( actual , expected . substring ( 1 ) , comparator , matchAgainstAnyProp ) ;
215
+ return ! deepCompare ( actual , expected . substring ( 1 ) , comparator , anyPropertyKey , matchAgainstAnyProp ) ;
210
216
} else if ( isArray ( actual ) ) {
211
217
// In case `actual` is an array, consider it a match
212
218
// if ANY of it's items matches `expected`
213
219
return actual . some ( function ( item ) {
214
- return deepCompare ( item , expected , comparator , matchAgainstAnyProp ) ;
220
+ return deepCompare ( item , expected , comparator , anyPropertyKey , matchAgainstAnyProp ) ;
215
221
} ) ;
216
222
}
217
223
@@ -220,21 +226,21 @@ function deepCompare(actual, expected, comparator, matchAgainstAnyProp, dontMatc
220
226
var key ;
221
227
if ( matchAgainstAnyProp ) {
222
228
for ( key in actual ) {
223
- if ( ( key . charAt ( 0 ) !== '$' ) && deepCompare ( actual [ key ] , expected , comparator , true ) ) {
229
+ if ( ( key . charAt ( 0 ) !== '$' ) && deepCompare ( actual [ key ] , expected , comparator , anyPropertyKey , true ) ) {
224
230
return true ;
225
231
}
226
232
}
227
- return dontMatchWholeObject ? false : deepCompare ( actual , expected , comparator , false ) ;
233
+ return dontMatchWholeObject ? false : deepCompare ( actual , expected , comparator , anyPropertyKey , false ) ;
228
234
} else if ( expectedType === 'object' ) {
229
235
for ( key in expected ) {
230
236
var expectedVal = expected [ key ] ;
231
237
if ( isFunction ( expectedVal ) || isUndefined ( expectedVal ) ) {
232
238
continue ;
233
239
}
234
240
235
- var matchAnyProperty = key === '$' ;
241
+ var matchAnyProperty = key === anyPropertyKey ;
236
242
var actualVal = matchAnyProperty ? actual : actual [ key ] ;
237
- if ( ! deepCompare ( actualVal , expectedVal , comparator , matchAnyProperty , matchAnyProperty ) ) {
243
+ if ( ! deepCompare ( actualVal , expectedVal , comparator , anyPropertyKey , matchAnyProperty , matchAnyProperty ) ) {
238
244
return false ;
239
245
}
240
246
}
0 commit comments