Awesome
Awesome Tiny JS
<div align="center"> <a href="https://github.com/thoughtspile/awesome-tiny-js#readme"> <img src="./awesome-logo.png" width="300" height="207"> </a> </div>Tiny front-end libraries to put your bundle on a diet. Rules:
- Size is under 2 kB-ish, min + gzip, with all dependencies, except where noted.
- For multi-purpose libraries, the size of a useful subset must be under 2 kB-ish.
- Useful client-side. I haven't figured out participation rules for node-only libraries, and I'm not too worried about them.
- Second-level libraries only allowed for React, Vue, Angular, svelte.
- 100+ GitHub stars or 500+ weekly npm installs to focus on tools with some community review.
- No zero-JS (CSS- or type-only) libraries. It's not awesome-css or something.
Contents
- UI Frameworks
- Event Emitters
- State Managers
- Routers and URL Utils
- API Layer
- I18N
- Dates and Time
- Generic Utilities
- Validation
- Unique ID Generation
- Colors
- Touch Gestures
- Text Search
UI Frameworks
UI frameworks (libraries?) provide declarative templates, event bindings, and observable state to update the view. I've been generous and expanded the size limit for this category to 4.5 kB (if you're boring, count them as 2 libraries), but also increased the star limit to 2K.
- preact - React-like API (pre-hooks). Cool ecosystem of similarly tiny tools and components. Highly recommended. <img align="top" height="24" src="./img/preact.svg">
The following libraries are small and cool, but note they're about 500x less popular than preact. Kudos for deconstrucing the very essence of a "framework":
- hyperapp - vDOM framework with pure JS syntax and immutable state, <img align="top" height="24" src="./img/hyperapp.svg">
- redom - Hyperapp-style templates with imperative event listeners and updates, <img align="top" height="24" src="./img/redom.svg">
Now, for the openly experimental UI libraries:
- fre - React-like library with hooks and concurrency, <img align="top" height="24" src="./img/fre.svg">
- van - vDOM-based framework optimized for no-build setups, <img align="top" height="24" src="./img/vanjs-core.svg">
- superfine - Hyperapp with state & effect hooks removed, <img align="top" height="24" src="./img/superfine.svg">
- arrowjs - Tagged templates + reactive data, <img align="top" height="24" src="./img/arrow-jscore.svg">
And if being declarative is not your thing:
- umbrella - jQuery-style DOM manipulation library, <img align="top" height="24" src="./img/umbrellajs.svg">
Event Emitters
Event emitter pattern is fairly easy to implement yourself, but why bother when you have these cool tools? With an arms race to build the smallest one, the limit is 0.5 kB.
- mitt - Plain event emitter that I use on most projects, <img align="top" height="24" src="./img/mitt.svg">
- nanoevents - Nicer unsubscribe API, but no
*
event, <img align="top" height="24" src="./img/nanoevents.svg"> - onfire.js - Also has
.once
method, <img align="top" height="24" src="./img/onfirejs.svg">
State Managers
State managers combine observable state with actions and framework bindings, intended for app-wide state.
- zustand - Simple stores with pleasant actions and selectors. Vanilla <img align="top" height="24" src="./img/zustandvanilla.svg">, React <img align="top" height="24" src="./img/zustand.svg">
- nanostores - Modular store with good tree-shaking support, <img align="top" height="24" src="./img/nanostores.svg"> vanilla, + React <img align="top" height="24" src="./img/nanostoresreact.svg"> extra. Supports all the top frameworks.
- exome - Atomic stores with lots of framework connectors, <img align="top" height="24" src="./img/exome.svg"> + React <img align="top" height="24" src="./img/exomereact.svg"> extra. Supports all the top frameworks.
- storeon - Minimal redux-styled store with lots of framework connectors, <img align="top" height="24" src="./img/storeon.svg">. React extra <img align="top" height="24" src="./img/storeonreact.svg"> + Vue, Svelte, Angular.
- unistore - Centralized store with actions, <img align="top" height="24" src="./img/unistore.svg"> + React <img align="top" height="24" src="./img/unistorereact.svg">
- teaful - Store with useState-like API, <img align="top" height="24" src="./img/teaful.svg">, including React / preact connector.
Signals
A signal-styled state manager provides observable values (aka signals), derived values and effects.
- @preact/signals - The OG signals from preact <img align="top" height="24" src="./img/preactsignals-core.svg"> core, <img align="top" height="24" src="./img/preactsignals-react.svg"> with react integration.
- usignal - A smaller signal implementation, <img align="top" height="24" src="./img/usignal.svg">
- hyperactiv - 4 functions to make objects observable and listen to changes, <img align="top" height="24" src="./img/hyperactiv.svg">
- flimsy - Signals from Solid (it almost fit into UI frameworks category itself). Author warning: it's probably buggy. <img align="top" height="24" src="./img/flimsy.svg">
Honorable mention: oby could make it if it had tree-shaking, but otherwise is around 7 kB.
Reactive Programming
Another well-known state management approach is reactive programmning — operating on event streams, applying filters and transforms to end up with an observable value. Think RxJS, but tiny:
- flyd - Rx-styled event streams, <img align="top" height="24" src="./img/flyd.svg">
- callbag-basics - Rx-style event streams, <img align="top" height="24" src="./img/callbag-basics.svg">
Routers and URL Utils
Do stuff on URL / history changes, with path matching and parsing:
- wouter - Declarative router for React / preact, <img align="top" height="24" src="./img/wouter.svg">, also available as a standalone hook: <img align="top" height="24" src="./img/wouteruse-browser-location.svg">
- @nanostores/router - Routes as a nanostores store (framework-agnostic), <img align="top" height="24" src="./img/nanostoresrouter.svg">
- navaid - History-based observable router, <img align="top" height="24" src="./img/navaid.svg">
Just want to parse or match URL paths without observing them? Here you go:
- matchit - Route parser and matcher in <img align="top" height="24" src="./img/matchit.svg">
- regexparam - Convert path to regexp in <img align="top" height="24" src="./img/regexparam.svg">
- qss - Parse querystrings in <img align="top" height="24" src="./img/qss.svg">. Not sure you need it, URL API support is good.
API Layer
fetch
API has some boilerplate associated with it: serialize & parse data, reject on non-200 response, etc. These tiny packages handle it for you:
- redaxios - Drop-in axios replacement for modern browsers, <img align="top" height="24" src="./img/redaxios.svg">
- wretch - Chainable API with error processing and lots of extra plugins, <img align="top" height="24" src="./img/wretch.svg">
- gretchen - Chainable API with type-safe errors, <img align="top" height="24" src="./img/gretchen.svg">
If for some reason you still need a fetch polyfill, try this one:
- unfetch - Loose fetch polyfill, <img align="top" height="24" src="./img/unfetch.svg">
I18N
A map of strings might seem enough to translate an app, but these tools also handle interpolation and some extra goodies:
- @nanostores/i18n - Detect locale, load dictionaries, format dates / numbers, <img align="top" height="24" src="./img/nanostoresin.svg"> including nanostores.
- eo-locale - Interpolation and dates / numbers, <img align="top" height="24" src="./img/eo-localecore.svg">, or <img align="top" height="24" src="./img/eo-localereact.svg"> with react bindings.
- rosetta - Bare-bones template strings (
{{hello}}, {{username}}
) and custom functions for everyting else, <img align="top" height="24" src="./img/rosetta.svg"> - lingui - Small core with template strings, <img align="top" height="24" src="./img/linguicore.svg">
Dates and Time
Date and time manipulation in pure JS is verbose. Luckily, two of the top date libraries have sensible size:
- date-fns - Not tiny as a whole, but most functions are under 1 kB each (format and parse are quite heavy).
- dayjs - Almost moment.js-compatible API, covers most use cases, <img align="top" height="24" src="./img/dayjsesm.svg">
And some more packages that only do formatting:
- tinytime - Simple date / time formatter:
{h}:{mm} -> 9:33
, <img align="top" height="24" src="./img/tinytime.svg"> - tinydate - Date / time formatter, only supports padded numeric output (
September -> 09
), <img align="top" height="24" src="./img/tinydate.svg"> - time-stamp - More of the same, <img align="top" height="24" src="./img/time-stamp.svg">
- ms - Parse & format ms durations, e.g.
"1m" <-> 60000
, <img align="top" height="24" src="./img/ms.svg"> - timeago.js - Format dates into stuff like X minutes ago or in X hours, <img align="top" height="24" src="./img/timeagojs.svg">
- fromnow - More of the same, <img align="top" height="24" src="./img/fromnow.svg">
Note that the built-in Intl.DateTimeFormat
has decent support.
Generic Utilities
Something you'd find in lodash or ramda, but smaller. Most are pretty similar and very small, with minor differences in package structure (single / package-per-helper) and tree shaking vs direct helper import.
- remeda - 90 tree-shakable helpers (list).
- rambda - 187 tree-shakable helpers (list).
- just - 82 helpers in separate packages (list).
- @fxts/core - 96 tree-shakable helpers. Lazy evaluation support.
Honorable mention: underscore, contains many sub-1 kB helpers. It does not tree-shake as well as the libraries above due to codebase structure.
Note: lodash itself is not tree-shakable, but has made many attempts at modulaity with lodash.method
packages, imports from lodash/method
, and lodash-es
, none of which work well in practice.
Also note that much of the original lodash functionality comes built-in with modern ES. Prefer native versions over libraries as your browser target allows.
Validation
To check if an object matches an expected schema, you'd often use zod, yup, joi or ajv. But 90% of the time you can get what you need in under 2 kB. Note: I compare a base validation subset (core + object / array + string / number / boolean) under tree-shaking to avoid punishing libs that have more features.
- v8n - zod-style API with fine-grained checks:
v8n().string().minLength(5).first("H").last("o")
. No tree shaking, <img align="top" height="24" src="./img/vn.svg"> - banditypes - The smallest validation library: <img align="top" height="24" src="./img/banditypes.svg">
- superstruct - The most popular modular validation library with good tree-shaking, <img align="top" height="24" src="./img/superstruct.svg">
- valibot - Another modular validation library, <img align="top" height="24" src="./img/valibot.svg">
- deep-waters - Composable functional validators, <img align="top" height="24" src="./img/deep-waterscompose-deep-watershasShape-deep-watersarrayOf-deep-watersisString-deep-watersisNumber-deep-watersisBoolean.svg">.
Unique ID Generation
Unique ID generation does not take a lot of code, but it's not someting I'd want to write myself. Limit is 500 bytes. Also note that the native crypto.randomUUID
has OK support.
- @lukeed/uuid - Real UUIDs, <img align="top" height="24" src="./img/lukeeduuid.svg">
- nanoid - Random IDs with larger alphabet, <img align="top" height="24" src="./img/nanoid.svg">
- uid - More of the same, <img align="top" height="24" src="./img/uid.svg">
- hexoid - Hexadecimal IDs, <img align="top" height="24" src="./img/hexoid.svg">
Colors
Color manipulation is rare in pure UI development, but very helpful for data visualization, and uses freaky math. Don't fry your brain, take these:
- colord - Manipulate colors and convert between spaces, <img align="top" height="24" src="./img/colord.svg">. Extra features come as plugins, 150b to 1.5 kB each.
- colr - More of the same, <img align="top" height="24" src="./img/colr.svg" >
- polychrome - More of the same, <img align="top" height="24" src="./img/polychrome.svg">
- randomcolor - Attractive random colors with configuration. <img align="top" height="24" src="./img/randomcolor.svg">
Touch Gestures
Touch gestures like swipe, drag, pinch or doubletap are a staple of mobile UX, but recognizing a series of touchmove / pointer events as a gesture is tricky, and testing is painful. Here are two libraries that do the heavy lifting for you:
- alloyfinger - Pan, swipe, tap, doubletap, longpress, and pinch / rotate. My personal favorite. <img align="top" height="24" src="./img/alloyfinger.svg">.
- tinygesture - Configurable pan, swipe, tap, doubletap, longpress. <img align="top" height="24" src="./img/tinygesture.svg">.
Even if you want to detect gestures yourself, juggling mouse, touch and pointer events is hard enough, and browser inconsistencies don't help. Here are two more libraries to assist with that:
- pointer-tracker - Unified interface for mouse, touch and pointer events, <img align="top" height="24" src="./img/pointer-tracker.svg">
- detect-it - Detect present and primary input method (touch / mouse) and supported events, <img align="top" height="24" src="./img/detect-it.svg">
Honorable mentions: any-touch attempts a modular approach to gesture detection, but the core is around 2 kB without any gesture recognizers. rc-gesture, used in ant design system, could be the only react component on the list, but babel-runtime / corejs polyfills hard-wired into the build push the ~2.5 kB size to over 10 kB.
Text Search
Text search is important for client-side filtering and autosuggests. Naive option.includes(search)
has no sensible order on the results, and ignoring word boundaries gives unexpected matches like spa -> newSPAper. First, here are some libraries that prioritize word matches:
- js-search - Feature-rich and customizable: multi-field indices, stop words, custom stemmers and tokenizers. <img align="top" height="24" src="./img/js-search.svg">
- ndx - Similar to js-search, differs in ranking and is less strict for multi-word queries (compare). Supports field weights. <img align="top" height="24" src="./img/ndx-ndxquery.svg">
- wade - Also similar, (compare) <img align="top" height="24" src="./img/wade.svg">
- libsearch - Index-free search (slower, but easier to use) with sane ordering <img align="top" height="24" src="./img/libsearch.svg">
One way to find sensible inexact matches is stemming — converting words to a root form. Walked will match walking, etc. Here are a few Porter stemmers for English language:
- stemmer - <img align="top" height="24" src="./img/stemmer.svg">
- porter-stemmer - <img align="top" height="24" src="./img/porter-stemmer.svg">
For non-English words, I only have honorable mentions: snowball-js is 17 kB with 15 languages, lunr-languages supports 30 languages but only works with lunr, the most promising one is natural but it depends on Node.js.
Fuzzy search
Fuzzy search is another take on inexact matching — the words can be modified. First, we have libraries that only allow insertion: spacecat -> SPACECrAfT. Not perfect for general-purpose text search, but great for filename, command, or URL lookups.
- fuzzy - Index-free, can highlight matches. <img align="top" height="24" src="./img/fuzzy.svg">
- fuzzy-search - With stateful index. <img align="top" height="24" src="./img/fuzzy-search.svg">
- fzy.js - Matches one string at a time, tree-shakeable scores and match highlighting. <img align="top" height="24" src="./img/fzyjs.svg"> total, or ~150 bytes for
hasMatch
only. - fuzzysearch - One string at a time, does not compute score / rank. <img align="top" height="24" src="./img/fuzzysearch.svg">
- liquidmetal - Quicksilver algorithm, prioritizes matches at start of word for command abbreviations (e.g.
gp
->git push
). One string at a time. <img align="top" height="24" src="./img/liquidmetal.svg"> - quick-score - Another quicksilver-based lib, tweaked for long strings. Built-in list filtering and sorting, <img align="top" height="24" src="./img/quick-score.svg"> or 1.2 kB for single-string scoring.
Finally, one library is specifically built for spellchecking:
- fuzzyset - Find misspellings, e.g. missipissi -> Missisipi, <img align="top" height="24" src="./img/fuzzyset.svg"> Commercial usage costs $42.
Contributing
Suggestions welcome! See contributing.md, or drop an issue.
Footnotes
See WIP for possibly awesome libraries I have found, but not yet analyzed deeply, and incubate for awesome libraries that don't meet popularity criteria yet.
Collected and reviewed by Vladimir Klepov in 2023.