Writing an Icon
component
We don't provide any pre-made, ready-to-use components. Such a solution would be too limited and opinionated for user-specific needs.
Instead, we offer a detailed yet simple guide on how to create your own components.
INFO
In this guide, we will use React, TypeScript, and Tailwind CSS.
Result component
From the start, I will show you the final result of this guide to give you a better understanding of what we are going to achieve.
- Supports grouped sprites with generated file names
- Type-safe
IconName
(format:spriteName/iconName
) for autocompletion and convenient usage - Autoscaling based on the icon's aspect ratio
- Open to any extension for your needs!
We will work with grouped hashed sprites with generated types and metadata
/
├── assets
│ ├── common
│ │ ├── left.svg
│ │ └── right.svg
│ └── actions
│ └── close.svg
├── public
+ └── sprites
+ ├── common.12ghS6Uj.svg
+ └── actions.1A34ks78.svg
└── src
+ └── sprite.gen.ts
/
├── assets
│ ├── common
│ │ ├── left.svg
│ │ └── right.svg
│ └── actions
│ └── close.svg
├── public
+ └── sprites
+ ├── common.12ghS6Uj.svg
+ └── actions.1A34ks78.svg
└── src
+ └── sprite.gen.ts
import clsx from 'clsx';
import type { SVGProps } from 'react';
import { SPRITES_META, type SpritesMap } from './sprite.gen';
// Our icon will extend an SVG element and accept all its props
export interface IconProps extends SVGProps<SVGSVGElement> {
name: AnyIconName;
}
// Merging all possible icon names as `sprite/icon` string
export type AnyIconName = { [Key in keyof SpritesMap]: IconName<Key> }[keyof SpritesMap];
// Icon name for a specific sprite, e.g. "common/left"
export type IconName<Key extends keyof SpritesMap> = `${Key}/${SpritesMap[Key]}`;
export function Icon({ name, className, ...props }: IconProps) {
const { viewBox, filePath, iconName, axis } = getIconMeta(name);
return (
<svg
// "icon" isn't inlined because of data-axis attribute
className={clsx('icon', className)}
viewBox={viewBox}
/**
* This prop is used by the "icon" class to set the icon's scaled size
* @see https://github.com/secundant/neodx/issues/92
*/
data-axis={axis}
// prevent icon from being focused when using keyboard navigation
focusable="false"
// hide icon from screen readers
aria-hidden
{...props}
>
{/* For example, "/sprites/common.svg#favourite". Change a base path if you don't store sprites under the "/sprites". */}
<use href={`/sprites/${filePath}#${iconName}`} />
</svg>
);
}
/**
* A function to get and process icon metadata.
* It was moved out of the Icon component to prevent type inference issues.
*/
const getIconMeta = <Key extends keyof SpritesMap>(name: IconName<Key>) => {
const [spriteName, iconName] = name.split('/') as [Key, SpritesMap[Key]];
const {
filePath,
items: {
[iconName]: { viewBox, width, height }
}
} = SPRITES_META[spriteName];
const axis = width === height ? 'xy' : width > height ? 'x' : 'y';
return { filePath, iconName, viewBox, axis };
};
import clsx from 'clsx';
import type { SVGProps } from 'react';
import { SPRITES_META, type SpritesMap } from './sprite.gen';
// Our icon will extend an SVG element and accept all its props
export interface IconProps extends SVGProps<SVGSVGElement> {
name: AnyIconName;
}
// Merging all possible icon names as `sprite/icon` string
export type AnyIconName = { [Key in keyof SpritesMap]: IconName<Key> }[keyof SpritesMap];
// Icon name for a specific sprite, e.g. "common/left"
export type IconName<Key extends keyof SpritesMap> = `${Key}/${SpritesMap[Key]}`;
export function Icon({ name, className, ...props }: IconProps) {
const { viewBox, filePath, iconName, axis } = getIconMeta(name);
return (
<svg
// "icon" isn't inlined because of data-axis attribute
className={clsx('icon', className)}
viewBox={viewBox}
/**
* This prop is used by the "icon" class to set the icon's scaled size
* @see https://github.com/secundant/neodx/issues/92
*/
data-axis={axis}
// prevent icon from being focused when using keyboard navigation
focusable="false"
// hide icon from screen readers
aria-hidden
{...props}
>
{/* For example, "/sprites/common.svg#favourite". Change a base path if you don't store sprites under the "/sprites". */}
<use href={`/sprites/${filePath}#${iconName}`} />
</svg>
);
}
/**
* A function to get and process icon metadata.
* It was moved out of the Icon component to prevent type inference issues.
*/
const getIconMeta = <Key extends keyof SpritesMap>(name: IconName<Key>) => {
const [spriteName, iconName] = name.split('/') as [Key, SpritesMap[Key]];
const {
filePath,
items: {
[iconName]: { viewBox, width, height }
}
} = SPRITES_META[spriteName];
const axis = width === height ? 'xy' : width > height ? 'x' : 'y';
return { filePath, iconName, viewBox, axis };
};
@layer components {
/*
Our base class for icons inherits the current text color and applies common styles.
We're using a specific component class to prevent potential style conflicts and utilize the [data-axis] attribute.
*/
.icon {
@apply select-none fill-current inline-block text-inherit box-content;
}
/* Set icon size to 1em based on its aspect ratio, so we can use `font-size` to scale it */
.icon[data-axis*='x'] {
/* scale horizontally */
@apply w-[1em];
}
.icon[data-axis*='y'] {
/* scale vertically */
@apply h-[1em];
}
}
@layer components {
/*
Our base class for icons inherits the current text color and applies common styles.
We're using a specific component class to prevent potential style conflicts and utilize the [data-axis] attribute.
*/
.icon {
@apply select-none fill-current inline-block text-inherit box-content;
}
/* Set icon size to 1em based on its aspect ratio, so we can use `font-size` to scale it */
.icon[data-axis*='x'] {
/* scale horizontally */
@apply w-[1em];
}
.icon[data-axis*='y'] {
/* scale vertically */
@apply h-[1em];
}
}
import svg from '@neodx/svg/vite';
import { defineConfig } from 'vite';
export default defineConfig({
plugins: [
svg({
root: 'assets',
// group icons by sprite name
group: true,
output: 'public/sprites',
// add hash to sprite file name
fileName: '{name}.{hash:8}.svg',
metadata: {
path: 'src/sprite.gen.ts',
// generate metadata
runtime: {
size: true,
viewBox: true
}
}
})
]
});
import svg from '@neodx/svg/vite';
import { defineConfig } from 'vite';
export default defineConfig({
plugins: [
svg({
root: 'assets',
// group icons by sprite name
group: true,
output: 'public/sprites',
// add hash to sprite file name
fileName: '{name}.{hash:8}.svg',
metadata: {
path: 'src/sprite.gen.ts',
// generate metadata
runtime: {
size: true,
viewBox: true
}
}
})
]
});
Step by step
Create a minimal working component
In the minimal approach, our component will accept only the name
(any string) prop and render the icon using the <use>
element.
import clsx from 'clsx';
import type { SVGProps } from 'react';
export interface IconProps extends SVGProps<SVGSVGElement> {
name: string;
}
export function Icon({ name, className, viewBox, ...props }: IconProps) {
return (
<svg
className={clsx(
'select-none fill-current inline-block text-inherit box-content w-[1em] h-[1em]',
className
)}
viewBox={viewBox}
// prevent icon from being focused when using keyboard navigation
focusable="false"
// hide icon from screen readers
aria-hidden
{...props}
>
<use href={`/sprite.svg#${name}`} />
</svg>
);
}
import clsx from 'clsx';
import type { SVGProps } from 'react';
export interface IconProps extends SVGProps<SVGSVGElement> {
name: string;
}
export function Icon({ name, className, viewBox, ...props }: IconProps) {
return (
<svg
className={clsx(
'select-none fill-current inline-block text-inherit box-content w-[1em] h-[1em]',
className
)}
viewBox={viewBox}
// prevent icon from being focused when using keyboard navigation
focusable="false"
// hide icon from screen readers
aria-hidden
{...props}
>
<use href={`/sprite.svg#${name}`} />
</svg>
);
}
import { Icon } from './icon';
export function SomeComponent() {
return (
<div>
<Icon name="icon-name" />
</div>
);
}
import { Icon } from './icon';
export function SomeComponent() {
return (
<div>
<Icon name="icon-name" />
</div>
);
}
It works, but we could see some issues:
name
prop is not type-safe, we can pass any stringviewBox
is missing, some icons may be rendered incorrectly/sprite.svg
is hardcoded, we can't use hashed file names or grouped spritesclassNames
containw-[1em] h-[1em]
styles, we can't use non-square icons (for example, logos)
Let's fix them!
Make name
prop type-safe
WARNING
In this guide I'll use spriteName/iconName
format to name icons, but it will be broken if you use /
in your icon or sprite name (for example, nested names), be careful.
As we faced in the metadata generation guide, we can use the generated SpritesMap
type, but, we should keep a single name
property for the DX and simplicity reasons.
Let's start implementing it:
Declare
IconName
typetsimport { type SpritesMap } from './sprite.gen'; // Icon name for a specific sprite export type IconName<Key extends keyof SpritesMap> = `${Key}/${SpritesMap[Key]}`;
import { type SpritesMap } from './sprite.gen'; // Icon name for a specific sprite export type IconName<Key extends keyof SpritesMap> = `${Key}/${SpritesMap[Key]}`;
Write a function to get and process icon metadata
tsimport { type SpritesMap, SPRITES_META } from './sprite.gen'; const getIconMeta = <Key extends keyof SpritesMap>(name: IconName<Key>) => { const [spriteName, iconName] = name.split('/') as [Key, SpritesMap[Key]]; const { filePath, items: { [iconName]: { viewBox } } } = SPRITES_META[spriteName]; return { filePath, iconName, viewBox }; };
import { type SpritesMap, SPRITES_META } from './sprite.gen'; const getIconMeta = <Key extends keyof SpritesMap>(name: IconName<Key>) => { const [spriteName, iconName] = name.split('/') as [Key, SpritesMap[Key]]; const { filePath, items: { [iconName]: { viewBox } } } = SPRITES_META[spriteName]; return { filePath, iconName, viewBox }; };
Update
Icon
componenttsximport clsx from 'clsx'; import type { SVGProps } from 'react'; import { SPRITES_META, type SpritesMap } from './sprite.gen'; export interface IconProps extends SVGProps<SVGSVGElement> { name: AnyIconName; } // All possible icon names export type AnyIconName = { [Key in keyof SpritesMap]: IconName<Key> }[keyof SpritesMap]; // Icon name for a specific sprite export type IconName<Key extends keyof SpritesMap> = `${Key}/${SpritesMap[Key]}`; export function Icon({ name /* ... */ }: IconProps) { const { viewBox, filePath, iconName, axis } = getIconMeta(name); return ( <svg viewBox={viewBox} // ... > <use href={`/sprites/${filePath}#${iconName}`} /> </svg> ); }
import clsx from 'clsx'; import type { SVGProps } from 'react'; import { SPRITES_META, type SpritesMap } from './sprite.gen'; export interface IconProps extends SVGProps<SVGSVGElement> { name: AnyIconName; } // All possible icon names export type AnyIconName = { [Key in keyof SpritesMap]: IconName<Key> }[keyof SpritesMap]; // Icon name for a specific sprite export type IconName<Key extends keyof SpritesMap> = `${Key}/${SpritesMap[Key]}`; export function Icon({ name /* ... */ }: IconProps) { const { viewBox, filePath, iconName, axis } = getIconMeta(name); return ( <svg viewBox={viewBox} // ... > <use href={`/sprites/${filePath}#${iconName}`} /> </svg> ); }
Detect icon major axis for correct scaling
How will SVG be scaled if source asset size is non-square? It will be forced to be square!
In the screenshot above, we can see that the right icon container is filled as a square, but the icon itself is not.
We're expecting the left icon behavior, let's fix it!
We already know width
and height
of the icon, so we can compare them and detect the major axis (or scale both axes if they are equal). I'll extract icon styles to the global @layer components
to use [data-axis*=]
selector and make Icon
component open for extension.
@layer components {
.icon {
/* reset styles and prevent icon from being selected */
@apply select-none fill-current inline-block text-inherit box-content;
}
.icon[data-axis*='x'] {
/* scale horizontally */
@apply w-[1em];
}
.icon[data-axis*='y'] {
/* scale vertically */
@apply h-[1em];
}
}
@layer components {
.icon {
/* reset styles and prevent icon from being selected */
@apply select-none fill-current inline-block text-inherit box-content;
}
.icon[data-axis*='x'] {
/* scale horizontally */
@apply w-[1em];
}
.icon[data-axis*='y'] {
/* scale vertically */
@apply h-[1em];
}
}
export function Icon({ name /* ... */ }: IconProps) {
const { viewBox, filePath, iconName, axis } = getIconMeta(name);
return (
<svg
className={clsx('icon', className)}
/**
* This prop is used by the "icon" class to set the icon's scaled size
* @see https://github.com/secundant/neodx/issues/92
*/
data-axis={axis}
// ...
/>
);
}
const getIconMeta = <Key extends keyof SpritesMap>(name: IconName<Key>) => {
const [spriteName, iconName] = name.split('/') as [Key, SpritesMap[Key]];
const {
filePath,
items: {
[iconName]: { viewBox, width, height }
}
} = SPRITES_META[spriteName];
const axis = width === height ? 'xy' : width > height ? 'x' : 'y';
return { filePath, iconName, viewBox, axis };
};
export function Icon({ name /* ... */ }: IconProps) {
const { viewBox, filePath, iconName, axis } = getIconMeta(name);
return (
<svg
className={clsx('icon', className)}
/**
* This prop is used by the "icon" class to set the icon's scaled size
* @see https://github.com/secundant/neodx/issues/92
*/
data-axis={axis}
// ...
/>
);
}
const getIconMeta = <Key extends keyof SpritesMap>(name: IconName<Key>) => {
const [spriteName, iconName] = name.split('/') as [Key, SpritesMap[Key]];
const {
filePath,
items: {
[iconName]: { viewBox, width, height }
}
} = SPRITES_META[spriteName];
const axis = width === height ? 'xy' : width > height ? 'x' : 'y';
return { filePath, iconName, viewBox, axis };
};