Prevent Scroll

Utility that prevents scrolling outside of a given DOM element. Works by preventing events that else would lead to scrolling.

Features

  • Supports nested scroll containers
  • Works both vertically and horizontally
  • Removes the body scrollbar without layout shift

Usage

By default, it also hides the scrollbar of the body element and adds padding to it to prevent the page from jumping. This behavior can be disabled and modified with the hideScrollbar, preventScrollbarShift, and preventScrollbarShiftMode props.

It also adds the CSS variable --scrollbar-width to the body element, indicating the width of the currently removed scrollbar. You can use this variable to add padding to fixed elements, like a topbar.

import { createPreventScroll } from 'corvu'
const DialogContent: Component<{
  open: boolean
}> = (props) => {
  const [ref, setRef] = createSignal(null)

  createPreventScroll({
    element: ref,
    enabled: () => props.open,
    hideScrollbar: true, // default
  })

  return (
    <Show when={props.open}>
      <div ref={setRef}>Dialog</div>
    </Show>
  )
}

Use the --scrollbar-width CSS variable to add padding to fixed elements to prevent the content from shifting when the scrollbar is removed:

<header
  class="fixed top-0 inset-x-0 z-50"
  style={{
    'padding-right': 'var(--scrollbar-width, 0)',
  }}
>
  Header
</header>

API reference

createPreventScroll

Function

Prevents scroll outside of the given element.

Props

element

MaybeAccessor<null | HTMLElement>

Prevent scroll outside of this element. If the element is null, scroll will be prevented on the whole page. *Default = null*

enabled

MaybeAccessor<boolean>

Whether scroll should be prevented. *Default = true*

hideScrollbar

MaybeAccessor<boolean>

Whether the scrollbar of the <​body> element should be hidden. *Default = true*

preventScrollbarShift

MaybeAccessor<boolean>

Whether padding should be added to the <​body> element to avoid layout shift. *Default = true*

preventScrollbarShiftMode

MaybeAccessor<'margin' | 'padding'>

Whether padding or margin should be used to avoid layout shift. *Default = 'padding'*

allowPinchZoom

MaybeAccessor<boolean>

Whether pinch zoom should be allowed. *Default = false*

corvu@0.2.3

Developed and designed by Jasmin