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

@magic/css

v0.7.56

Published

css in js

Downloads

1,769

Readme

@magic/css

parse/stringify/write css in js

html docs

NPM version Linux Build Status Windows Build Status Coverage Status Greenkeeper badge Known Vulnerabilities

installation:
  npm install --save-exact @magic/css

usage:

cli:

@magic/css includes a cli script that can handle most usecases the internal javascript api allows.

to use this cli from any directory, npm install -g @magic.css is a useful shortcut. after the global install, you can just call mcss from anywhere in your terminal.

after installation, add: "mcss": "mcss" to your package.json "scripts" section and then npm run mcss to see the help output below.

@magic/css
commands:
stringify - convert css in js to css
parse     - convert css in js to an array of key value pairs
full      - get a full result object.

flags:
--minified - output minified css - alias: ["--m", "-m"]
--help     - alias: ["-h"]
--out      - directory to write output files to. omit to print to stdout - alias: ["--o", "-o"]
--in       - directory with source files, needs index.js to exist - alias: ["--i", "-i"]

examples:
mcss parse --in ./styles --out ./css
mcss stringify --in ./styles --out ./css
mcss full --in ./styles --out ./css

library:

init
import css from '@magic/css'
all in one
import css from '@magic/css'

const style = {
  body: {
    color: 'green',
  },
  '.class': {
    color: 'orange',
  },
  '#id': {
    color: 'purple',
  },
}

css(style)

// returns
Object {
  // nicely formatted css string
  css: 'body {\n  color: green;\n}\n.class {\n  color: orange;\n}\n#id{\n  color: purple;\n}\n',
  // minimal whitespace
  minified: 'body{color:green}.class{color:orange}#id{color:purple}',
  // array of used classes if any
  classes: ['.class'],
  // array of used ids if any
  ids: ['#id'],
  // array of used selectors
  selectors: ['body', '.class', '.id'],
  // ast of this css object
  parsed: [
    ['body', { color: 'green' }],
    ['.class': { color: 'orange' }],
    ['#id': { color: 'purple' }],
  ],
}
styles
const style = {
  '.className': {
    color: 'green',
  },
}
await css.stringify(style)

// .className { color:green; }
hover/active etc
const style = {
  div: {
    color: 'red',
    '&:hover': {
      color: 'green',
    },
  },
}

await css.stringify(style)

// div { color: red; }
// div:hover { color: green; }
nesting / suffix

to suffix some of the selectors, add a & to any of them except the first one

const style = {
  div: {
    '.class1, &.class2': {
      color: 'red',
    },
  },
}

await css.stringify(style)

// div .class1, div.class2 { color: red; }

if you add the & at the start of the string, all of the selectors will have it applied

const style = {
  div: {
    '&.class1, .class2': {
      color: 'orange',
    },
  },
}

await css.stringify(style)

// div.class1, div.class2 { color: orange; }
prefix without space

to prefix the parent of the selector using the child selectors, add a & at the end of any selectors but the last one.

const style = {
  '.class': {
    'p&, :hover': {
      color: 'orange',
    },
  },
}

await css.stringify(style)

// p.class, .class:hover { color: orange; }
prefix all without space

if the & is at the end of a selector, the & will be applied to each of them.

const style = {
  '.class': {
    'div, p&': {
      color: 'orange';
    },
  },
}

await css.stringify(style)

// div.class, p.class { color: orange; }
prefix with space

to prefix the parent with a space, use a double && instead of a single &

const style = {
  '#id': {
    '.class2&&, .class3': {
      color: 'orange';
    },
  },
}

await css.stringify(style)

// .class2 #id, #id.class3 { color: orange; }
prefix all with space

if the && is at the end of a selector, the && will be applied to each of them.

const style = {
  '.class': {
    'div, p&&': {
      color: 'orange';
    },
  },
}

await css.stringify(style)

// div .class, p .class { color: orange; }
parse
const style = {
  '.className': {
    '#id': {
      color: 'orange',
    },
  },
}

css.parse(style)

// ast
stringify
const style = {
  '.className': {
    '#id': {
      color: 'white',
    },
  },
}

await css.stringify(
  style,
) // minified string
`.className #id{color:white;}`
write to filesystem
const style = {
  '.className': {
    '#id': {
      color: opts.textColor,
    },
  },
}

// writes styles to ./out.css
await css.write(style)

// writes styles to ./outfile.css
await css.write(style, { OUTFILE: './outfile.css' })
media queries

Mediaqueries can use the vars.widths object to determine appropriate breakpoint sizes.

Default widths:

opts.widths = {
  tablet: '500px',
  laptop: '900px',
  desktop: '1200px',
  agency: '1600px',
}
const style = {
  [`@media screen and (min-width: ${vars.widths?.tablet || '500px'})`]: {
    '#id': {
      color: opts.textColor,
    },
  },
}

await css.stringify(
  style,
) // css string
`
@media screen and (min-width: 500px) {
  #id {
    color: green;
  }
}
`
webfonts
const style = {
  '@font-face': {
    family: 'font-name',
    url: '/fonts/',
    styles: ['normal', 'italic'],
    weights: [400, 600],
  },
}

await css.stringify(
  style,
) // css string
`
@font-face {
  font-family: "font-name";
  font-style: normal;
  font-weight: normal;
  src: url('/fonts/font-name-400-normal.eot\');
  src: url('/fonts/font-name-400-normal.eot#iefix') format('embedded-opentype'),',
  url('/fonts/font-name-400-normal.ttf') format('truetype'),',
  url('/fonts/font-name-400-normal.woff') format('woff'),',
  url('/fonts/font-name-400-normal.woff2') format('woff2'),',
  url('/fonts/font-name-400-normal.svg#font-name') format('svg');',
}
`
// ... repeated for all styles and weights that were defined

CHANGELOG:

0.1.0

return classes and ids as well

0.2.0:

Added media queries

0.3.0:

  • returns a promise! no longer sync.
  • autoprefixer and postcss added

0.4.0:

supports @font-face declarations

0.4.1

added fontDir option to font-face declarations

0.4.2

update deps to fix security issues

0.4.3

added @keyframes for animations

0.4.4

css.parse now converts deep arrays into one object before parsing

0.4.5

  • css.stringify now makes a bit nicer mediaquery and keyframe lines.
  • css.parse should now order media queries to the end of the css

0.4.6

css.parse does a better job of keeping order of incoming objects intact

0.4.7

  • FIX: multiple parent selectors 'h1,h2,h3' that should get appended with a &:hover etc now correctly append the suffix to every parent
  • FEATURE: & can be at the end of a selector, in which case the selector gets prepended to it's parent.

0.5.0

FIX: css gets returned in almost the same order it got passed in. turns out that javascript objects do not make this as easy as hoped.

0.6.0

use esmodules

0.7.0

update @magic/types to not break if es modules get passed into the styles

0.7.1

  • update @magic/deep to use ecmascript module version
  • update postcss, autoprefixer

0.7.2

update dependencies

0.7.3

update dependencies

0.7.4

add @magic/log to dependencies

0.7.5

update autoprefixer

0.7.6

update postcss

0.7.7

update postcss

0.7.8

update @magic/deep

0.7.9

require node >= 13.5.0

0.7.10

update postcss

0.7.11

update autoprefixer

0.7.12

update postcss

0.7.13
  • bump required node version to 14.2.0
  • write is async now
0.7.14
  • update autoprefixer, dependencies
0.7.15
  • update dependencies
0.7.16

update autoprefixer

0.7.17

update autoprefixer

0.7.18
  • move no-spy to devdeps
  • update dependencies
0.7.19

update autoprefixer

0.7.20

update dependencies

0.7.21

update dependencies

0.7.22

update dependencies

0.7.23

update dependencies

0.7.24

update dependencies

0.7.25

update dependencies

0.7.26

update dependencies

0.7.27

update dependencies simplify font-face handling

0.7.28
  • bump required node version to 14.15.4
  • update dependencies
0.7.29
  • font-face handles non-array weights, types and styles
  • font-face uses woff and woff2 as default font types
  • font-face uses weight 400 and styles 'normal' as defaults
0.7.30

update dependencies

0.7.31

update dependencies (@magic/fs)

0.7.32
  • update dependencies
  • css.selectors returns arrays instead of strings
0.7.33

update dependencies

0.7.34

update dependency autoprefixer

0.7.35

update dependency @magic/fs

0.7.36

remove local implementation of camel2kebab, use @magic/cases instead

0.7.37

update dependencies

0.7.38

FIX: update @magic/cases, which fixes overflow-x and other css props that end with an uppercase character overflowX is output as overflowx in earlier versions, the regex expected lowercase characters after the uppercase char

0.7.39

update dependencies

0.7.40

update dependencies

0.7.41

update dependencies

0.7.42

update dependencies

0.7.43

  • update dependencies
  • added font v2 to allow handling of local() fonts (see https://magic.github.io/css/#styles-webfonts)
  • woff2 files are placed before woff files
  • use single quotes in output css
  • add cli

0.7.44

  • css props can be arrays to provide css overloads { color: ['green', 'red'] } turns into color: green; color: red;
  • update dependencies

0.7.45

update dependencies

0.7.46

update dependencies

0.7.47

update dependencies

0.7.48

update dependencies

0.7.49

update dependencies

0.7.50

update dependencies

0.7.51

update dependencies

0.7.52

update dependencies

0.7.53

  • update dependencies
  • update docs to reflect usage of vars.widths.{tablet|laptop|desktop}
  • add defaults for opts.widths, to make sure we can use media queries at all times.
opts.widths = {
  tablet: '500px',
  laptop: '900px',
  desktop: '1200px',
  agency: '1600px',
}

0.7.54

update dependencies

0.7.55

update dependencies

0.7.56

update dependencies

0.7.57 - unreleased

...