mirror of
https://github.com/EGroupware/egroupware.git
synced 2025-01-10 07:58:45 +01:00
301 lines
8.3 KiB
TypeScript
301 lines
8.3 KiB
TypeScript
/**
|
|
* EGroupware TapAndSwipe helper library
|
|
*
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
* @package Api
|
|
* @subpackage ui
|
|
* @link https://www.egroupware.org
|
|
* @author Hadi Nategh <hn@egroupware.org>
|
|
*/
|
|
export interface TapAndSwipeOptions {
|
|
// allow scrolling would stop swipe/tap events from being fired when there's scrolling available. It can be restricted
|
|
// to Vertical/Horizental/Both scrolling. If no value set it means not allowed.
|
|
allowScrolling? : string|null,
|
|
// tolorated pixel for tap event
|
|
threshold? : number,
|
|
// time delay for defirentiate between tap event and long tap event, threshold is in milliseconds
|
|
tapHoldThreshold? : number,
|
|
// DOM node(s) that event handler has to be bound to, it can be a selector or acctual HTMLElement
|
|
element? : string|HTMLElement,
|
|
// callback function being called on swipe gestures
|
|
swipe? : Function,
|
|
// callback function being called on tap
|
|
tap? : Function,
|
|
// callback function being called on long tap(tap and hold)
|
|
tapAndHold? : Function,
|
|
// tolerate pixel to fire swipe events
|
|
minSwipeThreshold? : number,
|
|
}
|
|
|
|
export type TapAndSwipeOptionsType = TapAndSwipeOptions;
|
|
|
|
export class tapAndSwipe {
|
|
static readonly _default : TapAndSwipeOptionsType = {
|
|
threshold : 5,
|
|
tapHoldThreshold : 3000,
|
|
minSwipeThreshold: 150,
|
|
allowScrolling : 'both',
|
|
swipe : function(){},
|
|
tap : function(){},
|
|
tapAndHold: function(){}
|
|
}
|
|
/**
|
|
* Keeps the touch X start point
|
|
* @private
|
|
*/
|
|
private _startX : number = null;
|
|
/**
|
|
* Keeps the touch Y start point
|
|
* @private
|
|
*/
|
|
private _startY : number = null;
|
|
/**
|
|
* Keeps the touch X end point
|
|
* @private
|
|
*/
|
|
private _endX : number = null;
|
|
/**
|
|
* Keeps the touch Y end point
|
|
* @private
|
|
*/
|
|
private _endY : number = null;
|
|
/**
|
|
* keeps the distance travelled between startX point and endX point
|
|
* @private
|
|
*/
|
|
private _distanceX : number = null;
|
|
|
|
/**
|
|
* keeps the distance travelled between startY point and endY point
|
|
* @private
|
|
*/
|
|
private _distanceY : number = null;
|
|
|
|
/**
|
|
* flag to keep the status of type of tap
|
|
* @private
|
|
*/
|
|
private _isTapAndHold : boolean = false;
|
|
/**
|
|
* keeps the timeout id for taphold
|
|
*/
|
|
private _tapHoldTimeout : number = null;
|
|
/**
|
|
* keeps the timeout id for tap
|
|
*/
|
|
private _tapTimeout : number = null;
|
|
|
|
/**
|
|
* keeps the contact point on touch start
|
|
* @private
|
|
*/
|
|
private _fingercount : number = null;
|
|
|
|
private _scrolledElementObj : {el : HTMLElement, scrollTop : number, scrollLeft : number} = null;
|
|
|
|
private _hasBeenScrolled : boolean = false;
|
|
|
|
private _scrollEventTriggered : boolean = false;
|
|
|
|
private _stillMoving : boolean = false;
|
|
|
|
/**
|
|
* Options
|
|
* @protected
|
|
*/
|
|
protected options : TapAndSwipeOptionsType = null;
|
|
|
|
/**
|
|
* Keeps the html node
|
|
*/
|
|
public element : HTMLElement = null;
|
|
|
|
/**
|
|
* Constructor
|
|
* @param _element
|
|
* @param _options
|
|
*/
|
|
public constructor(_element : string | HTMLElement, _options? : TapAndSwipeOptionsType)
|
|
{
|
|
this.options = {...tapAndSwipe._default, ..._options};
|
|
const element = _element||_options.element;
|
|
// Dont construct if the element is not there
|
|
if (!element || !(typeof element != 'string' && element instanceof EventTarget)) return;
|
|
|
|
this.element = (element instanceof EventTarget) ? element : document.querySelector(element);
|
|
this.element.addEventListener('touchstart', this._onTouchStart.bind(this), false);
|
|
this.element.addEventListener('touchend', this._ontouchEnd.bind(this), false);
|
|
this.element.addEventListener('touchmove', this._onTouchMove.bind(this), false);
|
|
this.element.addEventListener('touchcancel', this._onTouchCancel.bind(this), false);
|
|
}
|
|
|
|
_onScrolled(event)
|
|
{
|
|
this._scrollEventTriggered = true;
|
|
}
|
|
|
|
_onTouchCancel(event)
|
|
{
|
|
//cleanup tapHoldTimeout
|
|
window.clearTimeout(this._tapHoldTimeout);
|
|
//cleanup tapHoldTimeout
|
|
window.clearTimeout(this._tapTimeout);
|
|
}
|
|
|
|
_onTouchMove(event)
|
|
{
|
|
this._stillMoving = true;
|
|
}
|
|
/**
|
|
* on touch start event handler
|
|
* @param event
|
|
* @private
|
|
*/
|
|
private _onTouchStart(event : TouchEvent)
|
|
{
|
|
this._startX = event.changedTouches[0].pageX;
|
|
this._startY = event.changedTouches[0].pageY;
|
|
this._isTapAndHold = false;
|
|
this._fingercount = event.touches.length;
|
|
|
|
if(event.composedPath())
|
|
{
|
|
const scrolledItem = event.composedPath().filter(_item => {
|
|
if (_item instanceof HTMLElement) _item.addEventListener('scroll', this._onScrolled.bind(this), false);
|
|
return _item instanceof HTMLElement && this.element.contains(_item) && (_item.scrollTop != 0 || _item.scrollLeft !=0);
|
|
|
|
});
|
|
if (scrolledItem.length>0)
|
|
{
|
|
this._scrolledElementObj = {el: scrolledItem[0], scrollTop: scrolledItem[0].scrollTop, scrollLeft: scrolledItem[0].scrollLeft};
|
|
}
|
|
else
|
|
{
|
|
this._scrolledElementObj = null;
|
|
}
|
|
}
|
|
|
|
this._tapHoldTimeout = window.setTimeout(_=>{
|
|
this._isTapAndHold = true;
|
|
//check scrolling
|
|
if (this.options.allowScrolling && this._stillMoving)
|
|
{
|
|
return;
|
|
}
|
|
|
|
this.options.tapAndHold.call(this, event, this._fingercount);
|
|
}, this.options.tapHoldThreshold);
|
|
}
|
|
|
|
/**
|
|
* On touch end event handler
|
|
* @param event
|
|
* @private
|
|
*/
|
|
private _ontouchEnd(event : TouchEvent)
|
|
{
|
|
this._endX = event.changedTouches[0].pageX;
|
|
this._endY = event.changedTouches[0].pageY;
|
|
this._stillMoving = false;
|
|
|
|
if (this._scrolledElementObj) {
|
|
switch (this.options.allowScrolling)
|
|
{
|
|
case "vertical":
|
|
this._hasBeenScrolled = this._scrolledElementObj.el.scrollTop != this._scrolledElementObj.scrollTop;
|
|
break;
|
|
case "horizental":
|
|
this._hasBeenScrolled = this._scrolledElementObj.el.scrollLeft != this._scrolledElementObj.scrollLeft;
|
|
break;
|
|
case "both":
|
|
this._hasBeenScrolled = (this._scrolledElementObj.el.scrollTop != this._scrolledElementObj.scrollTop) ||
|
|
(this._scrolledElementObj.el.scrollLeft != this._scrolledElementObj.scrollLeft);
|
|
break;
|
|
default:
|
|
this._hasBeenScrolled = false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
this._hasBeenScrolled = false;
|
|
}
|
|
|
|
this._handler(event);
|
|
}
|
|
|
|
/**
|
|
* Handles the type of gesture and calls the right callback for it
|
|
* @param event
|
|
* @private
|
|
*/
|
|
private _handler(event : TouchEvent)
|
|
{
|
|
//cleanup tapHoldTimeout
|
|
window.clearTimeout(this._tapHoldTimeout);
|
|
this._distanceX = Math.abs(this._endX-this._startX);
|
|
this._distanceY = Math.abs(this._endY-this._startY);
|
|
const isTabOrHold = (this._endX == this._startX && this._endY == this._startY
|
|
|| (Math.sqrt((this._distanceX)*(this._distanceX) + (this._distanceY*2))
|
|
< this.options.threshold));
|
|
|
|
|
|
this._hasBeenScrolled = this._hasBeenScrolled ?? (!isTabOrHold && this._scrollEventTriggered);
|
|
//check scrolling
|
|
if (this.options.allowScrolling && this._hasBeenScrolled)
|
|
{
|
|
// let the scrolling happens
|
|
return;
|
|
}
|
|
|
|
// Tap & TapAndHold handler
|
|
if (isTabOrHold)
|
|
{
|
|
if (!this._isTapAndHold)
|
|
{
|
|
this._tapTimeout = window.setTimeout(_=> {
|
|
this.options.tap.call(this,event, this._fingercount);
|
|
}, 100);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// left swipe handler
|
|
if (this._endX + this.options.threshold < this._startX && this._distanceX > this._distanceY) {
|
|
if (this._distanceX < this.options.minSwipeThreshold) return;
|
|
this.options.swipe.call(this, event, 'left', this._distanceX, this._fingercount);
|
|
return;
|
|
}
|
|
|
|
// right swipe handler
|
|
if (this._endX - this.options.threshold > this._startX && this._distanceX > this._distanceY) {
|
|
if (this._distanceX < this.options.minSwipeThreshold) return;
|
|
this.options.swipe.call(this, event, 'right', this._distanceX, this._fingercount);
|
|
return;
|
|
}
|
|
|
|
// up swipe handler
|
|
if (this._endY + this.options.threshold < this._startY && this._distanceY > this._distanceX) {
|
|
if (this._distanceY < this.options.minSwipeThreshold) return;
|
|
this.options.swipe.call(this, event, 'up', this._distanceY, this._fingercount);
|
|
return;
|
|
}
|
|
|
|
// down swipe handler
|
|
if (this._endY - this.options.threshold > this._startY && this._distanceY > this._distanceX) {
|
|
if (this._distanceY < this.options.minSwipeThreshold) return;
|
|
this.options.swipe.call(this, event, 'down', this._distanceY, this._fingercount);
|
|
return;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* destroy the event listeners
|
|
*/
|
|
public destroy()
|
|
{
|
|
this.element.removeEventListener('touchstart', this._onTouchStart);
|
|
this.element.removeEventListener('touchend', this._ontouchEnd);
|
|
this.element.removeEventListener('touchcancel', this._onTouchCancel);
|
|
}
|
|
} |