blob: 6f942b4dd7692a3b762edc15144403d4013a735f [file] [log] [blame]
// Copyright (c) 2011, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
// @dart = 2.9
part of touch;
/**
* Click buster implementation, which is a behavior that prevents native clicks
* from firing at undesirable times. There are two scenarios where we may want
* to 'bust' a click.
*
* Buttons implemented with touch events usually have click handlers as well.
* This is because sometimes touch events stop working, and the click handler
* serves as a fallback. Here we use a click buster to prevent the native click
* from firing if the touchend event was successfully handled.
*
* When native scrolling behavior is disabled (see Scroller), click events will
* fire after the touchend event when the drag sequence is complete. The click
* event also happens to fire at the location of the touchstart event which can
* lead to some very strange behavior.
*
* This class puts a single click handler on the body, and calls preventDefault
* on the click event if we detect that there was a touchend event that already
* fired in the same spot recently.
*/
class ClickBuster {
/**
* The threshold for how long we allow a click to occur after a touchstart.
*/
static const _TIME_THRESHOLD = 2500;
/**
* The threshold for how close a click has to be to the saved coordinate for
* us to allow it.
*/
static const _DISTANCE_THRESHOLD = 25;
/**
* The list of coordinates that we use to measure the distance of clicks from.
* If a click is within the distance threshold of any of these coordinates
* then we allow the click.
*/
static DoubleLinkedQueue<num> _coordinates;
/** The last time preventGhostClick was called. */
static int _lastPreventedTime;
/**
* This handler will prevent the default behavior for any clicks unless the
* click is within the distance threshold of one of the temporary allowed
* coordinates.
*/
static void _onClick(Event e) {
if (TimeUtil.now() - _lastPreventedTime > _TIME_THRESHOLD) {
return;
}
final coord = new Coordinate.fromClient(e);
// TODO(rnystrom): On Android, we get spurious click events at (0, 0). We
// *do* want those clicks to be busted, so commenting this out fixes it.
// Leaving it commented out instead of just deleting it because I'm not sure
// what this code was intended to do to begin with.
/*
if (coord.x < 1 && coord.y < 1) {
// TODO(jacobr): implement a configurable logging framework.
// _logger.warning(
// "Not busting click on label elem at(${coord.x}, ${coord.y})");
return;
}
*/
DoubleLinkedQueueEntry<num> entry = _coordinates.firstEntry();
while (entry != null) {
if (_hitTest(
entry.element, entry.nextEntry().element, coord.x, coord.y)) {
entry.nextEntry().remove();
entry.remove();
return;
} else {
entry = entry.nextEntry().nextEntry();
}
}
// TODO(jacobr): implement a configurable logging framework.
// _logger.warning("busting click at ${coord.x}, ${coord.y}");
e.stopPropagation();
e.preventDefault();
}
/**
* This handler will temporarily allow a click to occur near the touch event's
* coordinates.
*/
static void _onTouchStart(Event e) {
TouchEvent te = e;
final coord = new Coordinate.fromClient(te.touches[0]);
_coordinates.add(coord.x);
_coordinates.add(coord.y);
new Timer(const Duration(milliseconds: _TIME_THRESHOLD), () {
_removeCoordinate(coord.x, coord.y);
});
_toggleTapHighlights(true);
}
/**
* Hit test for whether a coordinate is within the distance threshold of an
* event.
*/
static bool _hitTest(num x, num y, num eventX, num eventY) {
return (eventX - x).abs() < _DISTANCE_THRESHOLD &&
(eventY - y).abs() < _DISTANCE_THRESHOLD;
}
/**
* Remove one specified coordinate from the coordinates list.
*/
static void _removeCoordinate(num x, num y) {
DoubleLinkedQueueEntry<num> entry = _coordinates.firstEntry();
while (entry != null) {
if (entry.element == x && entry.nextEntry().element == y) {
entry.nextEntry().remove();
entry.remove();
return;
} else {
entry = entry.nextEntry().nextEntry();
}
}
}
/**
* Enable or disable tap highlights. They are disabled when preventGhostClick
* is called so that the flicker on links is not invoked when the ghost click
* does fire. This is due to a bug: links get highlighted even if the click
* event has preventDefault called on it.
*/
static void _toggleTapHighlights(bool enable) {
document.body.style.setProperty(
"-webkit-tap-highlight-color", enable ? "" : "rgba(0,0,0,0)", "");
}
/**
* Registers new touches to create temporary "allowable zones" and registers
* new clicks to be prevented unless they fall in one of the current
* "allowable zones". Note that if the touchstart and touchend locations are
* different, it is still possible for a ghost click to be fired if you
* called preventDefault on all touchmove events. In this case the ghost
* click will be fired at the location of the touchstart event, so the
* coordinate you pass in should be the coordinate of the touchstart.
*/
static void preventGhostClick(num x, num y) {
// First time this is called the following occurs:
// 1) Attaches a handler to touchstart events so that each touch will
// temporarily create an "allowable zone" for clicks to occur in.
// 2) Attaches a handler to click events so that each click will be
// prevented unless it is in an "allowable zone".
//
// Every time this is called (including the first) the following occurs:
// 1) Removes an allowable zone that contains the specified coordinate.
//
// How this enables click busting:
// 1) User performs first click.
// - No attached touchstart handler yet.
// - preventGhostClick is called before the click event occurs, it
// attaches the touchstart and click handlers.
// - The click handler captures the user's click event and prevents it
// from propagating since there is no "allowable zone".
//
// 2) User performs subsequent, to-be-busted click.
// - touchstart event triggers the attached handler and creates a
// temporary "allowable zone".
// - preventGhostClick is called and removes the "allowable zone".
// - The click handler captures the user's click event and prevents it
// from propagating since there is no "allowable zone".
//
// 3) User performs a should-not-be-busted click.
// - touchstart event triggers the attached handler and creates a
// temporary "allowable zone".
// - The click handler captures the user's click event and allows it to
// propagate since the click falls in the "allowable zone".
if (_coordinates == null) {
// Listen to clicks on capture phase so they can be busted before anything
// else gets a chance to handle them.
Element.clickEvent.forTarget(document, useCapture: true).listen((e) {
_onClick(e);
});
Element.focusEvent.forTarget(document, useCapture: true).listen((e) {
_lastPreventedTime = 0;
});
// Listen to touchstart on capture phase since it must be called prior to
// every click or else we will accidentally prevent the click even if we
// don't call preventGhostClick.
Function startFn = (e) {
_onTouchStart(e);
};
if (!Device.supportsTouch) {
startFn = mouseToTouchCallback(startFn);
}
var stream;
if (Device.supportsTouch) {
stream = Element.touchStartEvent.forTarget(document, useCapture: true);
} else {
stream = Element.mouseDownEvent.forTarget(document, useCapture: true);
}
EventUtil.observe(document, stream, startFn, true);
_coordinates = new DoubleLinkedQueue<num>();
}
// Turn tap highlights off until we know the ghost click has fired.
_toggleTapHighlights(false);
// Above all other rules, we won't bust any clicks if there wasn't some call
// to preventGhostClick in the last time threshold.
_lastPreventedTime = TimeUtil.now();
DoubleLinkedQueueEntry<num> entry = _coordinates.firstEntry();
while (entry != null) {
if (_hitTest(entry.element, entry.nextEntry().element, x, y)) {
entry.nextEntry().remove();
entry.remove();
return;
} else {
entry = entry.nextEntry().nextEntry();
}
}
}
}