diff --git a/src/ng/directive/input.js b/src/ng/directive/input.js index 9ddd15aa07b9..afa8347e23c6 100644 --- a/src/ng/directive/input.js +++ b/src/ng/directive/input.js @@ -1751,32 +1751,33 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$ var parsedNgModel = $parse($attr.ngModel), + parsedNgModelAssign = parsedNgModel.assign, + ngModelGet = parsedNgModel, + ngModelSet = parsedNgModelAssign, pendingDebounce = null, ctrl = this; - var ngModelGet = function ngModelGet() { - var modelValue = parsedNgModel($scope); - if (ctrl.$options && ctrl.$options.getterSetter && isFunction(modelValue)) { - modelValue = modelValue(); - } - return modelValue; - }; - - var ngModelSet = function ngModelSet(newValue) { - var getterSetter; - if (ctrl.$options && ctrl.$options.getterSetter && - isFunction(getterSetter = parsedNgModel($scope))) { - - getterSetter(ctrl.$modelValue); - } else { - parsedNgModel.assign($scope, ctrl.$modelValue); - } - }; - this.$$setOptions = function(options) { ctrl.$options = options; - - if (!parsedNgModel.assign && (!options || !options.getterSetter)) { + if (options && options.getterSetter) { + var invokeModelGetter = $parse($attr.ngModel + '()'), + invokeModelSetter = $parse($attr.ngModel + '($$$p)'); + + ngModelGet = function($scope) { + var modelValue = parsedNgModel($scope); + if (isFunction(modelValue)) { + modelValue = invokeModelGetter($scope); + } + return modelValue; + }; + ngModelSet = function($scope, newValue) { + if (isFunction(parsedNgModel($scope))) { + invokeModelSetter($scope, {$$$p: ctrl.$modelValue}); + } else { + parsedNgModelAssign($scope, ctrl.$modelValue); + } + }; + } else if (!parsedNgModel.assign) { throw $ngModelMinErr('nonassign', "Expression '{0}' is non-assignable. Element: {1}", $attr.ngModel, startingTag($element)); } @@ -2189,7 +2190,7 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$ } if (isNumber(ctrl.$modelValue) && isNaN(ctrl.$modelValue)) { // ctrl.$modelValue has not been touched yet... - ctrl.$modelValue = ngModelGet(); + ctrl.$modelValue = ngModelGet($scope); } var prevModelValue = ctrl.$modelValue; var allowInvalid = ctrl.$options && ctrl.$options.allowInvalid; @@ -2217,7 +2218,7 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$ }; this.$$writeModelToScope = function() { - ngModelSet(ctrl.$modelValue); + ngModelSet($scope, ctrl.$modelValue); forEach(ctrl.$viewChangeListeners, function(listener) { try { listener(); @@ -2313,7 +2314,7 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$ // ng-change executes in apply phase // 4. view should be changed back to 'a' $scope.$watch(function ngModelWatch() { - var modelValue = ngModelGet(); + var modelValue = ngModelGet($scope); // if scope model value and ngModel value are out of sync // TODO(perf): why not move this to the action fn? diff --git a/test/ng/directive/inputSpec.js b/test/ng/directive/inputSpec.js index f83f4ee58c90..b2ab27ac9469 100644 --- a/test/ng/directive/inputSpec.js +++ b/test/ng/directive/inputSpec.js @@ -2102,7 +2102,7 @@ describe('input', function() { expect(inputElm.val()).toBe('a'); }); - it('should always try to invoke a model if getterSetter is true', function() { + it('should try to invoke a function model if getterSetter is true', function() { compileInput( ''); @@ -2117,6 +2117,12 @@ describe('input', function() { expect(inputElm.val()).toBe('b'); expect(spy).toHaveBeenCalledWith('a'); expect(scope.name).toBe(spy); + }); + + it('should assign to non-function models if getterSetter is true', function() { + compileInput( + ''); scope.name = 'c'; changeInputValueTo('d'); @@ -2136,6 +2142,35 @@ describe('input', function() { 'ng-model-options="{ getterSetter: true }" />'); }); + it('should invoke a model in the correct context if getterSetter is true', function() { + compileInput( + ''); + + scope.someService = { + value: 'a', + getterSetter: function(newValue) { + this.value = newValue || this.value; + return this.value; + } + }; + spyOn(scope.someService, 'getterSetter').andCallThrough(); + scope.$apply(); + + expect(inputElm.val()).toBe('a'); + expect(scope.someService.getterSetter).toHaveBeenCalledWith(); + expect(scope.someService.value).toBe('a'); + + changeInputValueTo('b'); + expect(scope.someService.getterSetter).toHaveBeenCalledWith('b'); + expect(scope.someService.value).toBe('b'); + + scope.someService.value = 'c'; + scope.$apply(); + expect(inputElm.val()).toBe('c'); + expect(scope.someService.getterSetter).toHaveBeenCalledWith(); + }); + it('should assign invalid values to the scope if allowInvalid is true', function() { compileInput('');