npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

popupable

v1.6.3

Published

A lightweight, zero-dependency lightbox library using modern JavaScript and CSS.

Readme

popupable

A lightweight, zero-dependency lightbox library using modern JavaScript and CSS. Just add data-popupable to any image!

npm version jsDelivr License: MIT

Live Demo

Features

  • No dependencies
  • Animates open from the element's original position
  • Multiple built-in animation styles, with support for custom ones
  • Works with mouse, touch, and keyboard
  • Gallery groups with swipe, scroll, keyboard, and button navigation
  • Thumbnail strip and image counter
  • Pinch-to-zoom on touch, with optional click/tap-to-zoom support
  • Checkerboard background for transparent images
  • Attribute inheritance - set options on a parent to apply to all children
  • Customizable via CSS variables
  • Works in Vue, React, Svelte, and more

Quick Start

Install via npm

npm install popupable
import "popupable/styles.css"
import "popupable"

Or use via CDN

https://www.jsdelivr.com/package/npm/popupable

Add popups to your images

<img src="photo.jpg" data-popupable>
<img src="thumbnail.jpg" data-popupable data-popupable-src="full-res.jpg">
<img src="photo.jpg" data-popupable data-popupable-title="Sunset" data-popupable-description="Taken in July 2024">

Close by clicking again, or pressing Escape, Backspace, or Delete.

Settings

| Attribute | Description | |---|---| | data-popupable | Required. Marks an image as openable. Optionally set a value to give the popup a CSS id. | | data-popupable-src="url" | A different image to display when the popup is open, e.g. a higher resolution version. | | data-popupable-title="text" | Title text shown alongside the image. | | data-popupable-description="text" | Description text shown alongside the image. | | data-popupable-transparent | Shows a checkerboard background behind transparent images. | | data-popupable-maintain-aspect | Uses the element's rendered aspect ratio instead of the image's natural dimensions. | | data-popupable-no-upscale | Prevents the popup from scaling the image beyond its native resolution. | | data-popupable-zoomable | Enables click/tap-to-zoom and scroll wheel zoom. | | data-popupable-anim="name" | Sets the open/close animation style. | | data-popupable-group="name" | Groups images into a navigable gallery. Can be placed on a container element to group all data-popupable children. | | data-popupable-counter | Shows a "1 / N" counter when in a group. | | data-popupable-thumbnails | Shows a thumbnail strip when in a group. | | data-popupable-order="..." | Controls the order of UI elements. |

Inheritance

All data-popupable-* attributes except data-popupable itself are inherited. If an attribute isn't on the element, popupable walks up the DOM tree to find it on a parent. This lets you set options once on a container instead of repeating them on every child.

Set an attribute to "false" to explicitly unset an inherited value:

<div data-popupable-anim="pop" data-popupable-counter data-popupable-thumbnails>
  <img src="photo1.jpg" data-popupable>
  <img src="photo2.jpg" data-popupable data-popupable-counter="false">
</div>

Advanced Usage

Non-image elements

Any element can be made popupable. Use src or data-popupable-src to specify the image to display:

<div src="photo.jpg" style="background-image: url(photo.jpg)" data-popupable></div>
<div style="background-image: url(photo.jpg)" data-popupable data-popupable-src="photo.jpg"></div>

Hi-res source

Use data-popupable-src to display a different (e.g. higher resolution) image when the popup is open:

<img src="thumbnail.jpg" data-popupable data-popupable-src="full-res.jpg">

Transparent images

Add data-popupable-transparent to show a checkerboard background behind transparent images:

<img src="icon.png" data-popupable data-popupable-transparent>

Maintain aspect ratio

By default, the popup size is calculated from the image's natural dimensions. Add data-popupable-maintain-aspect to use the element's rendered aspect ratio instead, which is useful when the image is cropped or stretched via CSS:

<img src="photo.jpg" data-popupable data-popupable-maintain-aspect>

Prevent upscaling

Add data-popupable-no-upscale to prevent the popup from scaling the image beyond its native resolution:

<img src="pixel-art.png" data-popupable data-popupable-no-upscale>

Zoom

All images support pinch-to-zoom. Add data-popupable-zoomable to also enable tap/click-to-zoom and scroll wheel zoom:

<img src="map.jpg" data-popupable data-popupable-zoomable>

When zoomed in, pan by dragging and zoom in/out with the scroll wheel. Click the image or background, or pinch back to scale 1, to unzoom. A close button also appears in the corner when data-popupable-zoomable is set.

Groups / Galleries

Group multiple images together with data-popupable-group. Users can navigate with arrow buttons, swipe gestures, scroll wheel, or keyboard shortcuts.

Place data-popupable-group directly on elements or on a container to group all data-popupable children:

<!-- On each element -->
<img src="photo1.jpg" data-popupable data-popupable-group="holiday">
<img src="photo2.jpg" data-popupable data-popupable-group="holiday">
<img src="photo3.jpg" data-popupable data-popupable-group="holiday">

<!-- Or on a container (inherited) -->
<div data-popupable-group="holiday">
  <img src="photo1.jpg" data-popupable>
  <img src="photo2.jpg" data-popupable>
  <img src="photo3.jpg" data-popupable>
</div>

Counter and thumbnails

Add data-popupable-counter and/or data-popupable-thumbnails to show those UI elements in a gallery. These inherit, so placing them on the group container applies them to all members:

<div data-popupable-group="holiday" data-popupable-counter data-popupable-thumbnails>
  <img src="photo1.jpg" data-popupable>
  <img src="photo2.jpg" data-popupable>
  <img src="photo3.jpg" data-popupable>
</div>

Custom UI order

Control the order of UI elements around the image with data-popupable-order. The image token marks where the image sits; everything before it goes in the header, everything after goes in the footer.

<!-- Counter above image, thumbnails and content below -->
<img data-popupable data-popupable-order="counter,image,thumbnails,content">

Default order: counter,image,content,thumbnails

Keyboard navigation

| Keys | Action | |---|---| | Page Down D S | Next image | | Page Up A W | Previous image | | Home | First image | | End | Last image | | 19 | Jump to image by number |

Custom IDs

Set the value of data-popupable to give the popup a CSS id, useful for per-popup styling. In a gallery, the id updates as you navigate, so each image can have its own style:

<img src="red.jpg"   data-popupable="theme-red"   data-popupable-group="themed">
<img src="green.jpg" data-popupable="theme-green" data-popupable-group="themed">
<img src="blue.jpg"  data-popupable="theme-blue"  data-popupable-group="themed">
#theme-red   { --popupable-background: #3a0000cc; }
#theme-green { --popupable-background: #003a00cc; }
#theme-blue  { --popupable-background: #00003acc; }

Animation styles

Use data-popupable-anim to choose an open/close animation style. The default is expand.

| Value | Description | |---|---| | expand | Expands from the image's original position (default) | | pop | Scales up from a slightly smaller version at the final position | | line | Expands vertically from a thin horizontal line at the center, like a CRT TV powering on | | float | Fades in while rising upward into position |

<img src="photo.jpg" data-popupable data-popupable-anim="line">

The popup container receives a popupable-anim-{name} class (e.g. popupable-anim-float), which can be used to apply custom CSS for that animation style.

Custom animation styles

Register your own animation style by adding an object to window.popupableAnimTypes. The object must have a position method that receives the original element and the final expanded rect, and returns the starting rect for the open animation (which is also the ending rect for close). Additional static flags control the animation behaviour.

| Property | Type | Description | |---|---|---| | position(el, rect) | function | Returns { top, left, width, height } for the start of the open animation | | styles | boolean | Copies border, outline, and box-shadow from the element to the popup clone, then transitions them out | | fade | boolean | Fades opacity in/out alongside the geometry | | crossfade | boolean | Crossfades between the thumbnail and alternate source image as it opens | | hideSource | boolean | Hides the original element while the popup is open |

window.popupableAnimTypes.myAnim = {
  fade: true,
  position(original, rect) {
    return {
      top: rect.top + rect.height / 2,
      left: rect.left,
      width: rect.width,
      height: 0
    }
  }
}

Customization

Popups can be styled using CSS variables:

:root {
  /* Overlay */
  --popupable-background: #000B;        /* Backdrop color */
  --popupable-blur: 6px;                /* Backdrop blur amount */

  /* UI elements (header, footer, buttons) */
  --popupable-ui-background: #0008;     /* Header/footer/button background */
  --popupable-text-color: #fff;         /* Text and icon color */

  /* Spacing */
  --popupable-screen-padding: 40px;     /* Minimum gap between image and viewport edge */

  /* Animation */
  --popupable-open-duration: .25s;      /* Open/close transition duration */
  --popupable-open-easing: ease;        /* Open/close transition easing */
  --popupable-switch-duration: .25s;    /* Gallery navigation transition duration */
  --popupable-switch-easing: ease;      /* Gallery navigation transition easing */
}

How it works

popupable uses a clone-and-expand technique:

  1. Clones the image at its exact on-screen position and size
  2. Hides the original and appends the clone as a fixed overlay
  3. Animates the clone to fill the viewport, respecting aspect ratio and padding
  4. On close, reverses the animation back to the original position, then removes the overlay

This gives a seamless visual transition with no layout shifts.

License

MIT © Ewan Howell