From 769826edbd9ee2144e9272656dfbe6155b24d67d Mon Sep 17 00:00:00 2001 From: Mads Erik Forberg Date: Wed, 3 Jan 2024 14:20:07 +0100 Subject: [PATCH] feat: support setting rootMargin for `client:visible` (#9363) * feat: support setting rootMargin for `client:visible` This support adding optional `rootMargin` to the `IntersectionObserver` options. This gives the developer the optional choice to hydrate a bit before the astro-island enters the viewport. * chore: update type for client:visible * chore: added changeset * chore: update types * fix: check if value is string * Update stupid-peas-juggle.md * fix: update .changeset/stupid-peas-juggle.md Co-authored-by: Sarah Rainsberger --------- Co-authored-by: Florian Lefebvre Co-authored-by: Bjorn Lu Co-authored-by: Sarah Rainsberger --- .changeset/stupid-peas-juggle.md | 10 ++++++++++ packages/astro/src/@types/astro.ts | 2 +- packages/astro/src/runtime/client/visible.ts | 8 ++++++-- 3 files changed, 17 insertions(+), 3 deletions(-) create mode 100644 .changeset/stupid-peas-juggle.md diff --git a/.changeset/stupid-peas-juggle.md b/.changeset/stupid-peas-juggle.md new file mode 100644 index 000000000000..1e01c0996636 --- /dev/null +++ b/.changeset/stupid-peas-juggle.md @@ -0,0 +1,10 @@ +--- +'astro': minor +--- + +Extends the `client:visible` directive by adding an optional `rootMargin` property. This allows a component to be hydrated when it is close to the viewport instead of waiting for it to become visible. + +```html + + +``` diff --git a/packages/astro/src/@types/astro.ts b/packages/astro/src/@types/astro.ts index e9c73fc90f3a..ed6015757cc9 100644 --- a/packages/astro/src/@types/astro.ts +++ b/packages/astro/src/@types/astro.ts @@ -70,7 +70,7 @@ export interface AstroBuiltinProps { 'client:load'?: boolean; 'client:idle'?: boolean; 'client:media'?: string; - 'client:visible'?: boolean; + 'client:visible'?: string|boolean; 'client:only'?: boolean | string; } diff --git a/packages/astro/src/runtime/client/visible.ts b/packages/astro/src/runtime/client/visible.ts index de36b29098a5..9e625ca23df9 100644 --- a/packages/astro/src/runtime/client/visible.ts +++ b/packages/astro/src/runtime/client/visible.ts @@ -5,12 +5,16 @@ import type { ClientDirective } from '../../@types/astro.js'; * We target the children because `astro-island` is set to `display: contents` * which doesn't work with IntersectionObserver */ -const visibleDirective: ClientDirective = (load, _options, el) => { +const visibleDirective: ClientDirective = (load, options, el) => { const cb = async () => { const hydrate = await load(); await hydrate(); }; + const ioOptions = { + rootMargin: typeof options.value === 'string' ? options.value : undefined, + }; + const io = new IntersectionObserver((entries) => { for (const entry of entries) { if (!entry.isIntersecting) continue; @@ -19,7 +23,7 @@ const visibleDirective: ClientDirective = (load, _options, el) => { cb(); break; // break loop on first match } - }); + }, ioOptions); for (const child of el.children) { io.observe(child);