P69 enables use of compile time tokens within CSS strings for Node based projects.
It's just a glorified fiind and replace, i.e. it scans CSS strings for placeholder tokens which are substituted for user defined values.
Create an arbitary nested object containing your tokens. There are no standards or conventions on how one should name and organise them. Just keep it simple and do what works, not what everyone else is doing!
- P69: https://github.com/PaulioRandall/p69
- P69 Files: https://github.com/PaulioRandall/p69-files
- P69 Svelte: https://github.com/PaulioRandall/p69-svelte
- P69 Util: https://github.com/PaulioRandall/p69-util
import P69 from 'p69'
const mappings = {
color: {
normal: 'burlywood',
highlight: 'crimson ',
},
font: {
size: {
sm: '0.8rem',
md: '1rem',
lg: '1.2rem',
},
},
width: (size = 'md') => {
const sizes = {
xs: '5rem',
sm: '10rem',
md: '15rem',
lg: '20rem',
xl: '25rem',
}
return sizes[md]
},
}
const cssWithTokens = `
.my-class {
color: $color.normal;
font-weight: bold;
font-size: $font.size.md;
width: $width('lg');
}
.my-class:hover {
color: &color.highlight;
}
`
const css = P69.string(mappings, cssWithTokens)
// Expect CSS to equal:
`
.my-class {
color: burlywood;
font-size: 1rem;
width: 20rem;
}
.my-class:hover {
color: crimson;
}
`
You can pass multiple mappings. It will search each mapping in order until it finds a value, e.g.
P69.string([fonts, colors], cssWithTokens)
P69.string(
mappings,
cssWithTokens,
options: {
// onError is called when an error occurs.
//
// If the error isn't thrown then processing will
// continue for the remaining tokens.
onError: (err, token) => {
// By default, logs the error and carries on.
},
}
)
- All tokens must be prefixed with
$
. - Functions can have arguments, e.g.
$func(1, 2, 3)
. - A function with no arguments needs no parenthesis, e.g.
$func
==$func()
. - String arguments to functions do not require quotation but single or double quotes may be used for escaping characters.
- There is no special escape character, instead create a mapping to handle escaping.
- Any value type is allowed as token value except undefined and object.
- Functions are invoked and the result returned as the token value.
- But a function cannot return undefined, object, or another function (because recursion is unnecessary and would just cause problems).
- Async functions are not allowed either; fetch any external data before you start processing.
- Nulls are resolved to empty strings, discarding any suffix.
Harmless bug: you can pass arguments to a non-function but they're ignored. I may fix this in future.
There's no escape character for the $
symbol, but it's easy to write your own. A few possibilities:
export const escapeMethods = {
// Simplest approach is to use $$, $$$, $$$$, etc.
// Add more as you need.
$: '$',
$$: '$$',
$$$: '$$$',
// We can create a single function that handles an unbroken
// series of $.
//
// $$ => $
// $$(2) => $$
// $$(3) => $$$
$: (n = 1) => '$'.repeat(n),
// Create a 'literal' function that returns its first argument.
//
// $literal("$$$") => $$$
// $literal("$ one $$ two $$$ three") => $ one $$ two $$$ three
literal: (v = '') => v.toString(),
// The world's your Mollusc. Here's a quotation function.
//
// $quote('Lots of $$$') => "Lots of $$$"
// $quote('Lots of $$$', '`') => `Lots of $$$`
quote: (v, glyph = '"') => glyph + v.toString() + glyph,
}