2
0
mirror of https://github.com/iconify/iconify.git synced 2025-01-07 07:34:22 +00:00

feat(tailwind): new version with dynamic icons

BREAKING CHANGE: plugin uses named exports now, see README.md for usage
This commit is contained in:
Vjacheslav Trushkin 2023-01-12 20:02:54 +02:00
parent c052341bc2
commit 2841b3ff05
10 changed files with 319 additions and 233 deletions

View File

@ -7,22 +7,23 @@ This plugin creates CSS for over 100k open source icons.
## Usage
1. Install packages icon sets.
2. In `tailwind.config.js` import plugin and specify list of icons you want to load.
2. In `tailwind.config.js` import `addDynamicIconSelectors` from `@iconify/tailwind`.
## HTML
To use icon in HTML, it is as easy as adding 2 class names:
- Class name for icon set: `icon--{prefix}`.
- Class name for icon: `icon--{prefix}--{name}`.
To use icon in HTML, add class with class name like this: `icon-[mdi-light--home]`
```html
<span class="icon--mdi icon--mdi--home"></span>
<span class="icon-[mdi-light--home]"></span>
```
Why 2 class names? It reduces duplication and makes it easy to target all icons from one icon set.
Class name has 3 parts:
You can change that with options: you can change class names format, you can disable common selector. See [options for function used by plugin](https://docs.iconify.design/tools/utils/get-icons-css.html).
- Selectot prefix, which can be set in `prefix` option of plugin. Default value is `icon`.
- `-` to tell Tailwind that class name is not complete.
- `[{prefix}--{name}]` for icon name, where `{prefix}` is icon set prefix, `{name}` is icon name.
In Iconify all icon names use the following format: `{prefix}:{name}`. Due to limitations of Tailwind CSS, same format cannot be used with plugin, so instead, prefix and name are separated by double dash: `{prefix}--{name}`.
### Color, size, alignment
@ -35,7 +36,7 @@ Icon color cannot be changed for icons with hardcoded palette, such as most emoj
To align icon below baseline, add negative vertical alignment, like this (you can also use Tailwind class for that):
```html
<span class="icon--mdi icon--mdi--home" style="vertical-align: -0.125em"></span>
<span class="icon-[mdi--home]" style="vertical-align: -0.125em"></span>
```
## Installing icon sets
@ -55,10 +56,10 @@ See [Iconify documentation](https://docs.iconify.design/icons/json.html) for lis
Add this to `tailwind.config.js`:
```js
const iconifyPlugin = require('@iconify/tailwind');
const { addDynamicIconSelectors } = require('@iconify/tailwind');
```
Then in plugins section add `iconifyPlugin` with list of icons you want to load.
Then in plugins section add `addDynamicIconSelectors`.
Example:
@ -69,22 +70,45 @@ module.exports = {
extend: {},
},
plugins: [
// Iconify plugin with list of icons you need
iconifyPlugin(['mdi:home', 'mdi-light:account']),
// Iconify plugin
addDynamicIconSelectors(),
],
presets: [],
};
```
### Icon names
Unfortunately Tailwind CSS cannot dynamically find all icon names. You need to specify list of icons you want to use.
### Options
Plugin accepts options as a second parameter. You can use it to change selectors.
Plugin accepts options as a second parameter:
See [documentation for function used by plugin](https://docs.iconify.design/tools/utils/get-icons-css.html) for list of options.
- `prefix` is class name prefix. Default value is `icon`. Make sure there is no `-` at the end: it is added in classes, but not in plugin parameter.
- `overrideOnly`: set to `true` to generate rules that override only icon data. See below.
- `files`: list of custom files for icon sets. Key is icon set prefix, value is location of `.json` file with icon set in IconifyJSON format.
- `iconSet`: list of custom icon sets. Key is prefix, value is either icon set data in `IconifyJSON` format or a synchronous callback that returns `IconifyJSON` data.
#### overrideOnly
You can use `overrideOnly` to load some icons without full rules, such as changing icon on hover when main and hover icons are from the same icon set and have same width/height ratio.
Example of config:
```js
plugins: [
// `icon-`
addDynamicIconSelectors(),
// `icon-hover-`
addDynamicIconSelectors({
prefix: "icon-hover",
overrideOnly: true,
}),
],
```
and usage in HTML:
```html
<span class="icon-[mdi--arrow-left] hover:icon-hover-[mdi--arrow-right]"></span>
```
## License

View File

@ -2,7 +2,7 @@
"name": "@iconify/tailwind",
"description": "Iconify plugin for Tailwind CSS",
"author": "Vjacheslav Trushkin <cyberalien@gmail.com> (https://iconify.design)",
"version": "0.0.2",
"version": "0.1.0",
"license": "MIT",
"main": "./dist/plugin.js",
"types": "./dist/plugin.d.ts",

View File

@ -0,0 +1,43 @@
import { getIconsCSSData } from '@iconify/utils/lib/css/icons';
import { loadIconSet } from './loader';
import { getIconNames } from './names';
import type { CleanIconifyPluginOptions } from './options';
/**
* Get CSS rules for icons list
*/
export function getCSSRulesForIcons(
icons: string[] | string,
options: CleanIconifyPluginOptions = {}
): Record<string, Record<string, string>> {
const rules = Object.create(null) as Record<string, Record<string, string>>;
// Get all icons
const prefixes = getIconNames(icons);
// Parse all icon sets
for (const prefix in prefixes) {
const iconSet = loadIconSet(prefix, options);
if (!iconSet) {
throw new Error(`Cannot load icon set for "${prefix}"`);
}
const generated = getIconsCSSData(
iconSet,
Array.from(prefixes[prefix]),
options
);
const result = generated.common
? [generated.common, ...generated.css]
: generated.css;
result.forEach((item) => {
const selector =
item.selector instanceof Array
? item.selector.join(', ')
: item.selector;
rules[selector] = item.rules;
});
}
return rules;
}

View File

@ -0,0 +1,44 @@
import { getIconsCSSData } from '@iconify/utils/lib/css/icons';
import { matchIconName } from '@iconify/utils/lib/icon/name';
import { loadIconSet } from './loader';
import type { DynamicIconifyPluginOptions } from './options';
/**
* Get dynamic CSS rules
*/
export function getDynamicCSSRules(
icon: string,
options: DynamicIconifyPluginOptions = {}
): Record<string, string> {
const nameParts = icon.split(/--|\:/);
if (nameParts.length !== 2) {
throw new Error(`Invalid icon name: "${icon}"`);
}
const [prefix, name] = nameParts;
if (!prefix.match(matchIconName) || !name.match(matchIconName)) {
throw new Error(`Invalid icon name: "${icon}"`);
}
const iconSet = loadIconSet(prefix, options);
if (!iconSet) {
throw new Error(`Cannot load icon set for "${prefix}"`);
}
const generated = getIconsCSSData(iconSet, [name], {
iconSelector: '.icon',
});
if (generated.css.length !== 1) {
throw new Error(`Something went wrong generating "${icon}"`);
}
return {
// Common rules
...(options.overrideOnly || !generated.common?.rules
? {}
: generated.common.rules),
// Icon rules
...generated.css[0].rules,
};
}

View File

@ -1,159 +0,0 @@
import { getIconsCSSData } from '@iconify/utils/lib/css/icons';
import { matchIconName } from '@iconify/utils/lib/icon/name';
import { loadIconSet } from './loader';
import type { IconifyPluginOptions } from './options';
const missingIconsListError =
'TailwindCSS cannot dynamically find all used icons. Need to pass list of used icons to Iconify plugin.';
/**
* Get icon names from list
*/
function getIconNames(icons: string[] | string): Record<string, Set<string>> {
const prefixes = Object.create(null) as Record<string, Set<string>>;
// Add entry
const add = (prefix: string, name: string) => {
if (
typeof prefix === 'string' &&
prefix.match(matchIconName) &&
typeof name === 'string' &&
name.match(matchIconName)
) {
(prefixes[prefix] || (prefixes[prefix] = new Set())).add(name);
}
};
// Comma or space separated string
let iconNames: string[] | undefined;
if (typeof icons === 'string') {
iconNames = icons.split(/[\s,.]/);
} else if (icons instanceof Array) {
iconNames = [];
// Split each array entry
icons.forEach((item) => {
item.split(/[\s,.]/).forEach((name) => iconNames.push(name));
});
} else {
throw new Error(missingIconsListError);
}
// Parse array
if (iconNames?.length) {
iconNames.forEach((icon) => {
if (!icon.trim()) {
return;
}
// Attempt prefix:name split
const nameParts = icon.split(':');
if (nameParts.length === 2) {
add(nameParts[0], nameParts[1]);
return;
}
// Attempt icon class: .icon--{prefix}--{name}
// with or without dot
const classParts = icon.split('--');
if (classParts[0].match(/^\.?icon$/)) {
if (classParts.length === 3) {
add(classParts[1], classParts[2]);
return;
}
if (classParts.length === 2) {
// Partial match
return;
}
}
// Throw error
throw new Error(`Cannot resolve icon: "${icon}"`);
});
} else {
throw new Error(missingIconsListError);
}
return prefixes;
}
/**
* Get CSS rules for icon
*/
export function getCSSRules(
icons: string[] | string,
options: IconifyPluginOptions = {}
): Record<string, Record<string, string>> {
const rules = Object.create(null) as Record<string, Record<string, string>>;
// Get all icons
const prefixes = getIconNames(icons);
// Parse all icon sets
for (const prefix in prefixes) {
const iconSet = loadIconSet(prefix, options);
if (!iconSet) {
throw new Error(`Cannot load icon set for "${prefix}"`);
}
const generated = getIconsCSSData(
iconSet,
Array.from(prefixes[prefix]),
options
);
const result = generated.common
? [generated.common, ...generated.css]
: generated.css;
result.forEach((item) => {
const selector =
item.selector instanceof Array
? item.selector.join(', ')
: item.selector;
rules[selector] = item.rules;
});
}
return rules;
}
/**
* Get dynamic CSS rule
*/
export function getDynamicCSSRules(
selector: string,
icon: string,
options: IconifyPluginOptions = {}
): Record<string, string> {
const nameParts = icon.split('--');
let nameError = `Invalid icon name: "${icon}"`;
if (nameParts.length !== 2) {
if (nameParts.length === 1 && icon.indexOf(':') !== -1) {
nameError += `. "{prefix}:{name}" is not supported because of Tailwind limitations, use "{prefix}--{name}" (use double dash!) instead.`;
}
throw new Error(nameError);
}
const [prefix, name] = nameParts;
if (!prefix.match(matchIconName) || !name.match(matchIconName)) {
throw new Error(nameError);
}
const iconSet = loadIconSet(prefix, options);
if (!iconSet) {
throw new Error(`Cannot load icon set for "${prefix}"`);
}
const generated = getIconsCSSData(iconSet, [name], {
...options,
// One selector
iconSelector: selector,
commonSelector: selector,
overrideSelector: selector,
});
if (generated.css.length !== 1) {
throw new Error(`Something went wrong generating "${icon}"`);
}
return {
...(generated.common?.rules || {}),
...generated.css[0].rules,
};
}

View File

@ -0,0 +1,73 @@
import { matchIconName } from '@iconify/utils/lib/icon/name';
/**
* Get icon names from list
*/
export function getIconNames(
icons: string[] | string
): Record<string, Set<string>> | undefined {
const prefixes = Object.create(null) as Record<string, Set<string>>;
// Add entry
const add = (prefix: string, name: string) => {
if (
typeof prefix === 'string' &&
prefix.match(matchIconName) &&
typeof name === 'string' &&
name.match(matchIconName)
) {
(prefixes[prefix] || (prefixes[prefix] = new Set())).add(name);
}
};
// Comma or space separated string
let iconNames: string[] | undefined;
if (typeof icons === 'string') {
iconNames = icons.split(/[\s,.]/);
} else if (icons instanceof Array) {
iconNames = [];
// Split each array entry
icons.forEach((item) => {
item.split(/[\s,.]/).forEach((name) => iconNames.push(name));
});
} else {
return;
}
// Parse array
if (iconNames?.length) {
iconNames.forEach((icon) => {
if (!icon.trim()) {
return;
}
// Attempt prefix:name split
const nameParts = icon.split(':');
if (nameParts.length === 2) {
add(nameParts[0], nameParts[1]);
return;
}
// Attempt icon class: .icon--{prefix}--{name}
// with or without dot
const classParts = icon.split('--');
if (classParts[0].match(/^\.?icon$/)) {
if (classParts.length === 3) {
add(classParts[1], classParts[2]);
return;
}
if (classParts.length === 2) {
// Partial match
return;
}
}
// Throw error
throw new Error(`Cannot resolve icon: "${icon}"`);
});
} else {
return;
}
return prefixes;
}

View File

@ -1,29 +1,30 @@
import type { IconCSSIconSetOptions } from '@iconify/utils/lib/css/types';
import type { IconifyPluginLoaderOptions } from './loader';
/**
* Options for locating icon sets
* Common options
*/
export interface IconifyPluginFileOptions {
// Files
files?: Record<string, string>;
}
/**
* Options for matching dynamic icon names
*/
export interface IconifyPluginDynamicPrefixOptions {
// Dynamic prefix for selectors. Default is `icon`
// Allows using icon names like `<span class="icon[mdi--home]"></span>
// Where prefix and name are separated by '--' because Tailwind does not allow ':'
dynamicPrefix?: string;
}
/**
* All options
*/
export interface IconifyPluginOptions
extends IconCSSIconSetOptions,
IconifyPluginDynamicPrefixOptions,
IconifyPluginFileOptions {
export interface CommonIconifyPluginOptions extends IconifyPluginLoaderOptions {
//
}
/**
* Options for clean class names
*/
export interface CleanIconifyPluginOptions
extends CommonIconifyPluginOptions,
IconCSSIconSetOptions {
//
}
/**
* Options for dynamic class names
*/
export interface DynamicIconifyPluginOptions
extends CommonIconifyPluginOptions {
// Class prefix
prefix?: string;
// Inclue icon-specific selectors only
overrideOnly?: true;
}

View File

@ -1,13 +1,29 @@
import plugin from 'tailwindcss/plugin';
import { getCSSRules, getDynamicCSSRules } from './iconify';
import type { IconifyPluginOptions } from './options';
import { getCSSRulesForIcons } from './clean';
import { getDynamicCSSRules } from './dynamic';
import type {
CleanIconifyPluginOptions,
DynamicIconifyPluginOptions,
} from './options';
/**
* Iconify plugin
* Generate styles for dynamic selector: class="icon-[mdi-light--home]"
*/
function iconifyPlugin(
export function addDynamicIconSelectors(options?: DynamicIconifyPluginOptions) {
const prefix = options?.prefix || 'icon';
return plugin(({ matchComponents }) => {
matchComponents({
[prefix]: (icon: string) => getDynamicCSSRules(icon, options),
});
});
}
/**
* Generate styles for preset list of icons
*/
export function addCleanIconSelectors(
icons?: string[] | string,
options?: IconifyPluginOptions
options?: CleanIconifyPluginOptions
) {
const passedOptions =
typeof icons === 'object' && !(icons instanceof Array)
@ -16,32 +32,17 @@ function iconifyPlugin(
const passedIcons =
typeof icons !== 'object' || icons instanceof Array ? icons : void 0;
// Get selector for dynamic classes
const dynamicSelector = passedOptions.dynamicPrefix || 'icon';
// Get hardcoded list of icons
const rules = passedIcons
? getCSSRules(passedIcons, passedOptions)
? getCSSRulesForIcons(passedIcons, passedOptions)
: void 0;
return plugin(({ addUtilities, matchComponents }) => {
if (rules) {
addUtilities(rules);
}
matchComponents({
[dynamicSelector]: (icon: string) =>
getDynamicCSSRules(
`.${dynamicSelector}-[${icon}]`,
icon,
passedOptions
),
});
});
}
/**
* Export stuff
* Export types
*/
export default iconifyPlugin;
export type { IconifyPluginOptions };
export type { CleanIconifyPluginOptions, DynamicIconifyPluginOptions };

View File

@ -1,8 +1,8 @@
import { getCSSRules } from '../src/iconify';
import { getCSSRulesForIcons } from '../src/clean';
describe('Testing CSS rules', () => {
describe('Testing clean CSS rules', () => {
it('One icon', () => {
const data = getCSSRules('mdi-light:home');
const data = getCSSRulesForIcons('mdi-light:home');
expect(Object.keys(data)).toEqual([
'.icon--mdi-light',
'.icon--mdi-light--home',
@ -11,7 +11,7 @@ describe('Testing CSS rules', () => {
});
it('Multiple icons from same icon set', () => {
const data = getCSSRules([
const data = getCSSRulesForIcons([
// By name
'mdi-light:home',
// By selector
@ -32,7 +32,7 @@ describe('Testing CSS rules', () => {
});
it('Multiple icon sets', () => {
const data = getCSSRules([
const data = getCSSRulesForIcons([
// MDI Light
'mdi-light:home',
// Line MD
@ -49,7 +49,7 @@ describe('Testing CSS rules', () => {
it('Bad class name', () => {
let threw = false;
try {
getCSSRules(['icon--mdi-light--home test']);
getCSSRulesForIcons(['icon--mdi-light--home test']);
} catch {
threw = true;
}
@ -59,7 +59,7 @@ describe('Testing CSS rules', () => {
it('Bad icon set', () => {
let threw = false;
try {
getCSSRules('test123:home');
getCSSRulesForIcons('test123:home');
} catch {
threw = true;
}

View File

@ -0,0 +1,59 @@
import { getDynamicCSSRules } from '../src/dynamic';
describe('Testing dynamic CSS rules', () => {
it('One icon', () => {
const data = getDynamicCSSRules('mdi-light--home');
expect(typeof data['--svg']).toBe('string');
expect(data).toEqual({
'display': 'inline-block',
'width': '1em',
'height': '1em',
'background-color': 'currentColor',
'-webkit-mask': 'no-repeat center / 100%',
'mask': 'no-repeat center / 100%',
'-webkit-mask-image': 'var(--svg)',
'mask-image': 'var(--svg)',
'--svg': data['--svg'],
});
});
it('Only selectors that override icon', () => {
const data = getDynamicCSSRules('mdi-light--home', {
overrideOnly: true,
});
expect(typeof data['--svg']).toBe('string');
expect(data).toEqual({
'--svg': data['--svg'],
});
});
it('Missing icon', () => {
let threw = false;
try {
getDynamicCSSRules('mdi-light--missing-icon-name');
} catch {
threw = true;
}
expect(threw).toBe(true);
});
it('Bad icon name', () => {
let threw = false;
try {
getDynamicCSSRules('mdi-home');
} catch {
threw = true;
}
expect(threw).toBe(true);
});
it('Bad icon set', () => {
let threw = false;
try {
getDynamicCSSRules('test123:home');
} catch {
threw = true;
}
expect(threw).toBe(true);
});
});