了解一些实用且富有创意的列表样式设置方法。
提到列表时,您会想到什么?最明显的例子就是购物清单,它是最简单的列表,其中的项没有特定顺序。但我们在网络上会以各种方式使用列表。某个场馆即将举办的一系列音乐会?很可能是列表。多步预订流程?很可能是列表。图片库?即使这样,也能被视为一系列带说明的图片。
在本文中,我们将深入探讨 Web 上可用的不同 HTML 列表类型以及何时使用它们,包括您可能不熟悉的一些属性。我们还将介绍一些有用且富有创意的方法,以便使用 CSS 为它们设置样式。
何时使用列表
当需要按语义对项进行分组时,应使用 HTML 列表元素。辅助技术(例如屏幕阅读器)会通知用户存在列表以及列表中的项数量。例如,如果您想在购物网站上展示一张商品网格,了解这些信息会非常有帮助。因此,使用列表元素可能是一个不错的选择。
列出类型
在标记方面,我们可以选择三种不同的列表元素:
- 无序列表
- 有序列表
- 说明列表
具体选择哪个取决于用例。
无序列表 (ul)
当列表中的项没有任何特定顺序时,无序列表元素 (<ul>
) 最有用。默认情况下,这将显示为项目符号列表。例如,购物清单的顺序无关紧要。
在 Web 上,导航菜单就是一个更常见的示例。构建菜单时,最好将 ul
封装在 nav
元素中,并使用标签标识菜单,以便辅助技术使用。我们还应在菜单中标识当前页面,可以使用 aria-current
属性来实现:
<nav aria-label="Main">
<ul>
<li>
<a href="/page-1" aria-current="page">Menu item 1</a>
</li>
<li>
<a href="/page-2">Menu item 2</a>
</li>
<li>
<a href="/page-2">Menu item 2</a>
</li>
…
</ul>
</nav>
这篇介绍菜单结构的文章列出了一些建议,可确保所有用户都能访问我们的导航菜单。
有序列表 (ol)
当项的顺序很重要时(例如多步流程),有序列表元素 (<ol>
) 是最佳选择。默认情况下,列表项会编号。例如,一组说明,其中的步骤必须按顺序完成。
<ol>
和 <ul>
元素只能包含 <li>
元素作为其直接子级。
说明列表 (dl)
说明列表包含术语(<dt>
元素)和说明(<dd>
)。每个术语可以有多个说明。可能的用例包括术语表,或者餐厅菜单。默认情况下,描述列表不会显示任何标记,但浏览器往往会缩进 <dd>
元素。
在 HTML 中,您可以使用 <div>
将术语及其随附的说明分组。这对于样式设置非常有用,我们稍后会介绍。
<!-- This is valid -->
<dl>
<dt>Term 1</dt>
<dd>This is the first description of the first term in the list</dd>
<dd>This is the second description of the first term in the list</dd>
<dt>Term 2</dt>
<dd>This is the description of the second term in the list</dd>
</dl>
<!-- This is also valid -->
<dl>
<div>
<dt>Term 1</dt>
<dd>This is the first description of the first term in the list</dd>
<dd>This is the second description of the first term in the list</dd>
</div>
<div>
<dt>Term 2</dt>
<dd>This is the description of the second term in the list</dd>
</div>
</dl>
简单列表样式
列表最简单的用法之一是在一段正文中使用。通常,这些简单列表不需要精心设计样式,但我们可能需要在一定程度上自定义有序或无序列表的标记,例如使用品牌颜色,或为项目符号使用自定义图片。我们可以使用 list-style
和 ::marker
伪元素执行很多操作!
::marker
除了为列表标记设置一些基本样式之外,我们还可以创建循环圆点。在这里,我们为 ::marker
伪元素的 content
值使用了三个不同的图片网址,这让购物清单示例看起来更像是手写的(而不是只使用一张图片):
::marker {
content: url("/marker-1.svg") ' ';
}
li:nth-child(3n)::marker {
content: url("/marker-2.svg") ' ';
}
li:nth-child(3n - 1)::marker {
content: url("/marker-3.svg") ' ';
}
自定义计数器
对于某些有序列表,我们可能希望使用计数器值,但要将另一个值附加到该值。我们可以将 list-item
计数器用作标记的 content
属性的值,并附加任何其他内容:
::marker {
content: counter(list-item) '🐈 ';
}
我们的计数器会自动递增 1,但我们也可以通过在列表项上设置 counter-increment
属性,允许它们递增其他值。例如,这将使计数器每次增加 3 个:
li {
counter-increment: list-item 3;
}
我们还可以深入探讨计数器的更多内容。CSS 列表、标记和计数器一文更详细地介绍了一些可能性。
::marker 样式的限制
有时,我们可能需要更好地控制标记的位置和样式。例如,您无法使用 Flexbox 或网格来定位标记,这在某些情况下可能会成为缺点,因为您可能需要其他对齐方式。::marker
公开了数量有限的 CSS 属性以用于设置样式。如果我们的设计需要基本样式以外的其他内容,我们最好使用其他伪元素。
为看起来不像列表的列表设置样式
有时,我们可能希望以与默认样式完全不同的方式为列表设置样式。导航菜单通常就是这种情况,例如,我们通常希望移除所有标记,并可能使用 flexbox 水平显示列表。常见做法是将 list-style
属性设为 none
。这意味着,DOM 中将无法再访问标记伪元素。
使用 ::before 创建自定义标记
在 ::marker
出现之前,设置 ::before
伪元素的样式是创建自定义列表标记的常用方法。不过,即使现在,我们也可以在需要时更灵活地设置视觉上复杂的列表样式。
与 ::marker
一样,我们可以使用 content
属性添加自己的自定义项目符号样式。与使用 ::marker
不同,我们需要进行一些手动定位,因为我们无法获得 list-style-position
提供的自动定位优势。不过,我们可以使用 Flexbox 相对轻松地对其进行定位,并且它确实提供了更多对齐方式。例如,我们可以交替标记的位置:
如果我们使用 ::before
元素为有序列表设置样式,可能还希望使用计数器添加数字标记。
li::before {
counter-increment: list-item;
content: counter(list-item);
}
使用 ::before
而非 ::marker
后,我们可以完全访问 CSS 属性以实现自定义样式,还可以使用动画和过渡效果(::marker
对此类效果的支持有限)。
列出属性
有序列表元素接受一些可选属性,这些属性可帮助我们在各种使用情形中发挥作用。
反向列表
如果我们有一个过去一年的十大专辑列表,则可能需要从 10 到 1 进行倒计数。我们可以为此使用自定义计数器,并以负数递增它们。或者,我们也可以直接在 HTML 中使用 reversed
属性。我认为,除非计数器完全用于呈现目的,否则通常使用 reversed
属性,而不是在 CSS 中负增计数器,这样更有语义意义。如果 CSS 未能加载,您仍然会在 HTML 中看到数字正确倒计时。此外,我们还需要考虑屏幕阅读器如何解读列表。
请参阅此演示,了解 2021 年十大专辑。如果计数器是纯粹使用 CSS 递增的,那么使用屏幕阅读器访问该网页的用户可能会认为数字是从 1 开始递增的,因此 10 实际上是 1。
您可以在演示中看到,通过使用 reversed
属性,我们的标记已经具有正确的值,而我们无需额外付出任何努力!但是,如果我们使用 ::before
伪元素创建自定义列表标记,则需要调整计数器。我们只需指示列表项计数器以负方向递增即可:
li::before {
counter-increment: list-item -1;
content: counter(list-item);
}
在 Firefox 中,这已经足够了,但在 Chrome 和 Safari 中,标记将从 0 倒计时到 -10。我们可以通过向列表中添加 start
属性来解决此问题。
拆分列表
借助 start
属性,我们可以指定列表应从哪个数字值开始。这在以下情况下非常有用:您想将列表拆分为多个组。
我们继续使用前面的“热门 10 张专辑”示例。也许我们实际上想按 10 个为一组的方式统计前 20 张专辑。在这两个组之间还有一些其他网页内容。
我们需要在 HTML 中创建两个单独的列表,但如何确保计数器的正确性?目前的标记方式是,这两个列表都会从 10 倒计到 1,这并不是我们想要的。不过,我们可以在 HTML 中指定 start
属性值。如果我们向第一个列表添加 start
值 20,标记将再次自动更新!
<ol reversed start="20">
<li>...</li>
<li>...</li>
<li>...</li>
</ol>
多列列表布局
正如您在之前的演示中看到的,多列布局有时非常适合我们的列表。通过设置列宽,我们可以确保列表会自动响应,仅在有足够空间时才会跨越两个或更多列。我们还可以设置列之间的间距,并为增添美感而添加带样式的column-rule(使用与 border
属性类似的简写形式):
ol {
columns: 25rem;
column-gap: 7rem;
column-rule: 4px dotted turquoise;
}
使用列时,列表项中有时会出现不美观的换行,这并不总是我们想要的效果。
我们可以对列表项使用 break-inside: avoid
来防止这些强制换行:
li {
break-inside: avoid;
}
自定义属性
CSS 自定义属性为列表样式设置提供了各种可能性。如果我们知道列表项的索引,则可以使用该索引来计算属性值。遗憾的是,目前无法仅通过 CSS 来确定元素的索引(至少无法以可用的方式确定)。计数器仅允许我们在 content
属性中使用其值,而不允许进行计算。
不过,我们可以在 HTML 的 style
属性中设置元素的索引,这可以使计算更可行,尤其是在使用模板语言时。以下示例展示了如何使用 Nunjucks 进行设置:
<ol style="--length: items|length">
</ol>
Splitting.js 是一个在客户端执行类似功能的库。
使用自定义属性值,我们可以通过多种方式显示列表中的进度。一种方法是使用进度条来显示一系列步骤。在此示例中,我们使用带有线性渐变的伪元素为每个项创建一个条状标签,以显示用户在列表中浏览了多少内容。
li::before {
--stop: calc(100% / var(--length) * var(--i));
--color1: deeppink;
--color2: pink;
content: '';
background: linear-gradient(to right, var(--color1) var(--stop), var(--color2) 0);
}
我们还可以使用 hsl()
颜色函数,随着列表的滚动调整色相。我们可以使用自定义属性计算 hue
值。
说明列表样式
如前所述,我们可以选择将术语及其定义封装在 dl
中的 div
中,以便获得更多样式选项。例如,我们可能希望以网格形式显示列表。如果在列表中设置 display: grid
,但未在每个组周围添加封装容器 div
,则表示我们的字词和说明会放置在不同的网格单元格中。有时,这很有用,如以下示例所示,显示带有说明的馅饼菜单。
我们可以在列表本身上定义网格,并确保术语和说明始终按列对齐,列宽由最长的术语决定。
另一方面,如果我们想以卡片样式将术语及其说明明确分组,封装容器 <div>
会非常有用。
资源
- 列表列表和 ::marker 简介
- 使用 ::marker 的自定义标记
- 带有计数器的 CSS 列表