mirror of
https://github.com/Kpa-clawbot/meshcore-analyzer.git
synced 2026-06-04 08:51:19 +00:00
58282c91d8
## Summary Three follow-up fixes for #1065 gesture-hint discoverability: 1. **Touch-capability gate.** New `hasTouchCapability()` helper probes `'ontouchstart' in window`, `navigator.maxTouchPoints`, and `(pointer: coarse)`. Every `HINTS[*].relevant()` predicate now returns `false` immediately on mouse-only viewports, so desktop browsers no longer get "swipe a row left" tips. 2. **`width: fit-content` on the pill wrap.** The `.gesture-hint` block previously had no explicit width and defaulted to block-level full-width. Combined with `translateX(-50%)` on `.gesture-hint-bottom` this rendered as a 100vw-wide bar centered with a negative-X transform, i.e. pushed off-screen-left on narrow viewports (384px wrap on 390px viewport). 3. **CSS-parse safety.** Moved the in-body comment (which contained an em-dash) outside the rule block. An earlier attempt to add `width: fit-content` together with an in-body em-dash comment caused the parent `.gesture-hint` rule to vanish from the CSSOM in Chrome (children `.gesture-hint-*` remained). Putting the comment above the block sidesteps the parser bug. ## Test `test-issue-1065-gesture-hints-gates.js` — pure source-file assertions, no browser required. Red commit first (7 fails), green commit second (10/10 pass). Wired into `test-all.sh`. ## Verification After hot-deploy on staging: - Desktop (no touch): `document.querySelectorAll('.gesture-hint').length` === 0 - Mobile emulated (touch): hint rendered, `getBoundingClientRect().x >= 0`, `width <= 360`, `width < viewport_width` - CSSOM: parent `.gesture-hint` rule present with `width: fit-content` + `max-width: 360px` --------- Co-authored-by: openclaw-bot <bot@openclaw.local>
241 lines
8.0 KiB
JavaScript
241 lines
8.0 KiB
JavaScript
/* gesture-hints.js — Issue #1065
|
|
* First-visit gesture discoverability hints.
|
|
*
|
|
* - localStorage namespace: meshcore-gesture-hints-<hint>
|
|
* keys: row-swipe, tab-swipe, edge-drawer, pull-refresh
|
|
* value: "seen"
|
|
* - Show hint 800ms after page settle; auto-fade 8s; "Got it" dismisses.
|
|
* - aria-live=polite, role=status, no focus stealing, pointer-events:none.
|
|
* - prefers-reduced-motion: animation-name: none (style.css handles via media query).
|
|
* - Singleton + cleanup: module-scoped guard; SPA re-mount must not re-show dismissed.
|
|
* - Pull-to-refresh hint only when .pull-to-reconnect element exists in DOM.
|
|
* - Edge-drawer hint only at viewport > 768px (where edge-swipe drawer applies).
|
|
* - Row-swipe hint only on table pages: /#/packets, /#/nodes, etc.
|
|
*/
|
|
(function () {
|
|
'use strict';
|
|
if (window.__gestureHints1065Init) {
|
|
window.__gestureHints1065Init++;
|
|
return;
|
|
}
|
|
window.__gestureHints1065Init = 1;
|
|
|
|
var NS = 'meshcore-gesture-hints-';
|
|
// #1244: gesture hints are bottom-anchored pills. On /live they get
|
|
// buried below the absolute-positioned VCR bar (+ safe-area inset),
|
|
// appearing as orphan "Got it" litter visible only after scrolling.
|
|
// Option (a) from #1244 — disable hints on /live entirely. Swipe-nav
|
|
// discoverability doesn't apply on Live anyway (map drag, VCR
|
|
// controls, and feed all own touch).
|
|
function onLiveRoute() {
|
|
var h = location.hash || '';
|
|
return /^#\/live(\/|$|\?)/.test(h);
|
|
}
|
|
// #1065 follow-up: hints must only appear on touch-capable viewports.
|
|
// Mouse-only desktops (e.g. analyzer.00id.net opened in Chrome on a
|
|
// workstation) were getting "swipe a row left" tips that make no sense.
|
|
// Three independent probes — any positive answer counts.
|
|
function hasTouchCapability() {
|
|
try {
|
|
if ('ontouchstart' in window) return true;
|
|
if (navigator.maxTouchPoints && navigator.maxTouchPoints > 0) return true;
|
|
if (window.matchMedia && window.matchMedia('(pointer: coarse)').matches) return true;
|
|
} catch (_e) {}
|
|
return false;
|
|
}
|
|
var HINTS = {
|
|
'row-swipe': {
|
|
key: NS + 'row-swipe',
|
|
text: 'Tip: swipe a row left for quick actions.',
|
|
relevant: function () {
|
|
if (!hasTouchCapability()) return false;
|
|
if (onLiveRoute()) return false; // #1244
|
|
var h = location.hash || '';
|
|
return /^#\/(packets|nodes)/.test(h);
|
|
},
|
|
position: 'bottom',
|
|
},
|
|
'tab-swipe': {
|
|
key: NS + 'tab-swipe',
|
|
text: 'Tip: swipe left or right to switch tabs.',
|
|
relevant: function () {
|
|
if (!hasTouchCapability()) return false;
|
|
if (onLiveRoute()) return false; // #1244
|
|
return !!document.querySelector('[data-bottom-nav]');
|
|
},
|
|
position: 'bottom',
|
|
},
|
|
'edge-drawer': {
|
|
key: NS + 'edge-drawer',
|
|
text: 'Tip: swipe in from the left edge to open navigation.',
|
|
relevant: function () {
|
|
if (!hasTouchCapability()) return false;
|
|
if (onLiveRoute()) return false; // #1244
|
|
// nav-drawer.js: NARROW_MAX=768; edge-swipe drawer is the WIDE
|
|
// (>768) layout's nav UI. Below 768, the bottom-nav owns navigation.
|
|
return window.innerWidth > 768 && !!document.querySelector('.nav-drawer, [data-nav-drawer]');
|
|
},
|
|
position: 'top-left',
|
|
},
|
|
'pull-refresh': {
|
|
key: NS + 'pull-refresh',
|
|
text: 'Tip: pull down to refresh the connection.',
|
|
relevant: function () {
|
|
if (!hasTouchCapability()) return false;
|
|
if (onLiveRoute()) return false; // #1244
|
|
return !!document.querySelector('.pull-to-reconnect');
|
|
},
|
|
position: 'top',
|
|
},
|
|
};
|
|
|
|
var SHOW_DELAY_MS = 800;
|
|
var AUTO_FADE_MS = 8000;
|
|
|
|
var _shown = Object.create(null); // hint id → element (currently rendered)
|
|
var _scheduledTimer = null;
|
|
var _routeChangeBound = false;
|
|
|
|
function isSeen(id) {
|
|
try { return localStorage.getItem(HINTS[id].key) === 'seen'; }
|
|
catch (_e) { return false; }
|
|
}
|
|
function markSeen(id) {
|
|
try { localStorage.setItem(HINTS[id].key, 'seen'); } catch (_e) {}
|
|
}
|
|
function clearAll() {
|
|
try {
|
|
Object.keys(HINTS).forEach(function (id) { localStorage.removeItem(HINTS[id].key); });
|
|
} catch (_e) {}
|
|
}
|
|
|
|
function buildHintEl(id) {
|
|
var def = HINTS[id];
|
|
var wrap = document.createElement('div');
|
|
wrap.className = 'gesture-hint gesture-hint-' + def.position;
|
|
// Belt-and-suspenders: inline style guarantees pointer-events:none
|
|
// regardless of CSS load order or cascade collisions. The hint must
|
|
// never capture clicks; only the inner button does (via .gesture-hint-inner).
|
|
wrap.style.pointerEvents = 'none';
|
|
wrap.setAttribute('data-gesture-hint', id);
|
|
wrap.setAttribute('role', 'status');
|
|
wrap.setAttribute('aria-live', 'polite');
|
|
wrap.setAttribute('aria-atomic', 'true');
|
|
|
|
var inner = document.createElement('div');
|
|
inner.className = 'gesture-hint-inner';
|
|
|
|
var msg = document.createElement('span');
|
|
msg.className = 'gesture-hint-text';
|
|
msg.textContent = def.text;
|
|
inner.appendChild(msg);
|
|
|
|
var btn = document.createElement('button');
|
|
btn.type = 'button';
|
|
btn.className = 'gesture-hint-dismiss';
|
|
btn.setAttribute('data-gesture-hint-dismiss', '');
|
|
btn.setAttribute('aria-label', 'Dismiss hint');
|
|
btn.textContent = 'Got it';
|
|
btn.addEventListener('click', function (e) {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
dismiss(id);
|
|
});
|
|
inner.appendChild(btn);
|
|
|
|
wrap.appendChild(inner);
|
|
return wrap;
|
|
}
|
|
|
|
function show(id) {
|
|
if (_shown[id]) return;
|
|
if (isSeen(id)) return;
|
|
var def = HINTS[id];
|
|
if (!def || !def.relevant()) return;
|
|
|
|
var el = buildHintEl(id);
|
|
document.body.appendChild(el);
|
|
_shown[id] = el;
|
|
|
|
// Auto-fade after AUTO_FADE_MS — does NOT mark seen; user must explicitly dismiss
|
|
// (per AC: "Got it" button clears the flag).
|
|
var fadeTimer = setTimeout(function () {
|
|
if (_shown[id] === el) {
|
|
el.classList.add('gesture-hint-fading');
|
|
setTimeout(function () {
|
|
if (el.parentNode) el.parentNode.removeChild(el);
|
|
if (_shown[id] === el) delete _shown[id];
|
|
}, 350);
|
|
}
|
|
}, AUTO_FADE_MS);
|
|
el._gestureHintFadeTimer = fadeTimer;
|
|
}
|
|
|
|
function dismiss(id) {
|
|
var el = _shown[id];
|
|
markSeen(id);
|
|
if (el) {
|
|
if (el._gestureHintFadeTimer) clearTimeout(el._gestureHintFadeTimer);
|
|
if (el.parentNode) el.parentNode.removeChild(el);
|
|
delete _shown[id];
|
|
}
|
|
}
|
|
|
|
function scheduleHints() {
|
|
if (_scheduledTimer) clearTimeout(_scheduledTimer);
|
|
_scheduledTimer = setTimeout(function () {
|
|
_scheduledTimer = null;
|
|
Object.keys(HINTS).forEach(function (id) {
|
|
if (!isSeen(id)) show(id);
|
|
});
|
|
}, SHOW_DELAY_MS);
|
|
}
|
|
|
|
function onRouteChange() {
|
|
// Remove hints that are no longer relevant for the new route.
|
|
Object.keys(_shown).slice().forEach(function (id) {
|
|
var def = HINTS[id];
|
|
if (!def || !def.relevant()) {
|
|
var el = _shown[id];
|
|
if (el && el._gestureHintFadeTimer) clearTimeout(el._gestureHintFadeTimer);
|
|
if (el && el.parentNode) el.parentNode.removeChild(el);
|
|
delete _shown[id];
|
|
}
|
|
});
|
|
// Re-evaluate: show any not-yet-seen relevant hints.
|
|
scheduleHints();
|
|
}
|
|
|
|
function init() {
|
|
if (!_routeChangeBound) {
|
|
_routeChangeBound = true;
|
|
window.addEventListener('hashchange', onRouteChange);
|
|
}
|
|
scheduleHints();
|
|
}
|
|
|
|
if (document.readyState === 'loading') {
|
|
document.addEventListener('DOMContentLoaded', init, { once: true });
|
|
} else {
|
|
init();
|
|
}
|
|
|
|
window.GestureHints = {
|
|
show: show,
|
|
dismiss: dismiss,
|
|
reset: function () {
|
|
clearAll();
|
|
// Remove any visible.
|
|
Object.keys(_shown).slice().forEach(function (id) {
|
|
var el = _shown[id];
|
|
if (el && el._gestureHintFadeTimer) clearTimeout(el._gestureHintFadeTimer);
|
|
if (el && el.parentNode) el.parentNode.removeChild(el);
|
|
delete _shown[id];
|
|
});
|
|
},
|
|
_keys: function () {
|
|
return Object.keys(HINTS).map(function (id) { return HINTS[id].key; });
|
|
},
|
|
};
|
|
})();
|