diff --git a/src/typeahead/test/typeahead.spec.js b/src/typeahead/test/typeahead.spec.js index 825b008738..2afff921b3 100644 --- a/src/typeahead/test/typeahead.spec.js +++ b/src/typeahead/test/typeahead.spec.js @@ -457,6 +457,48 @@ describe('typeahead tests', function() { }); }); + describe('isSelectTrigger', function() { + it('should select a match when function returns true', function() { + $scope.isSelectTrigger = function() { + return true; + }; + var element = prepareInputEl('
'); + var inputEl = findInput(element); + + changeInputValueTo(element, 'b'); + triggerKeyDown(element, 13); + + expect($scope.result).toEqual('bar'); + expect(inputEl.val()).toEqual('bar'); + expect(element).toBeClosed(); + }); + it('should not select a match when function returns false', function() { + $scope.isSelectTrigger = function() { + return false; + }; + var element = prepareInputEl(''); + var inputEl = findInput(element); + + changeInputValueTo(element, 'b'); + triggerKeyDown(element, 13); + + // no change + expect($scope.result).toEqual('b'); + expect(inputEl.val()).toEqual('b'); + }); + it('should pass key event into select trigger function', function() { + $scope.isSelectTrigger = jasmine.createSpy('isSelectTrigger');//.and.returnValue(true); + var element = prepareInputEl(''); + var inputEl = findInput(element); + + changeInputValueTo(element, 'b'); + triggerKeyDown(element, 13); + + expect($scope.isSelectTrigger.calls.count()).toEqual(1); + expect($scope.isSelectTrigger.calls.argsFor(0)[0].which).toEqual(13); + }); + }); + describe('selecting a match', function() { it('should select a match on enter', function() { var element = prepareInputEl(''); diff --git a/src/typeahead/typeahead.js b/src/typeahead/typeahead.js index 14169c6371..0fdeee795c 100644 --- a/src/typeahead/typeahead.js +++ b/src/typeahead/typeahead.js @@ -42,7 +42,7 @@ angular.module('ui.bootstrap.typeahead', ['ui.bootstrap.debounce', 'ui.bootstrap originalScope.$watch(attrs.typeaheadMinLength, function (newVal) { minLength = !newVal && newVal !== 0 ? 1 : newVal; }); - + //minimal wait time after last character typed before typeahead kicks-in var waitTime = originalScope.$eval(attrs.typeaheadWaitMs) || 0; @@ -55,6 +55,11 @@ angular.module('ui.bootstrap.typeahead', ['ui.bootstrap.debounce', 'ui.bootstrap //binding to a variable that indicates if matches are being retrieved asynchronously var isLoadingSetter = $parse(attrs.typeaheadLoading).assign || angular.noop; + //a function to determine if an event should trigger selection + var isSelectTrigger = attrs.typeaheadIsSelectTrigger ? originalScope.$eval(attrs.typeaheadIsSelectTrigger) : function(evt) { + return evt.which === 13 || evt.which === 9; + }; + //a callback executed when a match is selected var onSelectCallback = $parse(attrs.typeaheadOnSelect); @@ -370,13 +375,15 @@ angular.module('ui.bootstrap.typeahead', ['ui.bootstrap.debounce', 'ui.bootstrap return; } + var triggerKey = isSelectTrigger(evt); + /** * if there's nothing selected (i.e. focusFirst) and enter or tab is hit * or * shift + tab is pressed to bring focus to the previous element * then clear the results */ - if (scope.activeIdx === -1 && (evt.which === 9 || evt.which === 13) || evt.which === 9 && !!evt.shiftKey) { + if (scope.activeIdx === -1 && triggerKey || evt.which === 9 && !!evt.shiftKey) { resetMatches(); scope.$digest(); return; @@ -385,36 +392,37 @@ angular.module('ui.bootstrap.typeahead', ['ui.bootstrap.debounce', 'ui.bootstrap evt.preventDefault(); var target; switch (evt.which) { - case 9: - case 13: - scope.$apply(function () { - if (angular.isNumber(scope.debounceUpdate) || angular.isObject(scope.debounceUpdate)) { - $$debounce(function() { - scope.select(scope.activeIdx, evt); - }, angular.isNumber(scope.debounceUpdate) ? scope.debounceUpdate : scope.debounceUpdate['default']); - } else { - scope.select(scope.activeIdx, evt); - } - }); - break; - case 27: + case 27: // escape evt.stopPropagation(); resetMatches(); originalScope.$digest(); break; - case 38: + case 38: // up arrow scope.activeIdx = (scope.activeIdx > 0 ? scope.activeIdx : scope.matches.length) - 1; scope.$digest(); target = popUpEl.find('li')[scope.activeIdx]; target.parentNode.scrollTop = target.offsetTop; break; - case 40: + case 40: // down arrow scope.activeIdx = (scope.activeIdx + 1) % scope.matches.length; scope.$digest(); target = popUpEl.find('li')[scope.activeIdx]; target.parentNode.scrollTop = target.offsetTop; break; + default: + if (triggerKey) { + scope.$apply(function() { + if (angular.isNumber(scope.debounceUpdate) || angular.isObject(scope.debounceUpdate)) { + $$debounce(function() { + scope.select(scope.activeIdx, evt); + }, angular.isNumber(scope.debounceUpdate) ? scope.debounceUpdate : scope.debounceUpdate['default']); + } else { + scope.select(scope.activeIdx, evt); + } + }); + } + break; } });