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

stylelint-plugin-rhythmguard

v1.4.2

Published

Stylelint plugin for spacing scale, token enforcement, and Tailwind class-string governance

Readme

stylelint-plugin-rhythmguard

High-precision spacing governance for CSS and design systems.

CI npm version npm downloads License: MIT Node

stylelint-plugin-rhythmguard enforces scale and token discipline across spacing, radius, typography, size, and translate motion offsets.

Demo

I built Rhythmguard after 20 years of watching teams ignore spacing scales and ship arbitrary pixel values everywhere.

It is built for teams that want:

  • zero random spacing values in production CSS
  • consistent numeric scales for radius, typography, and sizing primitives
  • token-first spacing workflows
  • predictable autofix behavior for large migrations
  • consistent layout rhythm across web surfaces

Rule Matrix

| Rule | What it does | Autofix | | --- | --- | --- | | rhythmguard/use-scale | Enforces spacing values must be on your configured scale | Yes, nearest safe value | | rhythmguard/prefer-token | Enforces token usage over raw spacing literals | Yes, with tokenMap | | rhythmguard/no-offscale-transform | Enforces scale-aligned translate* motion offsets | Yes, nearest safe value |

Installation

npm install --save-dev stylelint stylelint-plugin-rhythmguard

Drop-In for Existing Projects (Recommended)

If your project already uses Stylelint, you only need one command and one config block:

npm install --save-dev stylelint-plugin-rhythmguard
{
  "extends": ["stylelint-plugin-rhythmguard/configs/recommended"]
}

Quick Start

Recommended config

{
  "extends": ["stylelint-plugin-rhythmguard/configs/recommended"]
}

Strict config

{
  "extends": ["stylelint-plugin-rhythmguard/configs/strict"]
}

strict intentionally delegates transform translation enforcement to rhythmguard/no-offscale-transform to reduce overlapping warnings from use-scale.

Tailwind config

{
  "extends": ["stylelint-plugin-rhythmguard/configs/tailwind"]
}

Expanded config

{
  "extends": ["stylelint-plugin-rhythmguard/configs/expanded"]
}

expanded enables scale enforcement for spacing + radius + typography + size property groups.

Logical config

{
  "extends": ["stylelint-plugin-rhythmguard/configs/logical"]
}

logical composes Rhythmguard strict mode with stylelint-plugin-logical-css recommended rules.

Migration config

{
  "extends": ["stylelint-plugin-rhythmguard/configs/migration"]
}

migration keeps on-scale numeric values temporarily while auto-building token mappings from CSS custom properties and optional Tailwind spacing config.

Stable shared config entry points:

  • stylelint-plugin-rhythmguard/configs/recommended
  • stylelint-plugin-rhythmguard/configs/strict
  • stylelint-plugin-rhythmguard/configs/tailwind
  • stylelint-plugin-rhythmguard/configs/expanded
  • stylelint-plugin-rhythmguard/configs/logical
  • stylelint-plugin-rhythmguard/configs/migration

Comparison and Migration Recipes

Full custom setup

{
  "plugins": ["stylelint-plugin-rhythmguard"],
  "rules": {
    "rhythmguard/use-scale": [
      true,
      {
        "preset": "rhythmic-4",
        "propertyGroups": ["spacing", "radius"],
        "propertyScales": {
          "font-size": [12, 14, 16, 20, 24]
        },
        "units": ["px", "rem", "em"],
        "unitStrategy": "convert",
        "baseFontSize": 16,
        "tokenPattern": "^--space-",
        "tokenFunctions": ["var", "theme", "token"],
        "allowNegative": true,
        "allowPercentages": true,
        "fixToScale": true,
        "enforceInsideMathFunctions": true,
        "mathFunctionArguments": {
          "clamp": [1, 3]
        }
      }
    ],
    "rhythmguard/prefer-token": [
      true,
      {
        "tokenPattern": "^--space-",
        "allowNumericScale": false,
        "tokenMapFromCssCustomProperties": true,
        "tokenMapFromTailwindSpacing": true,
        "tailwindConfigPath": "./tailwind.config.mjs",
        "tokenMap": {
          "4px": "var(--space-1)",
          "8px": "var(--space-2)",
          "12px": "var(--space-3)",
          "16px": "var(--space-4)"
        }
      }
    ],
    "rhythmguard/no-offscale-transform": [
      true,
      {
        "scale": [0, 4, 8, 12, 16, 24, 32]
      }
    ]
  }
}

Presets and custom scales

Preset-based setup:

{
  "rules": {
    "rhythmguard/use-scale": [true, { "preset": "fibonacci" }]
  }
}

Custom scale setup:

{
  "rules": {
    "rhythmguard/use-scale": [true, { "customScale": [0, 6, 12, 18, 24, 36, 48] }]
  }
}

Scale resolution precedence:

  1. customScale (highest priority)
  2. scale
  3. preset
  4. default rhythmic-4 scale

Option Validation

Rhythmguard validates secondaryOptions for each rule before linting declarations.

  • Unknown option names fail fast with Stylelint invalid option warnings.
  • Invalid option shapes fail fast (for example string vs array mismatches).
  • properties string entries are validated against supported scale-targetable CSS property names.
  • propertyGroups values are validated against built-in groups: spacing, radius, typography, and size.
  • Math function argument maps are validated per function (calc, clamp, min, max) and positive 1-based argument indexes.

Example typo that now fails immediately:

{
  "rules": {
    "rhythmguard/use-scale": [true, { "sevverity": "warning" }]
  }
}

Built-in Scale Presets

| Preset | Pattern | Scale | | --- | --- | --- | | rhythmic-4 | 4pt rhythm | [0,4,8,12,16,24,32,40,48,64] | | rhythmic-8 | 8pt rhythm | [0,8,16,24,32,40,48,64,80,96] | | product-material-8dp | Material 8dp baseline + 4dp increments | [0,4,8,12,16,24,32,40,48,56,64,72,80] | | product-atlassian-8px | Atlassian-like product spacing progression | [0,2,4,6,8,12,16,20,24,32,40,48,64,80] | | product-carbon-2x | Carbon 2x spacing progression | [0,2,4,8,12,16,24,32,40,48,64,80] | | editorial-baseline-4 | editorial baseline rhythm at 4-unit cadence | [0,4,8,12,16,20,24,28,32,40,48,56,64] | | editorial-baseline-6 | editorial baseline rhythm at 6-unit cadence | [0,6,12,18,24,30,36,48,60,72] | | compact | dense UI spacing | [0,2,4,6,8,12,16,20,24,32] | | fibonacci | Fibonacci progression | [0,2,3,5,8,13,21,34,55,89] | | powers-of-two | geometric doubling | [0,2,4,8,16,32,64,128] | | golden-ratio | ratio 1.618 | generated modular sequence | | modular-major-second | ratio 1.125 | generated modular sequence | | modular-minor-third | ratio 1.2 | generated modular sequence | | modular-major-third | ratio 1.25 | generated modular sequence | | modular-augmented-fourth | ratio 1.414 | generated modular sequence | | modular-perfect-fourth | ratio 1.333 | generated modular sequence | | modular-perfect-fifth | ratio 1.5 | generated modular sequence |

Aliases:

  • 4ptrhythmic-4
  • 8ptrhythmic-8
  • materialproduct-material-8dp
  • atlassian-8product-atlassian-8px
  • carbonproduct-carbon-2x
  • baseline-4editorial-baseline-4
  • baseline-6editorial-baseline-6
  • goldengolden-ratio
  • major-secondmodular-major-second
  • minor-thirdmodular-minor-third
  • major-thirdmodular-major-third
  • augmented-fourthmodular-augmented-fourth
  • perfect-fourthmodular-perfect-fourth
  • perfect-fifthmodular-perfect-fifth

Preset Rationale

  • Product presets are based on widely-used design-system spacing frameworks.
  • Editorial presets model baseline-grid cadence used in long-form typography and column layouts.
  • Theory presets expose mathematically-derived modular scales from design theory and typographic proportion systems.
  • Full research notes and sources are documented in docs/SCALE_RESEARCH.md.

Community Scale Registry

Rhythmguard supports community-contributed scale presets from scales/community/*.json.

Current community scales

| Preset | Base | Pattern | Contributor | | --- | --- | --- | --- | | product-decimal-10 | 10 | Decimal-friendly dashboard/product cadence | Petri Lahdelma |

Contribute a scale

  1. Scaffold a new scale file:
npm run scales:add -- --name my-team-scale --base 8 --steps 0,4,8,12,16,24,32
  1. Validate:
npm run scales:validate
  1. Open a PR with your scale JSON.

Full specification and policy: docs/COMMUNITY_SCALES.md.

If your scale is private or very niche, keep it in your project config with customScale instead of contributing it to the shared registry.

Rule Details

rhythmguard/use-scale

Enforces spacing literals to stay on a configured numeric scale.

Checks:

  • margin*, padding*
  • gap, row-gap, column-gap
  • inset*, scroll-margin*, scroll-padding*
  • translate, translate-x, translate-y, translate-z
  • transform translation functions (translate, translateX, translateY, translateZ, translate3d)
  • optional property groups:
    • radius (border-radius*, corner radii, outline-offset)
    • typography (font-size, line-height, letter-spacing, word-spacing)
    • size (width, height, min/max size, logical inline-size/block-size)

Example:

/* ❌ Off-scale */
.card {
  margin: 13px;
  transform: translateY(18px);
}

/* ✅ On-scale */
.card {
  margin: 12px;
  transform: translateY(16px);
}

Options:

| Option | Type | Default | Description | | --- | --- | --- | --- | | preset | string | rhythmic-4 | Selects a built-in spacing scale | | customScale | Array<number|string> | undefined | Highest-priority custom scale override | | scale | Array<number|string> | [0,4,8,12,16,24,32,40,48,64] | Allowed spacing values | | units | string[] | ['px','rem','em'] | Units considered for scale enforcement | | unitStrategy | 'convert' \| 'exact' | 'convert' | convert: compare via px conversion (px/rem/em). exact: compare against same-unit scale values (for example vw, cqi) | | baseFontSize | number | 16 | Used for rem/em conversion | | tokenPattern | string | ^--space- | Regex for accepted token variable names | | tokenFunctions | string[] | ['var','theme','token'] | Functions treated as tokenized values | | allowNegative | boolean | true | Allows negative scale values | | allowPercentages | boolean | true | Allows % values without scale checks | | fixToScale | boolean | true | Enables nearest-value autofix | | enforceInsideMathFunctions | boolean | false | Lints calc()/clamp()/min()/max() internals | | mathFunctionArguments | Record<mathFn, number[]> | {} | Restricts linting to specific 1-based argument indexes per math function | | ignoreMathFunctionArguments | Record<mathFn, number[]> | {} | Excludes specific 1-based argument indexes per math function | | propertyGroups | Array<'spacing' \| 'radius' \| 'typography' \| 'size'> | ['spacing'] | Selects built-in property groups when properties is not provided | | properties | Array<string|RegExp> | built-in spacing patterns | Override targeted property set; string values may be supported property names or regex-like strings (/pattern/flags) | | propertyScales | Record<propertyOrRegex, scaleOrPreset> | {} | Per-property scale overrides (supports exact names or /regex/flags keys; stateful g/y flags are normalized for deterministic matching) |

rhythmguard/prefer-token

Enforces token usage for spacing declarations. This is ideal once your token system is stable.

Example:

/* ❌ Raw literals */
.stack {
  gap: 12px;
  padding: 16px;
}

/* ✅ Tokenized */
.stack {
  gap: var(--space-3);
  padding: var(--space-4);
}

Options:

| Option | Type | Default | Description | | --- | --- | --- | --- | | tokenPattern | string | ^--space- | Regex for accepted token variable names | | tokenFunctions | string[] | ['var','theme','token'] | Functions treated as tokenized values | | allowNumericScale | boolean | false | Temporary migration mode to permit on-scale literals | | preset | string | rhythmic-4 | Selects a built-in scale used in migration mode | | customScale | Array<number|string> | undefined | Highest-priority custom scale override | | scale | Array<number|string> | [0,4,8,12,16,24,32,40,48,64] | Used when allowNumericScale is enabled | | baseFontSize | number | 16 | Used for scale checks with rem/em | | unitStrategy | 'convert' \| 'exact' | 'convert' | Matching strategy when allowNumericScale is enabled | | units | string[] | ['px','rem','em'] | Units considered for numeric scale checks | | enforceInsideMathFunctions | boolean | false | Lints calc()/clamp()/min()/max() internals | | mathFunctionArguments | Record<mathFn, number[]> | {} | Restricts linting to specific 1-based argument indexes per math function | | ignoreMathFunctionArguments | Record<mathFn, number[]> | {} | Excludes specific 1-based argument indexes per math function | | tokenMap | Record<string,string> | {} | Enables autofix from raw value to token | | tokenMapFile | string | null | JSON file path to merge additional token mappings | | tokenMapFromCssCustomProperties | boolean | false | Auto-builds mappings from matching custom property declarations in the same stylesheet | | tokenMapFromTailwindSpacing | boolean | false | Auto-builds mappings from theme.spacing and theme.extend.spacing in Tailwind config | | tailwindConfigPath | string | null | Path to Tailwind config used by tokenMapFromTailwindSpacing (.js, .cjs, .mjs) | | ignoreValues | string[] | CSS global keywords + auto | Skips keyword literals | | propertyGroups | Array<'spacing' \| 'radius' \| 'typography' \| 'size'> | ['spacing'] | Selects built-in property groups when properties is not provided | | properties | Array<string|RegExp> | built-in spacing patterns | Override targeted property set; string values may be supported property names or regex-like strings (/pattern/flags) | | propertyScales | Record<propertyOrRegex, scaleOrPreset> | {} | Per-property scale overrides for numeric migration mode (stateful g/y flags are normalized for deterministic matching) |

rhythmguard/no-offscale-transform

Specialized guardrail for motion spacing consistency in translation transforms.

Example:

/* ❌ Off-scale motion */
.toast {
  transform: translateY(18px) scale(1);
}

/* ✅ Motion on spacing scale */
.toast {
  transform: translateY(16px) scale(1);
}

Options:

rhythmguard/no-offscale-transform accepts the same scale options as rhythmguard/use-scale (including unitStrategy, math argument targeting, and deterministic autofix), but only for transform translation properties. Its secondary options are also validated for unknown keys and invalid value shapes.

Tailwind CSS Integration

Rhythmguard works well in Tailwind projects, but it enforces what Stylelint can parse: CSS declarations.

What Rhythmguard covers in Tailwind projects

  • custom CSS in globals.css, components.css, utilities.css
  • CSS Modules (for example *.module.css)
  • declarations inside @layer blocks

What Rhythmguard does not cover

  • Tailwind class strings in templates/JSX/TSX, for example:
    • class="p-4 gap-2"
    • class="p-[13px] translate-y-[18px]"

Those are not Stylelint declaration nodes, so they are outside Stylelint rule scope.

Companion ESLint layer for class strings

Rhythmguard now ships an ESLint companion export for class-string governance:

// eslint.config.js (flat config)
import rhythmguard from 'stylelint-plugin-rhythmguard/eslint';

export default [
  {
    plugins: {
      'rhythmguard-tailwind': rhythmguard,
    },
    rules: {
      'rhythmguard-tailwind/tailwind-class-use-scale': ['error', { scale: [0, 4, 8, 12, 16, 24, 32] }],
    },
  },
];

This rule targets arbitrary spacing utilities such as p-[13px], gap-[18px], translate-x-[10px], and autofixes to the nearest configured scale value.

Recommended stack for full Tailwind enforcement

Use both layers:

  1. Stylelint + Rhythmguard for CSS declaration governance.
  2. Tailwind-aware class-string linting/formatting for template utility usage.

Suggested setup:

{
  "extends": ["stylelint-plugin-rhythmguard/configs/tailwind"]
}

Then pair with:

  • stylelint-plugin-rhythmguard/eslint for arbitrary spacing class-string scale enforcement.
  • eslint-plugin-tailwindcss for broader class-string linting and conventions.
  • prettier-plugin-tailwindcss for deterministic class ordering.

Detailed setup reference: docs/TAILWIND.md.

Tailwind token function support

By default, tokenFunctions includes theme, so values like theme(spacing.4) are treated as tokenized values.

This keeps CSS declaration enforcement and template class-string enforcement separated but coordinated.

Programmatic Presets

const rhythmguard = require('stylelint-plugin-rhythmguard');

console.log(rhythmguard.presets.listScalePresetNames());
console.log(rhythmguard.presets.listCommunityScalePresetNames());
console.log(rhythmguard.presets.getCommunityScaleMetadata('product-decimal-10'));
console.log(rhythmguard.presets.scales['rhythmic-4']);
console.log(Object.keys(rhythmguard.eslint.rules));

Autofix Philosophy

Rhythmguard only applies deterministic fixes:

  • nearest scale value for numeric off-scale literals
  • explicit tokenMap replacements for token migration

It will not guess token mappings without your map.

Compatibility

  • Stylelint: ^16.0.0 || ^17.0.0
  • Node.js: >=18.18.0
  • Module format: dual require + import entry points (CommonJS + ESM wrappers)
  • Note: Stylelint 16.0.0 has known autofix/API behavior differences; CI enforces floor compatibility and runs non-blocking full-suite observability on the floor version.

Development

npm install
npm run lint
npm test
npm run test:coverage

Performance Benchmarking

Compare runtime against stylelint-scales on a deterministic spacing corpus:

npm run bench:perf

Benchmark with autofix enabled:

npm run bench:perf:fix

Detailed methodology and custom args are documented in docs/BENCHMARKING.md.

Article

Used by and Community Examples

Public codebases currently used for production migration examples:

Want your team listed here?

  1. Open an issue with used-by in the title.
  2. Include one before/after diff and your Rhythmguard config.
  3. Add migration notes (false positives, rules enabled, rollout phase).

Release Workflow

  1. Create a GitHub release.
  2. release.yml runs the Node/Stylelint matrix validation.
  3. A tarball smoke test validates package exports and install behavior.
  4. If NPM_TOKEN is configured in repository secrets, the package is published to npm with provenance (npm publish --provenance).
  5. If NPM_TOKEN is not configured, publish is skipped with an explicit workflow notice.
  6. post-publish-smoke.yml verifies the published npm version can be installed and run in a clean project (and skips cleanly if the version is not on npm).

Support and Bug Reports

License

MIT. See LICENSE.