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

punctilio

v1.9.1

Published

Smart typography transformations: curly quotes, em-dashes, en-dashes, and more

Readme

punctilio (n.): precise observance of formalities.

Test Lint Coverage

Pretty good at making your text pretty. The most feature-complete and reliable English typography package. punctilio transforms plain ASCII into typographically correct Unicode, even across HTML element boundaries.

Smart quotes · Em/en dashes · Ellipses · Math symbols · Legal symbols · Arrows · Primes · Fractions · Superscripts · Ligatures · Non-breaking spaces · HTML-aware · Bri’ish localisation support

import { transform } from 'punctilio'

transform('"It\'s a beautiful thing, the destruction of words..." -- 1984')
// → “It’s a beautiful thing, the destruction of words…”—1984
npm install punctilio

Why punctilio?

As far as I can tell, punctilio is the most reliable and feature-complete. I built punctilio for my website. I wrote[^wrote] and sharpened the core regexes sporadically over several months, exhaustively testing edge cases. Eventually, I decided to spin off the functionality into its own package.

[^wrote]: While Claude is the number one contributor to this repository, that’s because Claude helped me port my existing code and added some features. The core regular expressions (e.g. dashes, quotes, multiplication signs) are human-written and were quite delicate. Those numerous commits don’t show in this repo’s history.

I tested punctilio 1.2.9 against smartypants 0.2.2, tipograph 0.7.4, smartquotes 2.3.2, typograf 7.6.0, and retext-smartypants 6.2.0.[^python] These other packages have spotty feature coverage and inconsistent impact on text. For example, smartypants mishandles quotes after em dashes (though quite hard to see in GitHub’s font) and lacks multiplication sign support.

[^python]: The Python libraries I found were closely related to the JavaScript packages. I tested them and found similar scores, so I don’t include separate Python results.

| Input | smartypants | punctilio | |:-----:|:-----------------:|:-------:| | 5x5 | 5x5 (✗) | 5×5 (✓) |

My benchmark.mjs measures how well libraries handle a wide range of scenarios. The benchmark normalizes stylistic differences (e.g. non-breaking vs regular space, British vs American dash spacing) for fair comparison.

| Package | Passed (of 159) | |--------:|:----------------| | punctilio | 154 (97%) | | tipograph | 92 (58%) | | typograf | 74 (47%) | | smartquotes | 72 (45%) | | smartypants | 68 (43%) | | retext-smartypants | 65 (41%) |

| Feature | Example | punctilio | smartypants | tipograph | smartquotes | typograf | |--------:|:-------:|:-------:|:-------:|:-------:|:-------:|:-------:| | Smart quotes | "hello" → “hello” | ✓ | ✓ | ✓ | ✓ | ✓ | | Leading apostrophe | 'Twas → ’Twas | ✓ | ✗ | ✗ | ◐ | ✗ | | Em dash | -- → — | ✓ | ✓ | ✗ | ✗ | ✓ | | En dash (ranges) | 1-5 → 1–5 | ✓ | ✗ | ✓ | ✗ | ✗ | | Minus sign | -5 → −5 | ✓ | ✗ | ✓ | ✗ | ✗ | | Ellipsis | ... → … | ✓ | ✓ | ✓ | ✗ | ✓ | | Multiplication | 5x5 → 5×5 | ✓ | ✗ | ✗ | ✗ | ◐ | | Math symbols | != → ≠ | ✓ | ✗ | ◐ | ✗ | ◐ | | Legal symbols | (c) 2004 → © 2004 | ✓ | ✗ | ◐ | ✗ | ✓ | | Arrows | -> → → | ✓ | ✗ | ◐ | ✗ | ◐ | | Prime marks | 5'10" → 5′10″ | ✓ | ✗ | ✓ | ✓ | ✗ | | Degrees | 20 C → 20 °C | ✓ | ✗ | ✗ | ✗ | ✓ | | Fractions | 1/2 → ½ | ✓ | ✗ | ✗ | ✗ | ✓ | | Superscripts | 2nd → 2ⁿᵈ | ✓ | ✗ | ✗ | ✗ | ✗ | | English localization | American / British | ✓ | ✗ | ✗ | ✗ | ✗ | | Ligatures | ?? → ⁇ | ✓ | ✗ | ✓ | ✗ | ✗ | | Non-English quotes | „Hallo” | ✗ | ✗ | ✓ | ✗ | ◐ | | Non-breaking spaces | Chapter 1 | ✓ | ✗ | ✗ | ✗ | ✓ |

Known limitations of punctilio

| Pattern | Behavior | Notes | |:--------|:---------|:------| | '99 but 5' clearance | 5' not converted to 5′ | Leading apostrophe is indistinguishable from an opening quote without semantic understanding | | «Bonjour» | Not spaced to « Bonjour » | French localization not supported |

Test suite

Setting aside the benchmark, punctilio’s test suite includes 1,100+ tests at 100% branch coverage, including edge cases derived from competitor libraries (smartquotes, retext-smartypants, typograf) and the Standard Ebooks typography manual. I also verify that all transformations are stable when applied multiple times.

Works with HTML DOMs via separation boundaries

Perhaps the most innovative feature of the library is that it properly handles DOMs! (This means it'll also work on Markdown: convert to HTML, transform with punctilio, convert back to Markdown.)

Other typography libraries take one of two approaches, both with drawbacks. 

  1. String-based libraries (like smartypants) transform plain text but are unaware of HTML structure. If you concatenate text from <em>Wait</em>..., transform it into Wait…, and then try to convert back—you've lost track of where the </em> belongs. 
  2. AST-based libraries (like rehype-retext) process each text node individually, preserving structure but losing cross-node information. A quote that opens inside <em>"Wait</em> and closes outside it ..." spans two text node. Processed independently, the library can't tell whether the final " is opening or closing, because it never sees both at once. 

punctilio introduces separation boundaries to get the best of both worlds:

  1. Flatten the parent container's contents to a string, delimiting element boundaries with a private-use Unicode character (U+E000) to avoid unintended matches.
  2. Every regex allows (and preserves) these characters, treating them as boundaries of a “permeable membrane” through which contextual information flows. For example, .U+E000.. still becomes …U+E000.
  3. Rehydrate the HTML AST. For all k, set element k’s text content to the segment starting at separator occurrence k.
import { transform, DEFAULT_SEPARATOR } from 'punctilio'

transform(`"Wait${DEFAULT_SEPARATOR}"`)
// → `“Wait”${DEFAULT_SEPARATOR}`
// The separator doesn’t block the information that this should be an end-quote!

For rehype / unified pipelines, use the built-in plugin which handles the separator logic automatically:

import rehypePunctilio from 'punctilio/rehype'

unified()
  .use(rehypeParse)
  .use(rehypePunctilio)
  .use(rehypeStringify)
  .process('<p><em>"Wait</em>..." -- she said</p>')
// → <p><em>“Wait</em>…”—she said</p>
//  The opening quote inside <em> and the closing quote outside it
//  are both resolved correctly across the element boundary.

For manual DOM walking or custom transforms, use transformElement from punctilio/rehype.

Options

punctilio doesn't enable all transformations by default. Fractions and degrees tend to match too aggressively (perfectly applying the degree transformation requires semantic meaning). Superscript letters and punctuation ligatures have spotty font support. Furthermore, ligatures = true can change the meaning of text by collapsing question and exclamation marks.

transform(text, {
  punctuationStyle: 'american' | 'british' | 'none',  // default: 'american'
  dashStyle: 'american' | 'british' | 'none',         // default: 'american'

  symbols: true,           // ellipsis, math, legal, arrows
  collapseSpaces: true,    // normalize whitespace
  fractions: false,        // 1/2 → ½
  degrees: false,          // 20 C → 20 °C
  superscript: false,      // 1st → 1ˢᵗ
  ligatures: false,        // ??? → ⁇, ?! → ⁈, !? → ⁉, !!! → !
  nbsp: true,              // non-breaking spaces (after honorifics, between numbers and units, etc.)
  checkIdempotency: true,  // verify transform(transform(x)) === transform(x)
})
  • Fully general prime mark conversion (e.g. 5'10"5′10″) requires semantic understanding to distinguish from closing quotes (e.g. "Term 1" should produce closing quotes). punctilio counts quotes to heuristically guess whether the matched number at the end of a quote (if not, it requires a prime mark). Other libraries like tipograph 0.7.4 use simpler patterns that make more mistakes.
  • The american style follows the Chicago Manual of Style:
    • Periods and commas go inside quotation marks (“Hello,” she said.)
    • Unspaced em-dashes between words (word—word)
  • The british style follows Oxford style:
    • Periods and commas go outside quotation marks (“Hello”, she said.)
    • Spaced en-dashes between words (word – word)
  • Setting either style to none skips the entire transform category: punctuationStyle: 'none' preserves straight quotes, apostrophes, and prime marks; dashStyle: 'none' preserves all hyphens, number ranges, date ranges, and minus signs.
  • punctilio is idempotent by design: transform(transform(text)) always equals transform(text). If performance is critical, set checkIdempotency: false to skip the verification pass.