From 6a0aa74726b17cd7d7ea1c05543583b94dae7f2b Mon Sep 17 00:00:00 2001 From: Yuri S Date: Tue, 28 Nov 2017 10:31:07 +0200 Subject: [PATCH 1/3] Fixed search for invalid label and/or value --- src/utils/defaultFilterOptions.js | 31 +++++++++++++++++++++---------- test/Select-test.js | 24 ++++++++++++++++++++++-- 2 files changed, 43 insertions(+), 12 deletions(-) diff --git a/src/utils/defaultFilterOptions.js b/src/utils/defaultFilterOptions.js index f3a153d403..cffeeb4bcd 100644 --- a/src/utils/defaultFilterOptions.js +++ b/src/utils/defaultFilterOptions.js @@ -20,24 +20,35 @@ function filterOptions (options, filterValue, excludeOptions, props) { if (excludeOptions && excludeOptions.indexOf(option[props.valueKey]) > -1) return false; if (props.filterOption) return props.filterOption.call(this, option, filterValue); if (!filterValue) return true; - var valueTest = String(option[props.valueKey]); - var labelTest = String(option[props.labelKey]); + + var value = option[props.valueKey]; + var label = option[props.labelKey]; + var hasValue = typeof (value) !== 'undefined' && value !== null && value !== ''; + var hasLabel = typeof (label) !== 'undefined' && label !== null && label !== ''; + + if (!hasValue && !hasLabel) { + return false; + } + + var valueTest = hasValue ? String(value) : null; + var labelTest = hasLabel ? String(label) : null; if (props.ignoreAccents) { - if (props.matchProp !== 'label') valueTest = stripDiacritics(valueTest); - if (props.matchProp !== 'value') labelTest = stripDiacritics(labelTest); + if (valueTest && props.matchProp !== 'label') valueTest = stripDiacritics(valueTest); + if (labelTest && props.matchProp !== 'value') labelTest = stripDiacritics(labelTest); } if (props.ignoreCase) { - if (props.matchProp !== 'label') valueTest = valueTest.toLowerCase(); - if (props.matchProp !== 'value') labelTest = labelTest.toLowerCase(); + if (valueTest && props.matchProp !== 'label') valueTest = valueTest.toLowerCase(); + if (labelTest && props.matchProp !== 'value') labelTest = labelTest.toLowerCase(); } + return props.matchPos === 'start' ? ( - (props.matchProp !== 'label' && valueTest.substr(0, filterValue.length) === filterValue) || - (props.matchProp !== 'value' && labelTest.substr(0, filterValue.length) === filterValue) + (valueTest && props.matchProp !== 'label' && valueTest.substr(0, filterValue.length) === filterValue) || + (labelTest && props.matchProp !== 'value' && labelTest.substr(0, filterValue.length) === filterValue) ) : ( - (props.matchProp !== 'label' && valueTest.indexOf(filterValue) >= 0) || - (props.matchProp !== 'value' && labelTest.indexOf(filterValue) >= 0) + (valueTest && props.matchProp !== 'label' && valueTest.indexOf(filterValue) >= 0) || + (labelTest && props.matchProp !== 'value' && labelTest.indexOf(filterValue) >= 0) ); }); } diff --git a/test/Select-test.js b/test/Select-test.js index c0f0437c4a..f9cae18b16 100644 --- a/test/Select-test.js +++ b/test/Select-test.js @@ -839,7 +839,7 @@ describe('Select', () => { }); }); - describe('searching', () => { + describe.only('searching', () => { let searchOptions = [ { value: 1, label: 'One' }, @@ -848,7 +848,11 @@ describe('Select', () => { { value: 20, label: 'Twenty' }, { value: 21, label: 'Twenty-one' }, { value: 34, label: 'Thirty-four' }, - { value: 54, label: 'Fifty-four' } + { value: 54, label: 'Fifty-four' }, + { value: null, label: null }, + { fish: 'salomon', type: 'inedible' }, + { value: 0, type: 'inedible' }, + { label: 0, type: 'inedible' }, ]; describe('with matchPos=any and matchProp=any', () => { @@ -880,6 +884,22 @@ describe('Select', () => { expect.it('to have text', 'Fifty-four') ]); }); + + it('should not match text when value and/or label are invalid for search', () => { + typeSearchText('ined'); + expect(ReactDOM.findDOMNode(instance), 'to contain elements matching', + '.Select-noresults'); + expect(ReactDOM.findDOMNode(instance), 'to contain no elements matching', + '.Select-option'); + }); + + it('should not match text when value and/or label are null', () => { + typeSearchText('null'); + expect(ReactDOM.findDOMNode(instance), 'to contain elements matching', + '.Select-noresults'); + expect(ReactDOM.findDOMNode(instance), 'to contain no elements matching', + '.Select-option'); + }); }); describe('with matchPos=start and matchProp=any', () => { From 2179f88935b4613186eaa7e4a060c7464d8905ba Mon Sep 17 00:00:00 2001 From: Yuri S Date: Tue, 28 Nov 2017 10:34:31 +0200 Subject: [PATCH 2/3] Removed only --- test/Select-test.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/Select-test.js b/test/Select-test.js index f9cae18b16..cb83002a9d 100644 --- a/test/Select-test.js +++ b/test/Select-test.js @@ -839,7 +839,7 @@ describe('Select', () => { }); }); - describe.only('searching', () => { + describe('searching', () => { let searchOptions = [ { value: 1, label: 'One' }, From 7e10e304b7334dc294285f008f1be4ed531c5c3b Mon Sep 17 00:00:00 2001 From: Yuri S Date: Tue, 28 Nov 2017 11:39:42 +0200 Subject: [PATCH 3/3] Extracted isValid function --- src/utils/defaultFilterOptions.js | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/src/utils/defaultFilterOptions.js b/src/utils/defaultFilterOptions.js index cffeeb4bcd..5ae9b72410 100644 --- a/src/utils/defaultFilterOptions.js +++ b/src/utils/defaultFilterOptions.js @@ -1,6 +1,10 @@ import stripDiacritics from './stripDiacritics'; import trim from './trim'; +function isValid(value) { + return typeof (value) !== 'undefined' && value !== null && value !== ''; +} + function filterOptions (options, filterValue, excludeOptions, props) { if (props.ignoreAccents) { filterValue = stripDiacritics(filterValue); @@ -23,8 +27,8 @@ function filterOptions (options, filterValue, excludeOptions, props) { var value = option[props.valueKey]; var label = option[props.labelKey]; - var hasValue = typeof (value) !== 'undefined' && value !== null && value !== ''; - var hasLabel = typeof (label) !== 'undefined' && label !== null && label !== ''; + var hasValue = isValid(value); + var hasLabel = isValid(label); if (!hasValue && !hasLabel) { return false;