diff --git a/js/ext/angular/test/scroll2.html b/js/ext/angular/test/scroll2.html
new file mode 100644
index 0000000000..1aa4818562
--- /dev/null
+++ b/js/ext/angular/test/scroll2.html
@@ -0,0 +1,197 @@
+
+
+
+ Scroll Click Tests
+
+
+
+
+
+
+
+ CLICK!
+ Mouse Move!
+ Touch Move!
+ Touch Cancel!
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/js/ext/angular/test/service/ionicTap.unit.js b/js/ext/angular/test/service/ionicTap.unit.js
index d2bca36234..f80e927bd2 100644
--- a/js/ext/angular/test/service/ionicTap.unit.js
+++ b/js/ext/angular/test/service/ionicTap.unit.js
@@ -5,30 +5,13 @@ describe('Ionic Tap', function() {
ionic.tap.reset();
});
- it('Should not focus on an input if it has scrolled', function() {
- var targetEle = {
- dispatchEvent: function() {},
- focus: function() { this.isFocused = true; }
- };
-
- ionic.tap.setStart({clientX: 100, clientY: 100});
-
- targetEle.tagName = 'INPUT';
- var e = {
- clientX: 100, clientY: 200,
- preventDefault: function() {}
- };
- ionic.tap.simulateClick(targetEle, e);
- expect(targetEle.isFocused).toBeUndefined();
- });
-
it('Should focus on an input if it hasnt scrolled', function() {
var targetEle = {
dispatchEvent: function() {},
focus: function() { this.isFocused = true; }
};
- ionic.tap.setStart({clientX: 100, clientY: 100});
+ ionic.tap.setTouchStart({clientX: 100, clientY: 100});
targetEle.tagName = 'INPUT';
var e = {
@@ -46,7 +29,7 @@ describe('Ionic Tap', function() {
focus: function() {}
};
- ionic.tap.setStart({ clientX: 100, clientY: 100 });
+ ionic.tap.setTouchStart({ clientX: 100, clientY: 100 });
var e = {
clientX: 100, clientY: 100,
preventDefault: function() { this.preventedDefault = true }
@@ -68,46 +51,46 @@ describe('Ionic Tap', function() {
e.preventedDefault = false;
});
- it('Should setStart and hasScrolled true if >= touch tolerance', function() {
- ionic.tap.setStart({ clientX: 100, clientY: 100 });
+ it('Should setTouchStart and hasTouchScrolled true if >= touch tolerance', function() {
+ ionic.tap.setTouchStart({ clientX: 100, clientY: 100 });
- var s = ionic.tap.hasScrolled({ clientX: 111, clientY: 100 });
+ var s = ionic.tap.hasTouchScrolled({ clientX: 111, clientY: 100 });
expect(s).toEqual(true);
- s = ionic.tap.hasScrolled({ clientX: 89, clientY: 100 });
+ s = ionic.tap.hasTouchScrolled({ clientX: 89, clientY: 100 });
expect(s).toEqual(true);
- s = ionic.tap.hasScrolled({ clientX: 100, clientY: 107 });
+ s = ionic.tap.hasTouchScrolled({ clientX: 100, clientY: 107 });
expect(s).toEqual(true);
- s = ionic.tap.hasScrolled({ clientX: 100, clientY: 93 });
+ s = ionic.tap.hasTouchScrolled({ clientX: 100, clientY: 93 });
expect(s).toEqual(true);
- s = ionic.tap.hasScrolled({ clientX: 100, clientY: 200 });
+ s = ionic.tap.hasTouchScrolled({ clientX: 100, clientY: 200 });
expect(s).toEqual(true);
});
- it('Should setStart and hasScrolled false if less than touch tolerance', function() {
- ionic.tap.setStart({ clientX: 100, clientY: 100 });
+ it('Should setTouchStart and hasTouchScrolled false if less than touch tolerance', function() {
+ ionic.tap.setTouchStart({ clientX: 100, clientY: 100 });
- var s = ionic.tap.hasScrolled({ clientX: 100, clientY: 100 });
+ var s = ionic.tap.hasTouchScrolled({ clientX: 100, clientY: 100 });
expect(s).toEqual(false);
- s = ionic.tap.hasScrolled({ clientX: 104, clientY: 100 });
+ s = ionic.tap.hasTouchScrolled({ clientX: 104, clientY: 100 });
expect(s).toEqual(false);
- s = ionic.tap.hasScrolled({ clientX: 96, clientY: 100 });
+ s = ionic.tap.hasTouchScrolled({ clientX: 96, clientY: 100 });
expect(s).toEqual(false);
- s = ionic.tap.hasScrolled({ clientX: 100, clientY: 102 });
+ s = ionic.tap.hasTouchScrolled({ clientX: 100, clientY: 102 });
expect(s).toEqual(false);
- s = ionic.tap.hasScrolled({ clientX: 100, clientY: 98 });
+ s = ionic.tap.hasTouchScrolled({ clientX: 100, clientY: 98 });
expect(s).toEqual(false);
});
- it('Should not be hasScrolled if 0 coordinates', function() {
- var s = ionic.tap.hasScrolled({ clientX: 0, clientY: 0 });
+ it('Should not be hasTouchScrolled if 0 coordinates', function() {
+ var s = ionic.tap.hasTouchScrolled({ clientX: 0, clientY: 0 });
expect(s).toEqual(false);
});
diff --git a/js/utils/activator.js b/js/utils/activator.js
index 36317ba807..10e46d9299 100644
--- a/js/utils/activator.js
+++ b/js/utils/activator.js
@@ -44,8 +44,8 @@
document.body.removeEventListener('mousedown', ionic.activator.start);
touchMoveClearTimer = setTimeout(function(){
document.body.addEventListener('touchmove', onTouchMove, false);
- }, 85);
- setTimeout(activateElements, 85);
+ }, 80);
+ setTimeout(activateElements, 80);
} else {
document.body.addEventListener('mousemove', clear, false);
ionic.requestAnimationFrame(activateElements);
@@ -79,7 +79,7 @@
}
function onTouchMove(e) {
- if( ionic.tap.hasScrolled(e) ) {
+ if( ionic.tap.hasTouchScrolled(e) ) {
clear();
}
}
diff --git a/js/utils/tap.js b/js/utils/tap.js
index 7e2f50870b..40a4108c47 100644
--- a/js/utils/tap.js
+++ b/js/utils/tap.js
@@ -10,6 +10,7 @@
var tapCoordinates = {}; // used to remember coordinates to ignore if they happen again quickly
var startCoordinates = {}; // used to remember where the coordinates of the start of a touch
var clickPreventTimerId;
+ var _hasTouchScrolled = false; // if the touchmove already exceeded the touchmove tolerance
ionic.tap = {
@@ -21,10 +22,10 @@
var e = orgEvent.gesture.srcEvent; // evaluate the actual source event, not the created event by gestures.js
var ele = e.target; // get the target element that was actually tapped
- if( ionic.tap.isRecentTap(e) || e.type === 'touchcancel' ) {
+ if( ionic.tap.isRecentTap(e) || ionic.tap.hasTouchScrolled(e) || e.type === 'touchcancel') {
// if a tap in the same area just happened,
// or it was a touchcanel event, don't continue
- console.debug('tapInspect', 'isRecentTap', ele.tagName, 'type:', e.type);
+ console.debug('tapInspect stopEvent', e.type, ele.tagName);
return stopEvent(e);
}
@@ -60,7 +61,7 @@
return;
}
- console.debug('simulateClick', ele.tagName, ele.className);
+ console.debug('simulateClick', e.type, ele.tagName, ele.className);
var c = ionic.tap.getCoordinates(e);
@@ -73,9 +74,7 @@
ele.dispatchEvent(clickEvent);
if(ele.tagName === 'INPUT' || ele.tagName === 'TEXTAREA') {
- if(!ionic.tap.hasScrolled(e)) {
- ele.focus();
- }
+ ele.focus();
e.preventDefault();
} else {
ionic.tap.blurActive();
@@ -108,9 +107,9 @@
// a tap has already happened at these coordinates recently, ignore this event
console.debug('preventGhostClick', 'isRecentTap', e.target.tagName);
- } else if(ionic.tap.hasScrolled(e)) {
+ } else if(ionic.tap.hasTouchScrolled(e)) {
// this click's coordinates are different than its touchstart/mousedown, must have been scrolling
- console.debug('preventGhostClick', 'hasScrolled');
+ console.debug('preventGhostClick', 'hasTouchScrolled');
}
var c = ionic.tap.getCoordinates(e);
@@ -118,7 +117,7 @@
})());
- if(e.target.control || ionic.tap.isRecentTap(e) || ionic.tap.hasScrolled(e)) {
+ if(e.target.control || ionic.tap.isRecentTap(e) || ionic.tap.hasTouchScrolled(e)) {
return stopEvent(e);
}
@@ -144,7 +143,9 @@
return { x:0, y:0 };
},
- hasScrolled: function(event) {
+ hasTouchScrolled: function(event) {
+ if(_hasTouchScrolled) return true;
+
// check if this click's coordinates are different than its touchstart/mousedown
var c = ionic.tap.getCoordinates(event);
@@ -218,8 +219,18 @@
}
},
- setStart: function(e) {
+ setTouchStart: function(e) {
+ _hasTouchScrolled = false;
startCoordinates = ionic.tap.getCoordinates(e);
+ document.body.addEventListener('touchmove', ionic.tap.onTouchMove, false);
+ },
+
+ onTouchMove: function(e) {
+ if( ionic.tap.hasTouchScrolled(e) ) {
+ _hasTouchScrolled = true;
+ document.body.removeEventListener('touchmove', ionic.tap.onTouchMove);
+ console.debug('hasTouchScrolled');
+ }
},
reset: function() {
@@ -254,6 +265,6 @@
// remember where the user first started touching the screen
// so that if they scrolled, it shouldn't fire the click
- document.addEventListener('touchstart', ionic.tap.setStart, false);
+ document.addEventListener('touchstart', ionic.tap.setTouchStart, false);
})(this, document, ionic);