2021-06-09 01:19:03 +12:00
|
|
|
<script>
|
|
|
|
import { onMount, onDestroy } from "svelte"
|
|
|
|
import { builderStore } from "../store"
|
|
|
|
import Indicator from "./Indicator.svelte"
|
2021-06-10 20:05:08 +12:00
|
|
|
import { domDebounce } from "../utils/domDebounce"
|
2021-06-09 01:19:03 +12:00
|
|
|
|
|
|
|
let indicators = []
|
|
|
|
let interval
|
|
|
|
let componentId
|
|
|
|
let componentName
|
|
|
|
|
|
|
|
const updatePosition = () => {
|
|
|
|
let newIndicators = []
|
|
|
|
|
|
|
|
if (componentId) {
|
|
|
|
const parents = document.getElementsByClassName(componentId)
|
|
|
|
|
|
|
|
// Batch reads to minimize reflow
|
|
|
|
const scrollX = window.scrollX
|
|
|
|
const scrollY = window.scrollY
|
|
|
|
|
|
|
|
for (let i = 0; i < parents.length; i++) {
|
|
|
|
const child = parents[i]?.childNodes?.[0]
|
|
|
|
if (child) {
|
|
|
|
const elBounds = child.getBoundingClientRect()
|
|
|
|
newIndicators.push({
|
2021-06-09 02:16:37 +12:00
|
|
|
top: elBounds.top + scrollY - 2,
|
|
|
|
left: elBounds.left + scrollX - 2,
|
|
|
|
width: elBounds.width + 4,
|
|
|
|
height: elBounds.height + 4,
|
2021-06-09 01:19:03 +12:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
indicators = newIndicators
|
|
|
|
}
|
2021-06-10 20:05:08 +12:00
|
|
|
const debouncedUpdate = domDebounce(updatePosition)
|
2021-06-09 01:19:03 +12:00
|
|
|
|
|
|
|
const onMouseOver = e => {
|
|
|
|
const element = e.target.closest("[data-type='component']")
|
2021-06-10 20:05:08 +12:00
|
|
|
const newId = element?.dataset?.id
|
|
|
|
const newName = element?.dataset?.name
|
|
|
|
if (newId !== componentId) {
|
|
|
|
componentId = newId
|
|
|
|
componentName = newName
|
|
|
|
debouncedUpdate()
|
|
|
|
}
|
2021-06-09 01:19:03 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
const onMouseLeave = () => {
|
|
|
|
componentId = null
|
|
|
|
componentName = null
|
|
|
|
}
|
|
|
|
|
|
|
|
onMount(() => {
|
2021-06-10 20:05:08 +12:00
|
|
|
debouncedUpdate()
|
|
|
|
interval = setInterval(debouncedUpdate, 100)
|
|
|
|
document.addEventListener("mouseover", onMouseOver)
|
|
|
|
document.addEventListener("mouseleave", onMouseLeave)
|
|
|
|
document.addEventListener("scroll", debouncedUpdate, true)
|
2021-06-09 01:19:03 +12:00
|
|
|
})
|
|
|
|
|
|
|
|
onDestroy(() => {
|
|
|
|
clearInterval(interval)
|
2021-06-10 20:05:08 +12:00
|
|
|
document.removeEventListener("mouseover", onMouseOver)
|
|
|
|
document.removeEventListener("mouseleave", onMouseLeave)
|
|
|
|
document.removeEventListener("scroll", debouncedUpdate, true)
|
2021-06-09 01:19:03 +12:00
|
|
|
})
|
|
|
|
</script>
|
|
|
|
|
2021-06-10 20:05:08 +12:00
|
|
|
{#key componentId}
|
|
|
|
{#if componentId !== $builderStore.selectedComponentId}
|
|
|
|
{#each indicators as indicator, idx}
|
|
|
|
<Indicator
|
|
|
|
top={indicator.top}
|
|
|
|
left={indicator.left}
|
|
|
|
width={indicator.width}
|
|
|
|
height={indicator.height}
|
|
|
|
text={idx === 0 ? componentName : null}
|
|
|
|
color="rgb(120, 170, 244)"
|
|
|
|
transition
|
|
|
|
/>
|
|
|
|
{/each}
|
|
|
|
{/if}
|
|
|
|
{/key}
|