核心概念
在 Tailwind 项目中添加自定义样式的最佳实践。
通常,使用框架时最大的挑战是弄清楚当框架无法处理您需要的东西时,您应该怎么做。
Tailwind 从头开始设计为可扩展和可自定义的,因此无论您构建什么,都不会感到在与框架作斗争。
本指南涵盖了诸如自定义设计令牌、在必要时如何打破这些约束、添加您自己的自定义 CSS 以及使用插件扩展框架等主题。
如果您想更改诸如调色板、间距比例、排版比例或断点之类的内容,请在 CSS 中使用 @theme
指令添加您的自定义设置
@theme { --font-display: "Satoshi", "sans-serif"; --breakpoint-3xl: 1920px; --color-avocado-100: oklch(0.99 0 0); --color-avocado-200: oklch(0.98 0.04 113.22); --color-avocado-300: oklch(0.94 0.11 115.03); --color-avocado-400: oklch(0.92 0.19 114.08); --color-avocado-500: oklch(0.84 0.18 117.33); --color-avocado-600: oklch(0.53 0.12 118.34); --ease-fluid: cubic-bezier(0.3, 0, 0, 1); --ease-snappy: cubic-bezier(0.2, 0, 0, 1); /* ... */}
在主题变量文档中了解有关自定义主题的更多信息。
虽然您通常可以使用一组受约束的设计令牌构建大部分精心设计的设计,但偶尔您需要打破这些约束才能使事情达到像素完美。
当您发现自己确实需要类似 top: 117px
的东西来使背景图像处于正确的位置时,请使用 Tailwind 的方括号表示法动态生成具有任何任意值的类
<div class="top-[117px]"> <!-- ... --></div>
这基本上就像内联样式,其主要优点是您可以将其与诸如 hover
之类的交互式修饰符和诸如 lg
之类的响应式修饰符结合使用
<div class="top-[117px] lg:top-[344px]"> <!-- ... --></div>
这适用于框架中的所有内容,包括背景颜色、字体大小、伪元素内容等
<div class="bg-[#bada55] text-[22px] before:content-['Festivus']"> <!-- ... --></div>
如果您引用 CSS 变量作为任意值,则可以使用自定义属性语法
<div class="fill-(--my-brand-color) ..."> <!-- ... --></div>
这只是 fill-[var(--my-brand-color)]
的简写,它会自动为您添加 var()
函数。
如果您需要使用 Tailwind 没有开箱即用的实用程序的 CSS 属性,您也可以使用方括号表示法来编写完全任意的 CSS
<div class="[mask-type:luminance]"> <!-- ... --></div>
这真的像内联样式,但同样具有可以使用修饰符的优点
<div class="[mask-type:luminance] hover:[mask-type:alpha]"> <!-- ... --></div>
这对于诸如 CSS 变量之类的内容也很有用,尤其是当它们需要在不同条件下更改时
<div class="[--scroll-offset:56px] lg:[--scroll-offset:44px]"> <!-- ... --></div>
任意变体就像任意值,但用于进行动态选择器修改,就像您可以对内置的伪类变体(如 hover:{utility}
或响应式变体(如 md:{utility}
)进行修改一样,但直接在 HTML 中使用方括号表示法。
<ul role="list"> {#each items as item} <li class="lg:[&:nth-child(-n+3)]:hover:underline">{item}</li> {/each}</ul>
在任意变体文档中了解更多信息。
当任意值需要包含空格时,请改用下划线 (_
),Tailwind 会在构建时自动将其转换为空格
<div class="grid grid-cols-[1fr_500px_2fr]"> <!-- ... --></div>
在下划线很常见但空格无效的情况下,Tailwind 会保留下划线而不是将其转换为空格,例如在 URL 中
<div class="bg-[url('/what_a_rush.png')]"> <!-- ... --></div>
在极少数情况下,您实际上需要使用下划线,但由于空格也有效,因此它是不明确的,请使用反斜杠转义下划线,Tailwind 不会将其转换为空格
<div class="before:content-['hello\_world']"> <!-- ... --></div>
如果您使用的是类似于 JSX 的东西,其中反斜杠从渲染的 HTML 中删除,请使用 String.raw(),以便反斜杠不被视为 JavaScript 转义字符
<div className={String.raw`before:content-['hello\_world']`}> <!-- ... --></div>
Tailwind 中的许多实用程序共享一个公共命名空间,但映射到不同的 CSS 属性。例如,text-lg
和 text-black
都共享 text-
命名空间,但一个是用于 font-size
,另一个是用于 color
。
使用任意值时,Tailwind 通常可以根据您传入的值自动处理这种歧义
<!-- Will generate a font-size utility --><div class="text-[22px]">...</div><!-- Will generate a color utility --><div class="text-[#bada55]">...</div>
但有时它确实是不明确的,例如在使用 CSS 变量时
<div class="text-(--my-var)">...</div>
在这些情况下,您可以通过在值之前添加 CSS 数据类型 来“提示”Tailwind 底层类型
<!-- Will generate a font-size utility --><div class="text-(length:--my-var)">...</div><!-- Will generate a color utility --><div class="text-(color:--my-var)">...</div>
虽然 Tailwind 旨在处理您的大部分样式需求,但在您需要时,没有任何东西可以阻止您仅编写纯 CSS
@import "tailwindcss/";.my-custom-style { /* ... */}
如果您只想为页面设置一些默认值(如文本颜色、背景颜色或字体系列),最简单的选择是将一些类添加到 html
或 body
元素
<!doctype html><html lang="en" class="bg-gray-100 font-serif text-gray-900"> <!-- ... --></html>
这会将您的基本样式决策保留在您的标记中,与所有其他样式一起,而不是将它们隐藏在单独的文件中。
如果您想为特定的 HTML 元素添加自己的默认基本样式,请使用 @layer
指令将这些样式添加到 Tailwind 的 base
层
@layer base { h1 { font-size: var(--text-2xl); } h2 { font-size: var(--text-xl); }}
对于您要添加到项目中并且仍然希望能够使用实用程序类覆盖的任何更复杂的类,请使用 components
层。
传统上,这些将是类似于 card
、btn
、badge
的类 — 这类东西。
@layer components { .card { background-color: var(--color-white); border-radius: var(--rounded-lg); padding: var(--spacing-6); box-shadow: var(--shadow-xl); }}
通过在 components
层中定义组件类,您仍然可以在必要时使用实用程序类来覆盖它们
<!-- Will look like a card, but with square corners --><div class="card rounded-none"> <!-- ... --></div>
使用 Tailwind,您可能不需要像您想象的那样经常使用这些类型的类。请阅读我们关于管理重复项的指南,了解我们的建议。
components
层也是放置您正在使用的任何第三方组件的自定义样式的好地方
@layer components { .select2-dropdown { /* ... */ }}
除了使用 Tailwind 随附的实用程序外,您还可以添加自己的自定义实用程序。当您希望在项目中使用 Tailwind 没有开箱即用的实用程序的 CSS 功能时,这会很有用。
使用 @utility
指令将自定义实用程序添加到您的项目中
@utility content-auto { content-visibility: auto;}
您现在可以在 HTML 中使用此实用程序
<div class="content-auto"> <!-- ... --></div>
它还可以与诸如 hover
、focus
和 lg
之类的变体一起使用
<div class="hover:content-auto"> <!-- ... --></div>
自定义实用程序会自动与框架中的所有内置实用程序一起插入到 utilities
层。
如果您的自定义实用程序比单个类名更复杂,请使用嵌套来定义该实用程序
@utility scrollbar-hidden { &::-webkit-scrollbar { display: none; }}
除了使用 @utility
指令注册简单的实用程序外,您还可以注册接受参数的功能实用程序
@utility tab-* { tab-size: --value(--tab-size-*);}
特殊的 --value()
函数用于解析实用程序值。
使用 --value(--theme-key-*)
语法根据一组主题键解析实用程序值
@theme { --tab-size-2: 2; --tab-size-4: 4; --tab-size-github: 8;}@utility tab-* { tab-size: --value(--tab-size-*);}
这将匹配诸如 tab-1
、tab-4
和 tab-github
之类的实用程序。
要将值解析为裸值,请使用 --value({type})
语法,其中 {type}
是您要验证裸值的数据类型
@utility tab-* { tab-size: --value(integer);}
这将匹配诸如 tab-1
和 tab-76
之类的实用程序。
要支持任意值,请使用 --value([{type}])
语法(请注意方括号)来告诉 Tailwind 支持哪些类型作为任意值
@utility tab-* { tab-size: --value([integer]);}
这将匹配类似 tab-[1]
和 tab-[76]
的实用工具。 如果你想支持任何数据类型,你可以使用 --value([*])
。
--value()
函数的所有三种形式都可以在规则中作为多个声明使用,任何无法解析的声明将在输出中被省略
@theme { --tab-size-github: 8;}@utility tab-* { tab-size: --value([integer]); tab-size: --value(integer); tab-size: --value(--tab-size-*);}
这使得在必要时可以针对每种情况区别对待值,例如将裸整数转换为百分比
@utility opacity-* { opacity: --value([percentage]); opacity: calc(--value(integer) * 1%); opacity: --value(--opacity-*);}
--value()
函数还可以接受多个参数,如果你不需要在不同情况下区别对待返回值,它可以从左到右解析它们
@theme { --tab-size-github: 8;}@utility tab-* { tab-size: --value(--tab-size-*, integer, [integer]);}@utility opacity-* { opacity: calc(--value(integer) * 1%); opacity: --value(--opacity-*, [percentage]);}
为了支持负值,将单独的正值和负值实用工具注册到单独的声明中
@utility inset-* { inset: calc(--var(--spacing) * --value([percentage], [length]));}@utility -inset-* { inset: calc(--var(--spacing) * --value([percentage], [length]) * -1);}
修饰符使用 --modifier()
函数处理,该函数的工作方式与 --value()
函数完全相同,但作用于修饰符(如果存在)
@utility text-* { font-size: --value(--font-size-*, [length]); line-height: --modifier(--line-height-*, [length], [*]);}
如果修饰符不存在,任何依赖于修饰符的声明都不会包含在输出中。
为了处理分数,我们依赖 CSS 的 ratio
数据类型。如果将其与 --value()
一起使用,则表示 Tailwind 将该值和修饰符视为单个值
@utility aspect-* { aspect-ratio: --value(--aspect-ratio-*, ratio, [ratio]);}
这将匹配类似 aspect-square
、aspect-3/4
和 aspect-[7/9]
的实用工具。
使用 @variant
指令在自定义 CSS 中应用 Tailwind 变体
.my-element { background: white; @variant dark { background: black; }}
如果你需要同时应用多个变体,请使用嵌套
@variant dark { @variant hover { background: black; }}