mirror of
https://github.com/thelounge/thelounge
synced 2025-01-02 07:38:44 +00:00
105 lines
2.5 KiB
JavaScript
105 lines
2.5 KiB
JavaScript
"use strict";
|
|
|
|
const viewport = document.getElementById("viewport");
|
|
const menu = document.getElementById("sidebar");
|
|
const sidebarOverlay = document.getElementById("sidebar-overlay");
|
|
|
|
let touchStartPos = null;
|
|
let touchCurPos = null;
|
|
let touchStartTime = 0;
|
|
let menuWidth = 0;
|
|
let menuIsOpen = false;
|
|
let menuIsMoving = false;
|
|
|
|
class SlideoutMenu {
|
|
static enable() {
|
|
document.body.addEventListener("touchstart", onTouchStart, {passive: true});
|
|
}
|
|
|
|
static toggle(state) {
|
|
menuIsOpen = state;
|
|
viewport.classList.toggle("menu-open", state);
|
|
}
|
|
|
|
static isOpen() {
|
|
return menuIsOpen;
|
|
}
|
|
}
|
|
|
|
function onTouchStart(e) {
|
|
if (e.touches.length !== 1) {
|
|
onTouchEnd();
|
|
return;
|
|
}
|
|
|
|
const touch = e.touches.item(0);
|
|
|
|
menuWidth = parseFloat(window.getComputedStyle(menu).width);
|
|
|
|
if (!menuIsOpen || touch.screenX > menuWidth) {
|
|
touchStartPos = touch;
|
|
touchCurPos = touch;
|
|
touchStartTime = Date.now();
|
|
|
|
document.body.addEventListener("touchmove", onTouchMove, {passive: true});
|
|
document.body.addEventListener("touchend", onTouchEnd, {passive: true});
|
|
}
|
|
}
|
|
|
|
function onTouchMove(e) {
|
|
const touch = touchCurPos = e.touches.item(0);
|
|
let distX = touch.screenX - touchStartPos.screenX;
|
|
const distY = touch.screenY - touchStartPos.screenY;
|
|
|
|
if (!menuIsMoving) {
|
|
// tan(45°) is 1. Gestures in 0°-45° (< 1) are considered horizontal, so
|
|
// menu must be open; gestures in 45°-90° (>1) are considered vertical, so
|
|
// chat windows must be scrolled.
|
|
if (Math.abs(distY / distX) >= 1) {
|
|
onTouchEnd();
|
|
return;
|
|
}
|
|
|
|
const devicePixelRatio = window.devicePixelRatio || 2;
|
|
|
|
if (Math.abs(distX) > devicePixelRatio) {
|
|
viewport.classList.toggle("menu-dragging", true);
|
|
menuIsMoving = true;
|
|
}
|
|
}
|
|
|
|
if (menuIsOpen) {
|
|
distX += menuWidth;
|
|
}
|
|
|
|
if (distX > menuWidth) {
|
|
distX = menuWidth;
|
|
} else if (distX < 0) {
|
|
distX = 0;
|
|
}
|
|
|
|
menu.style.transform = "translate3d(" + distX + "px, 0, 0)";
|
|
sidebarOverlay.style.opacity = distX / menuWidth;
|
|
}
|
|
|
|
function onTouchEnd() {
|
|
const diff = touchCurPos.screenX - touchStartPos.screenX;
|
|
const absDiff = Math.abs(diff);
|
|
|
|
if (absDiff > menuWidth / 2 || Date.now() - touchStartTime < 180 && absDiff > 50) {
|
|
SlideoutMenu.toggle(diff > 0);
|
|
}
|
|
|
|
document.body.removeEventListener("touchmove", onTouchMove);
|
|
document.body.removeEventListener("touchend", onTouchEnd);
|
|
viewport.classList.toggle("menu-dragging", false);
|
|
menu.style.transform = null;
|
|
sidebarOverlay.style.opacity = null;
|
|
|
|
touchStartPos = null;
|
|
touchCurPos = null;
|
|
touchStartTime = 0;
|
|
menuIsMoving = false;
|
|
}
|
|
|
|
module.exports = SlideoutMenu;
|