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 🙏

© 2025 – Pkg Stats / Ryan Hefner

@luciodale/swipe-bar

v1.0.9

Published

Toggling and swiping sidebars like in a native app, built with React. Inspired by the native iOS swipe gestures.

Readme

SwipeBar

🎮 LIVE DEMO📦 NPM Package

A native swipe bar experience

SwipeBar provides an intuitive, native-app-like experience for mobile and desktop sidebars. Swipe from the edge to open, drag to close, and enjoy smooth animations with full customization control.

Features

  • 📦 Lightweight - zero dependencies
  • 📱 Native Mobile Gestures - Swipe from screen edges just like native apps
  • 🖱️ Desktop Support - Works seamlessly with mouse interactions
  • 🎨 Fully Customizable - Complete control over styling, animations, and behavior
  • 🎯 Smart Positioning - Automatically adapts between absolute and relative positioning
  • 🎭 Custom Toggles - Use built-in toggle icons or provide your own components
  • Performant - Relies on transform and opacity properties
  • 🎚️ Configurable Thresholds - Fine-tune edge activation and drag sensitivity
  • 🌓 Overlay Support - Optional backdrop overlay with customizable colors
  • 📏 Responsive - Media query support for different screen sizes
  • 🔧 Programmatic Control - Open and close sidebars from anywhere in your app

Installation

npm install @luciodale/swipe-bar
# or
yarn add @luciodale/swipe-bar
# or
bun add @luciodale/swipe-bar

Quick Start

1. Install

npm install @luciodale/swipe-bar

2. Add the Provider

Wrap your app with SwipeBarProvider:

import { SwipeBarProvider } from "@luciodale/swipe-bar";

function App() {
  return (
    <SwipeBarProvider>
      {/* Your app content */}
    </SwipeBarProvider>
  );
}

3. Add Left Sidebar

import { SwipeBarLeft } from "@luciodale/swipe-bar";

<SwipeBarLeft className="bg-blue-500 ...">
  <nav>Your sidebar content</nav>
</SwipeBarLeft>

4. Add Right Sidebar

import { SwipeBarRight } from "@luciodale/swipe-bar";

<SwipeBarRight className="bg-blue-500 ...">
  <div>Settings panel</div>
</SwipeBarRight>

5. Programmatic Control

Use the context hook to control sidebars from anywhere:

import { useSwipeBarContext } from "@luciodale/swipe-bar";

function MyComponent() {
  const { openSidebar, closeSidebar, isLeftOpen, isRightOpen } = useSwipeBarContext();
  
  return (
    <button onClick={() => openSidebar("left")}>
      Open Left Sidebar
    </button>
  );
}

Props & Configuration

Provider Props

Configure global defaults for all sidebars:

<SwipeBarProvider
  sidebarWidthPx={320}
  transitionMs={300}
  edgeActivationWidthPx={40}
  dragActivationDeltaPx={20}
  showOverlay={true}
  isAbsolute={false}
  overlayBackgroundColor="rgba(0, 0, 0, 0.5)"
  toggleIconColor="white"
  toggleIconSizePx={40}
  toggleIconEdgeDistancePx={40}
  showToggle={true}
  mediaQueryWidth={640}
  swipeBarZIndex={30}
  toggleZIndex={15}
  overlayZIndex={20}
>
  {children}
</SwipeBarProvider>

Sidebar Props

Both SwipeBarLeft and SwipeBarRight accept the same props, which override provider defaults:

| Prop | Type | Default | Description | |------|------|---------|-------------| | className | string | - | CSS classes for the sidebar container | | ToggleComponent | ReactNode | - | Custom toggle button component | | sidebarWidthPx | number | 320 | Width of the sidebar in pixels | | transitionMs | number | 200 | Animation duration in milliseconds | | edgeActivationWidthPx | number | 40 | Touch zone width from screen edge to activate swipe | | dragActivationDeltaPx | number | 20 | Minimum drag distance to activate sidebar | | showOverlay | boolean | true | Show backdrop overlay when sidebar is open | | isAbsolute | boolean | false | Use absolute positioning (overlay mode) | | overlayBackgroundColor | string | "rgba(0, 0, 0, 0.5)" | Overlay background color | | toggleIconColor | string | "white" | Color of the built-in toggle icon | | toggleIconSizePx | number | 40 | Size of the toggle icon | | toggleIconEdgeDistancePx | number | 40 | Distance of toggle from screen edge | | showToggle | boolean | true | Show the toggle button | | mediaQueryWidth | number | 640 | Max screen width for swipe gestures (in pixels) | | swipeBarZIndex | number | 30 | Z-index for left and right swipe bars | | toggleZIndex | number | 15 | Z-index for left and right toggle buttons | | overlayZIndex | number | 20 | Z-index for the overlay backdrop |

Context API

const {
  isLeftOpen,           // boolean - is left sidebar open
  isRightOpen,          // boolean - is right sidebar open
  openSidebar,          // (side: "left" | "right") => void
  closeSidebar,         // (side: "left" | "right") => void
  globalOptions,        // current global options
  setGlobalOptions,     // update global options
} = useSwipeBarContext();

Key Concepts

isAbsolute Mode

By default, sidebars push content to the side. Set isAbsolute={true} to make sidebars overlay on top of content instead:

<SwipeBarLeft isAbsolute={true}>
  {/* This sidebar will overlay content */}
</SwipeBarLeft>

Note: On mobile (below mediaQueryWidth), sidebars automatically switch to absolute positioning for a better mobile experience.

Media Query Width

Control when swipe gestures are enabled based on screen size:

<SwipeBarProvider mediaQueryWidth={640}>
  {/* Swipe gestures only work on screens ≤640px wide */}
</SwipeBarProvider>

This is useful if you want swipe interactions only on mobile/tablet devices.

Toggle Button Behavior

The built-in toggle button automatically hides when:

  • The overlay is visible to handle closing instead

Custom Toggles

Replace the default toggle with your own component:

<SwipeBarLeft 
  ToggleComponent={
    <button className="my-custom-toggle">
      ☰
    </button>
  }
>
  {/* content */}
</SwipeBarLeft>

Examples

Basic Sidebar

import { SwipeBarProvider, SwipeBarLeft } from "@luciodale/swipe-bar";

function App() {
  return (
    <SwipeBarProvider>
      <SwipeBarLeft className="bg-gray-800 text-white p-4">
        <h2>Navigation</h2>
        <nav>
          <a href="/">Home</a>
          <a href="/about">About</a>
          <a href="/contact">Contact</a>
        </nav>
      </SwipeBarLeft>
      
      <main>
        {/* Your main content */}
      </main>
    </SwipeBarProvider>
  );
}

Glassmorphism Sidebar

<SwipeBarLeft 
  className="bg-white/10 backdrop-blur-2xl border-r border-white/20"
  overlayBackgroundColor="rgba(0, 0, 0, 0.3)"
>
  {/* Glass effect sidebar content */}
</SwipeBarLeft>

Full-Width Mobile Sidebar

<SwipeBarLeft 
  sidebarWidthPx={window.innerWidth}
  isAbsolute={true}
>
  {/* Full screen mobile menu */}
</SwipeBarLeft>

Programmatic Control

function Header() {
  const { openSidebar } = useSwipeBarContext();
  
  return (
    <header>
      <button onClick={() => openSidebar("left")}>Menu</button>
      <h1>My App</h1>
      <button onClick={() => openSidebar("right")}>Settings</button>
    </header>
  );
}

Browser Support

  • Chrome/Edge (latest)
  • Firefox (latest)
  • Safari (latest)
  • Mobile Safari (iOS 12+)
  • Chrome Android (latest)

Touch events and smooth animations are supported across all modern browsers.

Contributing

Contributions are welcome! Feel free to:

  • Report bugs
  • Suggest new features
  • Submit pull requests
  • Improve documentation

License

MIT