Skip to content

PaulioRandall/p69

Repository files navigation

Made to be Plundered Latest version Release date

P69

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!

Contents

Example

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)

^Back to contents

Options

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.
		},
	}
)

^Back to contents

Rules for Token Mappings

  1. All tokens must be prefixed with $.
  2. Functions can have arguments, e.g. $func(1, 2, 3).
  3. A function with no arguments needs no parenthesis, e.g. $func == $func().
  4. String arguments to functions do not require quotation but single or double quotes may be used for escaping characters.
  5. There is no special escape character, instead create a mapping to handle escaping.
  6. Any value type is allowed as token value except undefined and object.
  7. Functions are invoked and the result returned as the token value.
  8. But a function cannot return undefined, object, or another function (because recursion is unnecessary and would just cause problems).
  9. Async functions are not allowed either; fetch any external data before you start processing.
  10. 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.

^Back to contents

Escaping the prefix

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,
}

^Back to contents