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

@thi.ng/imgui

v3.2.72

Published

Immediate mode GUI with flexible state handling & data only shape output

Readme

@thi.ng/imgui

npm version npm downloads Mastodon Follow

[!NOTE] This is one of 214 standalone projects, maintained as part of the @thi.ng/umbrella ecosystem and anti-framework.

🚀 Please help me to work full-time on these projects by sponsoring me. Thank you! ❤️

About

Immediate mode GUI with flexible state handling & data only shape output.

screenshot

Currently still somewhat bare-bones, but already usable & customizable immediate mode GUI implementation, primarily for @thi.ng/hdom-canvas / @thi.ng/hiccup-canvas and @thi.ng/webgl, however with no direct dependency on either and only outputting data structures.

IMGUI components are largely ephemeral and expressed as simple functions, producing a visual representation of a user state value. IMGUIs are reconstructed from scratch each frame and don't exist otherwise (apart from some rudimentary input state & config). If a component's function isn't called again, it won't exist in the next frame shown to the user. Components only return a new value if an user interaction produced a change. Additionally, each component produces a number of shapes & text labels, all of which are collected internally and are, from the user's POV, a mere side effect. At the end of the update cycle IMGUI produces a tree of @thi.ng/hiccup-canvas compatible elements, which can be easily converted into other formats (incl. SVG).

Note: The WebGL conversion still in the early stages and not yet published, pending ongoing development in other packages...

Current features

  • No direct user state mutation (unlike most other IMGUI impls)
  • Flexible & nestable grid layout with support for cell-spans
  • Theme stack for scoped theme switches / overrides
  • Stack for scoped disabled GUI elements & to create modals
  • Hashing & caching of component local state & draws shapes / resources
  • Hover-based mouse cursor overrides
  • Hover tooltips
  • Re-usable hover & activation behaviors (for creating new components)
  • Fully keyboard controllable & Tab-focus switching / highlighting
  • All built-in components based on @thi.ng/geom shape primitives

Available components / widgets

The above screenshot shows most of the currently available components:

  • Push button (horizontal / vertical)
  • Icon button (w/ opt text label)
  • 2x dial types & dial groups (h / v)
  • Dropdown
  • Radial menu
  • Radio button group (h / v)
  • Ramp (keyframe-based control curve editor)
  • Slider & slider groups (h / v)
  • Text input (single line, filtered input)
  • Text label
  • Toggle button
  • XY pad

All components are:

  • Skinnable (via theme)
  • Keyboard controllable (incl. focus switching)
  • Support tooltips

State handling

All built-in components only return a result value if the component was interacted with and would result in a state change (i.e. a slider has been dragged or button pressed). So, unlike the traditional IMGUI pattern (esp. in languages with pointer support), none of the components here directly manipulate user state and this task is left entirely to the user. This results in somewhat slightly more verbose code, but offers complete freedom WRT how user state is & can be organized. Also, things like undo / redo become easier to handle this way.

import { Atom, History } from "@thi.ng/atom";
import { toggle } from "@thi.ng/imgui";

// example state (see @thi.ng/atom)
const STATE = new History(new Atom({ foo: true }));

//...IMGUI & layout initialization omitted here

// get atom snapshot
const curr = STATE.deref();

// toggle component will only return result if user clicked it
let res = toggle({
    gui,
    layout,
    id: "foo",
    value: curr.foo,
    label: curr.foo ? "ON" : "OFF"
});

// conditional immutable update (w/ automatic undo snapshot)
if (res !== undefined) STATE.resetIn("foo", res);

Layout support

Most component functions exist in two versions: Using a @thi.ng/layout-compatible grid layout manager or not (e.g. dial vs. dialRaw). The latter versions are more "low-level" & verbose to use, but offer complete layout freedom and are re-used by other component types.

The components in this package not needing a layout manager are only expecting a ILayout or IGridLayout interface, allowing for custom implementations. Furthermore / alternatively, the @thi.ng/layout package also defines a LayoutBox interface, which can be passed instead and too is the type ILayout implementations are expected to produce when allocating space for a component.

The GridLayout class supports infinite nesting and column/row-based space allocation, based on an initial configuration and supporting multiple column/row spans.

screenshot

The code producing this structure:

import { gridLayout } from "@thi.ng/layout";

// create a single column layout @ position 10,10 / 200px wide
// the last values are row height and cell spacing
const layout = gridLayout(10, 10, 200, 1, 16, 4);

// get next layout box (1st row)
// usually you don't need to call .next() manually, but merely pass
// the layout instance to a component...
layout.next();
// { x: 10, y: 10, w: 200, h: 16, cw: 200, ch: 16, gap: 4 }

// 2nd row
layout.next();
// { x: 10, y: 30, w: 200, h: 16, cw: 200, ch: 16, gap: 4 }

// create nested 2-column layout (3rd row)
const twoCols = layout.nest(2);

twoCols.next();
// { x: 10, y: 50, w: 98, h: 16, cw: 98, ch: 16, gap: 4 }

twoCols.next();
// { x: 112, y: 50, w: 98, h: 16, cw: 98, ch: 16, gap: 4 }

// now nest 3-columns in the 1st column of twoCols
// (i.e. now each column is 1/6th of the main layout's width)
const inner = twoCols.nest(3);

// allocate with col/rowspan, here 1 column x 4 rows
inner.next([1, 4])
// { x: 10, y: 70, w: 30, h: 76, cw: 30, ch: 16, gap: 4 }
inner.next([1, 4])
// { x: 44, y: 70, w: 30, h: 76, cw: 30, ch: 16, gap: 4 }
inner.next([1, 4])
// { x: 78, y: 70, w: 30, h: 76, cw: 30, ch: 16, gap: 4 }

// back to twoCols (2nd column)
twoCols.next([1, 2]);
// { x: 112, y: 70, w: 98, h: 36, cw: 98, ch: 16, gap: 4 }

Key controls

The entire UI is fully keyboard controllable, built-in behaviors:

| Keys | Scope | Description | |-----------------------------|------------------|-------------------------------| | Tab / Shift+Tab | Global | Switch focus | | Enter / Space | Global | Activate focused button | | Up / Down or drag mouse | Slider, Dial, XY | Adjust value | | Shift+Up/Down | Slider, Dial, XY | Adjust value (5x step) | | Left/Right | Radial menu | Navigate menu CW/CCW | | Left/Right | Textfield | Move cursor to prev/next word | | Left/Right | XY | Adjust X value | | Alt+Left/Right | Textfield | Move cursor to prev/next word |

More complex behaviors can be achieved in user land. E.g. in the demo, holding down Alt whilst adjusting a slider or dial group will set all values uniformly...

Current limitations

Some of the most obvious missing features:

  • [ ] variable width font support (currently monospace only)
  • [ ] more granular theme options
  • [ ] theme-aware layouting (font size, padding etc.)
  • [ ] image / texture support (Tex ID abstraction)
  • [ ] windows / element containers
  • [ ] menu / tree components
  • [ ] scrolling / clipping
  • [ ] drag & drop

Status

STABLE - used in production

Search or submit any issues for this package

Related packages

Installation

yarn add @thi.ng/imgui

ESM import:

import * as imgui from "@thi.ng/imgui";

Browser ESM import:

<script type="module" src="https://esm.run/@thi.ng/imgui"></script>

JSDelivr documentation

Package sizes (brotli'd, pre-treeshake): ESM: 7.48 KB

Dependencies

Note: @thi.ng/api is in most cases a type-only import (not used at runtime)

Usage examples

Three projects in this repo's /examples directory are using this package:

| Screenshot | Description | Live demo | Source | |:--------------------------------------------------------------------------------------------------------------------|:-------------------------------------------|:---------------------------------------------------|:----------------------------------------------------------------------------------------| | | Interactive inverse FFT toy synth | Demo | Source | | | Canvas based Immediate Mode GUI components | Demo | Source | | | Minimal IMGUI usage example | Demo | Source |

API

Generated API docs

TODO

Authors

If this project contributes to an academic publication, please cite it as:

@misc{thing-imgui,
  title = "@thi.ng/imgui",
  author = "Karsten Schmidt",
  note = "https://thi.ng/imgui",
  year = 2019
}

License

© 2019 - 2026 Karsten Schmidt // Apache License 2.0