111 lines
2.7 KiB
Svelte
111 lines
2.7 KiB
Svelte
|
<script>
|
||
|
import { onMount, onDestroy } from "svelte"
|
||
|
import Indicator from "./Indicator.svelte"
|
||
|
import { domDebounce } from "../utils/domDebounce"
|
||
|
|
||
|
export let componentId
|
||
|
export let color
|
||
|
export let transition
|
||
|
export let zIndex
|
||
|
|
||
|
let indicators = []
|
||
|
let interval
|
||
|
let text
|
||
|
$: visibleIndicators = indicators.filter(x => x.visible)
|
||
|
|
||
|
let updating = false
|
||
|
let observers = []
|
||
|
let callbackCount = 0
|
||
|
let nextIndicators = []
|
||
|
|
||
|
const createIntersectionCallback = idx => entries => {
|
||
|
if (callbackCount >= observers.length) {
|
||
|
return
|
||
|
}
|
||
|
nextIndicators[idx].visible = entries[0].isIntersecting
|
||
|
if (++callbackCount === observers.length) {
|
||
|
indicators = nextIndicators
|
||
|
updating = false
|
||
|
}
|
||
|
}
|
||
|
|
||
|
const updatePosition = () => {
|
||
|
if (updating) {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// Sanity check
|
||
|
if (!componentId) {
|
||
|
indicators = []
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// Reset state
|
||
|
updating = true
|
||
|
callbackCount = 0
|
||
|
observers.forEach(o => o.disconnect())
|
||
|
observers = []
|
||
|
nextIndicators = []
|
||
|
|
||
|
// Determine next set of indicators
|
||
|
const parents = document.getElementsByClassName(componentId)
|
||
|
if (parents.length) {
|
||
|
text = parents[0].dataset.name
|
||
|
}
|
||
|
|
||
|
// Batch reads to minimize reflow
|
||
|
const scrollX = window.scrollX
|
||
|
const scrollY = window.scrollY
|
||
|
|
||
|
// Extract valid children
|
||
|
const children = Array.from(parents)
|
||
|
.map(parent => parent?.childNodes?.[0])
|
||
|
.filter(child => child != null)
|
||
|
|
||
|
children.forEach((child, idx) => {
|
||
|
const callback = createIntersectionCallback(idx)
|
||
|
const threshold = children.length > 1 ? 1 : 0
|
||
|
const observer = new IntersectionObserver(callback, { threshold })
|
||
|
observer.observe(child)
|
||
|
observers.push(observer)
|
||
|
|
||
|
const elBounds = child.getBoundingClientRect()
|
||
|
nextIndicators.push({
|
||
|
top: elBounds.top + scrollY - 2,
|
||
|
left: elBounds.left + scrollX - 2,
|
||
|
width: elBounds.width + 4,
|
||
|
height: elBounds.height + 4,
|
||
|
visible: false,
|
||
|
})
|
||
|
})
|
||
|
}
|
||
|
const debouncedUpdate = domDebounce(updatePosition)
|
||
|
|
||
|
onMount(() => {
|
||
|
debouncedUpdate()
|
||
|
interval = setInterval(debouncedUpdate, 100)
|
||
|
document.addEventListener("scroll", debouncedUpdate, true)
|
||
|
})
|
||
|
|
||
|
onDestroy(() => {
|
||
|
clearInterval(interval)
|
||
|
document.removeEventListener("scroll", debouncedUpdate, true)
|
||
|
observers.forEach(o => o.disconnect())
|
||
|
})
|
||
|
</script>
|
||
|
|
||
|
{#key componentId}
|
||
|
{#each visibleIndicators as indicator, idx}
|
||
|
<Indicator
|
||
|
top={indicator.top}
|
||
|
left={indicator.left}
|
||
|
width={indicator.width}
|
||
|
height={indicator.height}
|
||
|
text={idx === 0 ? text : null}
|
||
|
{transition}
|
||
|
{zIndex}
|
||
|
{color}
|
||
|
/>
|
||
|
{/each}
|
||
|
{/key}
|