From c1ecaadf9f6ee422fe38d06094beb5ed39d3fbc5 Mon Sep 17 00:00:00 2001 From: Paul Gschwendtner Date: Fri, 10 Mar 2017 20:27:39 +0100 Subject: [PATCH] fix(slide-toggle): invalid change events with no new value Currently the slide-toggle will always emit a change event on dragend. This is not always correct, because the `checked` value only changes if 50% of the distance have been dragged. Fixes #3526 --- src/lib/slide-toggle/slide-toggle.spec.ts | 16 ++++++++++++++++ src/lib/slide-toggle/slide-toggle.ts | 6 +++++- 2 files changed, 21 insertions(+), 1 deletion(-) diff --git a/src/lib/slide-toggle/slide-toggle.spec.ts b/src/lib/slide-toggle/slide-toggle.spec.ts index 92d851c950f5..4599ead4f14d 100644 --- a/src/lib/slide-toggle/slide-toggle.spec.ts +++ b/src/lib/slide-toggle/slide-toggle.spec.ts @@ -563,6 +563,22 @@ describe('MdSlideToggle', () => { expect(testComponent.lastEvent.checked).toBe(true); })); + it('should not emit a change event when the value did not change', fakeAsync(() => { + expect(slideToggle.checked).toBe(false); + + gestureConfig.emitEventForElement('slidestart', slideThumbContainer); + gestureConfig.emitEventForElement('slide', slideThumbContainer, { deltaX: 0 }); + gestureConfig.emitEventForElement('slideend', slideThumbContainer); + + // Flush the timeout for the slide ending. + tick(); + + expect(slideThumbContainer.classList).not.toContain('mat-dragging'); + expect(slideToggle.checked).toBe(false); + expect(testComponent.lastEvent) + .toBeFalsy('Expected the slide-toggle to not emit a change event.'); + })); + it('should update the checked property of the input', fakeAsync(() => { expect(inputElement.checked).toBe(false); diff --git a/src/lib/slide-toggle/slide-toggle.ts b/src/lib/slide-toggle/slide-toggle.ts index ab4a5de00d66..bc96c9ca0f8f 100644 --- a/src/lib/slide-toggle/slide-toggle.ts +++ b/src/lib/slide-toggle/slide-toggle.ts @@ -258,8 +258,12 @@ export class MdSlideToggle implements AfterContentInit, ControlValueAccessor { _onDragEnd() { if (this._slideRenderer.dragging) { + let _previousChecked = this.checked; this.checked = this._slideRenderer.dragPercentage > 50; - this._emitChangeEvent(); + + if (_previousChecked !== this.checked) { + this._emitChangeEvent(); + } // The drag should be stopped outside of the current event handler, because otherwise the // click event will be fired before and will revert the drag change.