@recastui/tailwind
Version:
Recastui Design System Tailwindcss plugin
231 lines (218 loc) • 8.96 kB
JavaScript
// Inspired by code from Ross Bratton's windy-radix-palette - https://github.com/brattonross/windy-radix-palette
const defaultTheme = require('tailwindcss/defaultTheme');
const plugin = require('tailwindcss/plugin');
const defaultColors = require('./colors');
const typographyColors = ['gray', 'mauve', 'slate', 'sage', 'olive', 'sand'];
const recastuiPlugin = plugin.withOptions(
({ colors = defaultColors, rootSelector = ':root' } = {}) => {
let themeColors = {};
let darkThemeColors = {};
for (const color in colors) {
for (const [key, value] of Object.entries(colors[color])) {
if (color.includes('Dark')) {
const colorName = color.replace('Dark', '');
darkThemeColors[`--${colorName}${key}`] = value;
} else {
themeColors[`--${color}${key}`] = value;
}
}
}
return ({ addBase, addUtilities, config, theme }) => {
const [darkMode, className = '.dark'] = [].concat(config('darkMode', 'media'));
if (darkMode === 'class') {
addBase({
[rootSelector]: themeColors,
[className]: darkThemeColors,
});
} else {
addBase({
[rootSelector]: themeColors,
'@media (prefers-color-scheme: dark)': {
[rootSelector]: darkThemeColors,
},
});
}
addBase({
html: {
'-webkit-font-smoothing': 'antialiased',
fontFeatureSettings: 'ss01',
},
body: {
['background-color']: theme('colors.gray.1'),
color: theme('colors.gray.12'),
},
'h1,h2,h3,h4,h5,h6,th': {
fontFamily: theme('fontFamily.display'),
fontWeight: theme('fontWeight.semibold'),
},
// Form elements updates
// input type dates update calendar icon
'::-webkit-calendar-picker-indicator': {
background:
'url() center/80% no-repeat',
},
// input type dates update calendar icon dark theme
'.dark ::-webkit-calendar-picker-indicator': {
background:
'url() center/80% no-repeat',
},
// Link type external
'a[data-external="true"]:after': {
display: 'inline-block',
width: '24px',
height: '24px',
verticalAlign: 'middle',
content: '',
background:
'url() center/80% no-repeat',
},
// Link type external dark theme
'.dark a[data-external="true"]:after': {
background:
'url() center/80% no-repeat',
},
});
addUtilities({
'.visually-hidden': {
border: 0,
clip: 'rect(0px, 0px, 0px, 0px)',
height: '1px',
width: '1px',
margin: '-1px',
padding: '0px',
overflow: 'hidden',
whiteSpace: 'nowrap',
position: 'absolute',
},
});
};
},
({ colors = defaultColors } = {}) => {
const themeColors = {};
for (const [colorName, colorObj] of Object.entries(colors)) {
if (colorName.includes('Dark')) {
continue;
}
const themeColor = {};
for (const key of Object.keys(colorObj)) {
themeColor[key] = `var(--${colorName}${key})`;
}
themeColors[colorName] = themeColor;
}
return {
theme: {
fontSize: {
xs: ['0.75rem', { lineHeight: '1rem' }],
sm: ['0.875rem', { lineHeight: '1.5rem' }],
base: ['1rem', { lineHeight: '1.5rem' }],
lg: ['1.125rem', { lineHeight: '1.75rem' }],
xl: ['1.25rem', { lineHeight: '2rem' }],
'2xl': ['1.5rem', { lineHeight: '2.5rem' }],
'3xl': ['2rem', { lineHeight: '2.5rem' }],
'4xl': ['2.5rem', { lineHeight: '3rem' }],
'5xl': ['3rem', { lineHeight: '3.5rem' }],
'6xl': ['3.75rem', { lineHeight: '1' }],
'7xl': ['4.5rem', { lineHeight: '1' }],
'8xl': ['6rem', { lineHeight: '1' }],
'9xl': ['8rem', { lineHeight: '1' }],
},
colors: {
inherit: 'inherit',
current: 'currentColor',
transparent: 'transparent',
black: '#000000',
white: '#ffffff',
...themeColors,
},
extend: {
fontFamily: {
sans: ['Lato', ...defaultTheme.fontFamily.sans],
display: ['Poppins', ...defaultTheme.fontFamily.sans],
},
maxWidth: {
'8xl': '88rem',
},
ringColor: ({ theme }) => ({
DEFAULT: theme('colors.gray.10'),
...theme('colors'),
}),
ringOpacity: ({ theme }) => ({
DEFAULT: '0',
...theme('opacity'),
}),
ringWidth: {
DEFAULT: '2px',
},
ringOffsetWidth: {
DEFAULT: '2px',
},
spacing: {
4.5: '1.125rem',
},
boxShadow: {
sm: '0 1px 2px 0 var(--slate6)',
DEFAULT: '0 1px 3px 0 var(--slate6), 0 1px 2px -1px var(--slate6)',
md: '0 4px 6px -1px var(--slate6), 0 2px 4px -2px var(--slate6)',
lg: '0 10px 15px -3px var(--slate6), 0 4px 6px -4px var(--slate6)',
xl: '0 20px 25px -5px var(--slate6), 0 8px 10px -6px var(--slate6)',
'2xl': '0 25px 50px -12px var(--slate8)',
inner: 'inset 0 2px 4px 0 var(--slate8)',
none: 'none',
},
lineClamp: {
7: '7',
8: '8',
9: '9',
10: '10',
},
typography: ({ theme }) => {
const themes = {};
for (const color of typographyColors) {
themes[color] = {
css: {
'--tw-prose-body': theme(`colors.${color}[12]`),
'--tw-prose-headings': theme(`colors.${color}[12]`),
'--tw-prose-lead': theme(`colors.${color}[11]`),
'--tw-prose-links': theme(`colors.${color}[12]`),
'--tw-prose-bold': theme(`colors.${color}[12]`),
'--tw-prose-counters': theme(`colors.${color}[11]`),
'--tw-prose-bullets': theme(`colors.${color}[7]`),
'--tw-prose-hr': theme(`colors.${color}[6]`),
'--tw-prose-quotes': theme(`colors.${color}[12]`),
'--tw-prose-quote-borders': theme(`colors.${color}[6]`),
'--tw-prose-captions': theme(`colors.${color}[11]`),
'--tw-prose-code': theme(`colors.${color}[12]`),
'--tw-prose-pre-code': theme(`colors.${color}[12]`),
'--tw-prose-pre-bg': theme(`colors.${color}[3]`),
'--tw-prose-th-borders': theme(`colors.${color}[7]`),
'--tw-prose-td-borders': theme(`colors.${color}[6]`),
'--tw-prose-invert-body': theme(`colors.${color}[12]`),
'--tw-prose-invert-headings': theme(`colors.${color}[12]`),
'--tw-prose-invert-lead': theme(`colors.${color}[11]`),
'--tw-prose-invert-links': theme(`colors.${color}[12]`),
'--tw-prose-invert-bold': theme(`colors.${color}[12]`),
'--tw-prose-invert-counters': theme(`colors.${color}[11]`),
'--tw-prose-invert-bullets': theme(`colors.${color}[7]`),
'--tw-prose-invert-hr': theme(`colors.${color}[6]`),
'--tw-prose-invert-quotes': theme(`colors.${color}[12]`),
'--tw-prose-invert-quote-borders': theme(`colors.${color}[6]`),
'--tw-prose-invert-captions': theme(`colors.${color}[11]`),
'--tw-prose-invert-code': theme(`colors.${color}[12]`),
'--tw-prose-invert-pre-code': theme(`colors.${color}[12]`),
'--tw-prose-invert-pre-bg': theme(`colors.${color}[3]`),
'--tw-prose-invert-th-borders': theme(`colors.${color}[7]`),
'--tw-prose-invert-td-borders': theme(`colors.${color}[6]`),
},
};
}
return themes;
},
},
},
variants: {
lineClamp: ['responsive', 'hover', 'focus'],
},
};
},
);
module.exports = recastuiPlugin;