import { type Cache, type Directive, Phase } from "@mizu/internal/engine"
export type * from "@mizu/internal/engine"
export const _refresh = {
name: "*refresh",
phase: Phase.POSTPROCESSING,
init(renderer) {
renderer.cache<Cache<typeof _refresh>>(this.name, new WeakMap())
},
setup(_, __, { state }) {
if (!("$refresh" in state)) {
Object.assign(state, { $refresh: false })
}
return { state }
},
async execute(renderer, element, { attributes: [attribute], cache, ...options }) {
const value = await renderer.evaluate(element, attribute.value, options) as string
if (value === null) {
clearTimeout(cache.get(element)?.id)
cache.delete(element)
return
}
const interval = Number.parseInt(`${1000 * Number(value)}`)
if ((Number.isNaN(interval)) || (interval <= 0)) {
renderer.warn(`[${this.name}] expects a finite positive number but got ${value}, ignoring`, element)
return
}
const cached = cache.get(element) ?? cache.set(element, { interval, id: NaN }).get(element)!
if (((cached.interval !== interval) && (!Number.isNaN(cached.id))) || (options.state[renderer.internal("refreshing")])) {
clearTimeout(cached.id)
cached.id = NaN
}
},
cleanup(renderer, element, { cache, ...options }) {
if ((renderer.isComment(element)) && (cache.has(renderer.cache("*").get(element)!))) {
element = renderer.cache("*").get(element)!
clearTimeout(cache.get(element)?.id)
cache.delete(element)
return
}
if (!Number.isNaN(cache.get(element)?.id)) {
return
}
cache.get(element)!.id = setTimeout(() => {
if (element.isConnected) {
renderer.render(element as HTMLElement, { ...options, state: { ...options.state, $refresh: true, [renderer.internal("refreshing")]: true } })
}
}, cache.get(element)!.interval)
},
} as Directive<WeakMap<HTMLElement | Comment, { id: number; interval: number }>>
export default _refresh
|