levkin.ca/shared/stack-scroll.js
ilia cfe1cf3922 Fix stack cover: single sticky folder unit, hide inactive bodies.
Tested all four variants in browser — only active layer body visible while
tabs stay staggered; scroll covers previous layers correctly.

Co-authored-by: Cursor <cursoragent@cursor.com>
2026-05-20 23:11:57 -04:00

54 lines
1.9 KiB
JavaScript

/** Shared scroll-depth tracking + jump-to-layer for stack variants */
export function initStackScroll(options = {}) {
const {
sectionSelector = '.scroll-section',
depthEl = document.getElementById('depth'),
depthPrefix = 'L',
tabSelector = '[data-goto], .jump',
panelSelector = '.folder, .layer, .frame, .unit',
} = options;
const sections = document.querySelectorAll(sectionSelector);
if (!sections.length) return;
const mid = () => window.innerHeight * 0.42;
function updateDepth() {
let active = 0;
sections.forEach((sec) => {
const r = sec.getBoundingClientRect();
if (r.top <= mid() && r.bottom > mid()) active = Number(sec.dataset.layer);
});
if (depthEl) depthEl.textContent = `${depthPrefix}${active}`;
document.querySelectorAll('.stack-ruler button, .stack-ruler [data-goto], .tab-rail button, .tab[data-goto]').forEach((el) => {
const n = el.dataset.layer ?? el.dataset.goto;
if (n === undefined) return;
el.classList.toggle('active', Number(n) === active);
});
sections.forEach((sec) => {
const layer = Number(sec.dataset.layer);
const isActive = layer === active;
sec.classList.toggle('is-active', isActive);
const panel = sec.querySelector(panelSelector);
if (!panel) return;
panel.style.zIndex = isActive ? 100 : 10 + layer;
});
}
document.querySelectorAll(tabSelector).forEach((tab) => {
tab.addEventListener('click', (e) => {
e.preventDefault();
const layer = tab.dataset.goto;
const target = document.querySelector(`${sectionSelector}[data-layer="${layer}"]`);
if (!target) return;
const y = target.getBoundingClientRect().top + window.scrollY - 48;
window.scrollTo({ top: Math.max(0, y), behavior: 'smooth' });
});
});
window.addEventListener('scroll', updateDepth, { passive: true });
updateDepth();
}