Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/noriller/easy-tailwind
An easier way of writing Tailwind classes.
https://github.com/noriller/easy-tailwind
javascript nextjs react tailwind tailwind-css tailwindcss typescript
Last synced: 7 days ago
JSON representation
An easier way of writing Tailwind classes.
- Host: GitHub
- URL: https://github.com/noriller/easy-tailwind
- Owner: Noriller
- License: mit
- Created: 2022-12-10T13:41:47.000Z (about 2 years ago)
- Default Branch: master
- Last Pushed: 2022-12-23T15:46:05.000Z (about 2 years ago)
- Last Synced: 2025-01-16T05:07:01.005Z (7 days ago)
- Topics: javascript, nextjs, react, tailwind, tailwind-css, tailwindcss, typescript
- Language: TypeScript
- Homepage: https://www.npmjs.com/package/easy-tailwind
- Size: 169 KB
- Stars: 172
- Watchers: 3
- Forks: 2
- Open Issues: 3
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Easy Tailwind
An easier way of writing Tailwind classes.
[Example to play around](https://stackblitz.com/edit/easy-tailwind?file=src/App.jsx)
## Table of Contents
- [What this is and what this isn't](#what-this-is-and-what-this-isnt)
- [Installation](#installation)
- [Setup](#setup)
- [How to Use](#how-to-use)
- [Break Lines](#break-lines)
- [Use Objects](#use-objects)
- [Conditional Classes](#conditional-classes)
- [Rules for it to Work](#rules-for-it-to-work)
- [Does it Support XYZ?](#does-it-support-xyz)
- [Final Considerations](#final-considerations)
- [Why "Easy" Tailwind?](#why-easy-tailwind)## What this is and what this isn't
### What this isn't
This is not [`WindiCSS`](https://github.com/windicss/windicss), [`UnoCSS`](https://github.com/unocss/unocss) or any other CSS lib or framework. This isn't meant to replace them.
This is meant to be used with Tailwind. So, if you're not using Tailwind, you don't want this.
### What this might be
If you use [`classnames`](https://github.com/JedWatson/classnames), [`clsx`](https://github.com/lukeed/clsx/) and other utilities to have conditional classes, then this might be a replacement for them.
This doesn't cover all cases they do, and you could use all of them in conjunction if you want (they would wrap `e`/`etw` functions).
But if you just use them for class toggling and use Tailwind, then you might want to consider replacing them with this.
### What this is
This is a utility to be used with Tailwind. If you're using Tailwind, you want to consider using this.
This is a tool to increase Developer Experience. The Tailwind world-class extension still works, even while writing with EasyTailwind. (It doesn't show the whole CSS class generated when using the modifiers, but it shows the important part.)
This is a tool for cleaner code. You might not agree, but I developed that in mind.
This is like "table salt", salt is good but you don't want to cover everything in it.
If you have just a couple of classes then there's no need to call it. Call it when you have multiple classes, especially with modifiers or when you need to toggle classes.
Go to: [Table of Contents](#table-of-contents)
## Installation
Install with your preferred manager:
```bash
npm i easy-tailwind
``````bash
yarn add easy-tailwind
``````bash
pnpm add easy-tailwind
```Go to: [Table of Contents](#table-of-contents)
## Setup
The configuration file is usually located in the `root` of the application and looks something like `tailwind.config.cjs`. The basic configuration you need is:
```js
// tailwind.config.cjs
const { content } = require('easy-tailwind/transform');
/** @type {import('tailwindcss').Config} */
module.exports = {
content,
// ...
theme: {
// ...
},
plugins: [
// ...
],
};
```If you need to override something in the `content` config:
```js
// tailwind.config.cjs
const { replacer } = require('easy-tailwind/transform');
/** @type {import('tailwindcss').Config} */
module.exports = {
content: {
files: [
'!node_modules',
'./**/*.{js,ts,jsx,tsx,html,vue,svelte,astro}' // here you can specify your own files and directories
],
transform: {
DEFAULT: replacer, // default is applied to all files types
},
}
// ...
theme: {
// ...
},
plugins: [
// ...
],
};
```### Specific configuration for frameworks
Right now, the only specific one is `React`, instead of importing from `easy-tailwind/transform`, import from `easy-tailwind/transform/react`
In the example for `React`:
```js
// tailwind.config.cjs
const { content } = require('easy-tailwind/transform/react');
/** @type {import('tailwindcss').Config} */
module.exports = {
content,
// ...
theme: {
// ...
},
plugins: [
// ...
],
};
```Where `content` is equivalent to:
```js
// tailwind.config.cjs
const { replacer } = require('easy-tailwind/transform');
/** @type {import('tailwindcss').Config} */
module.exports = {
content: {
files: ['!node_modules', './**/*.{js,ts,jsx,tsx,html}'],
transform: {
DEFAULT: replacer,
},
}
// ...
theme: {
// ...
},
plugins: [
// ...
],
};
```Go to: [Table of Contents](#table-of-contents)
### Renaming the exports
If you rename the exports to something other than `e` or `etw`, or maybe you want to use only one because you're already using another function with the same name, then you need to change the `replacer` with the `customNameReplacer` (exported from `easy-tailwind/transform`).
Example:
```js
// tailwind.config.cjs
const { customNameReplacer } = require('easy-tailwind/transform');
/** @type {import('tailwindcss').Config} */
module.exports = {
content: {
files: [
'!node_modules',
'./**/*.{js,ts,jsx,tsx,html,vue,svelte,astro}' // here you can specify your own files and directories
],
transform: {
DEFAULT: customNameReplacer('newFuncName1', 'newFuncName2'), // default is applied to all files types
'some-file-extension': customNameReplacer('etw'), // this one you know you're only using `etw`
'some-other-file-extension': customNameReplacer('newFuncName'), // this one you know you're only using `newFuncName'
},
}
// ...
theme: {
// ...
},
plugins: [
// ...
],
};
```Go to: [Table of Contents](#table-of-contents)
## How to use
First, import `e` or `etw`:
```js
import { e } from 'easy-tailwind';
// e and etw resolve to the same function
```This is a pure function, so you can be sure that it will always return the same values as you pass the same values:
```js
e(
'some base classes here',
'breaking the line because',
'it was getting too long',
{
mod1: 'classes with mod1',
mod2: [
'classes with only mod2',
{
subMod1: 'nested classes with both',
},
],
},
);
// this will return:
'some base classes here breaking the line because it was getting too long mod1:classes mod1:with mod1:mod1 mod2:classes mod2:with mod2:only mod2:mod2 mod2:subMod1:nested mod2:subMod1:classes mod2:subMod1:with mod2:subMod1:both';
```Now use it where you would use the Tailwind classes.
Example below will use the React syntax, but as long as you can call it, it will probably work:
```js
EasyTailwind!!!
```Which is way faster and easier to understand, maintain and debug than:
```js
"text-lg font-medium text-black hover:underline hover:decoration-black sm:text-base sm:text-blue-500 sm:hover:decoration-cyan-500 lg:text-2xl lg:text-green-500 lg:hover:decoration-amber-500"
```> βΉοΈ Sense of style not included. π€£
Go to: [Table of Contents](#table-of-contents)
### Break lines
One of the uses is to "break lines" of the styles.
For this, just split the classes into multiple strings and put each one in a single line.
Example:
```js
Multiple lines!
```Go to: [Table of Contents](#table-of-contents)
### Use Objects
This is what you were waiting for, objects where the keys are applied to whatever the value is, even nested structures.
Example:
```js
Objects!
```This will create exactly what you would expect:
```js
"sm:text-sm sm:text-blue-500 md:text-base md:text-green-500 lg:text-lg lg:text-black lg:hover:text-red-500"
```Each key (sm, md, lg, [&_li], group-hover, ...) will apply to everything inside the value.
Each value can be a string, another object, or an array with strings and/or objects.
> β οΈ The ordering is applied in the order of the object.
>
> Some Tailwind classes don't work properly depending on the order,
>
> so always check if what you will be building is valid.Go to: [Table of Contents](#table-of-contents)
### Conditional classes
One thing we usually need is conditional classes, we got you covered!
As long as you follow [the rules](#rules-for-it-to-work):
- Use boolean values for conditional expressions (ternary, &&, ||, ??, etc...)
- Don't add variables other than the boolean for the conditional expressionsExample:
```js
const boolean = Math.random() > 0.5;
// ...
Conditional classes!
```Go to: [Table of Contents](#table-of-contents)
## Rules for it to work
1. Use boolean values for conditional expressions (ternary, &&, ||, ??, etc...)
2. Don't add variables other than the boolean for the conditional expressionsGo to: [Table of Contents](#table-of-contents)
### Known limitations
Those limitations are about the `replacer` in the transform.
Right now, depending on what you try to use in the `e` function, it will not work and you will be given a warning (check the terminal where you're running your `dev` or `build` script).
When you're inspecting it in the browser, it will show the classes as normal, but it won't work (the function works as normal, but the transformer won't be able to parse it and the classes won't be added and sent to the browser).
In those cases, you can append them separately from those you use with `EasyTailwind`.
Examples:
```js
className={`${variable} ${Math.random() > 0.5 ? "more" : "less"} ${e("here goes the safe to parse classes")}`}
```### Why is this necessary?
Tailwind works with the JIT compiler that can create new classes on the fly and inject them.
For it to work, they need to scan all the files looking for the classes, but when you use `EasyTailwind`,
you're basically compressing many of the classes you're trying to use. So we need to add an extra step to Tailwind.In the Tailwind config (`tailwind.config.cjs`) you add the files it will scan for tailwind classes and a transform that uses a function that will resolve ahead of time what `EasyTailwind` can produce, so Tailwind can inject ahead of time all possible classes.
However, the more complicated and inclusive you want it to scan for, the more you lose performance (for running in dev mode and for build).
The best balance to be able to accept having conditional classes while minimizing the impact on performance is to simplify this, looking for only a boolean variable and not something that can be as simple as a variable or as complex as complex can be.
See more at [Tailwind "Transforming source files"](https://tailwindcss.com/docs/content-configuration#transforming-source-files).
Go to: [Table of Contents](#table-of-contents)
## Does it Support XYZ?
Probably.
If you can use `e('tw classes')` and it generates the classes (even if they don't actually work), then just follow the setup part.
If you want me to add a custom `content` for the framework you're using, feel free to open a PR. =D
Go to: [Table of Contents](#table-of-contents)
## Final Considerations
These are mostly 'pure' functions, so we don't need to worry about getting "stale".
More functionalities are welcome, but ultimately this package can have months between any updates.
This doesn't mean that it's "dead", just that it's doing what it needs.Today it works with Tailwind v3, I'm not sure if with lower versions or for higher versions.
As long as the `content` part doesn't change, then you can just import and use it.
If it changes, you have the `replacer` for the transformations (as long as it supports it) but expect updates as soon as possible.Go to: [Table of Contents](#table-of-contents)
### Why "Easy" Tailwind?
I'm lazy.
And while I'm already productive using Tailwind, I don't like to keep repeating the same modifiers over and over again.
So, it's "easy" to type.
Another thing is about reading the classes. It's easy to get a long string with all classes jumbled together, even with extensions sorting and linting them it's hard to keep in mind everything that's happening at once.
So, it's "easy" to read.
And well, naming is hard and I went with the first thing I thought about. =p
Go to: [Table of Contents](#table-of-contents)
## Work with me
### Hit me up at Discord
### Or Donate
- [$5 Nice job! Keep it up.](https://www.paypal.com/donate/?business=VWNG7KZD9SS4S&no_recurring=0¤cy_code=USD&amount=5)
- [$10 I really liked that, thank you!](https://www.paypal.com/donate/?business=VWNG7KZD9SS4S&no_recurring=0¤cy_code=USD&amount=10)
- [$42 This is exactly what I was looking for.](https://www.paypal.com/donate/?business=VWNG7KZD9SS4S&no_recurring=0¤cy_code=USD&amount=42)
- [$1K WOW. Did not know javascript could do that!](https://www.paypal.com/donate/?business=VWNG7KZD9SS4S&no_recurring=0¤cy_code=USD&amount=1000)
- [$5K I need something done ASAP! Can you do it for yesterday?](https://www.paypal.com/donate/?business=VWNG7KZD9SS4S&no_recurring=0¤cy_code=USD&amount=5000)
- [$10K Please consider this: quit your job and work with me!](https://www.paypal.com/donate/?business=VWNG7KZD9SS4S&no_recurring=0¤cy_code=USD&amount=10000)
- [$??? Shut up and take my money!](https://www.paypal.com/donate/?business=VWNG7KZD9SS4S&no_recurring=0¤cy_code=USD)## Thatβs it! π