前言
本文将介绍一种滑动显示效果,通过HTML和CSS实现。这种效果可以在网页中展示一组数字,并且在鼠标悬停或获得焦点时产生交互效果。我们将使用一个无序列表(ul)来容纳数字,并为每个数字创建一个列表项(li)。每个列表项包含一个数字(span),通过设置不同的样式和过渡效果,实现数字的滑动显示。
为了达到预期的效果,我们使用了一些CSS属性和伪类。通过设置元素的box-sizing属性为border-box,确保元素的宽度和高度包括内边距和边框。整个页面采用网格布局(display: grid),并通过place-items属性将内容居中显示。
在页面背景上,我们添加了一层透明的网格线条效果。这是通过两个线性渐变背景和一个遮罩(mask)实现的。遮罩的方向和角度以及线条的颜色可以通过修改变量来进行调整。
每个数字的样式定义在.digit类中。数字之间的间距通过padding属性控制,并且可以根据需要进行调整。数字在获得焦点或鼠标悬停时,通过设置伪类:hover和:focus-visible的样式,实现数字的动态效果。
代码解析
<section>
<p>滑动显示</p>
<ul class="code">
<li tabindex="0" class="digit">
<span>0</span>
</li>
<li tabindex="0" class="digit">
<span>3</span>
</li>
<li tabindex="0" class="digit">
<span>4</span>
</li>
<li tabindex="0" class="digit">
<span>8</span>
</li>
<li tabindex="0" class="digit">
<span>7</span>
</li>
<li tabindex="0" class="digit">
<span>2</span>
</li>
</ul>
</section>
这部分是HTML代码,定义了一个包含滑动显示效果的数字组合。使用<section>
元素标签包裹,表示这段内容是一个独立的节(section)。<p>
标签用于显示文本 “滑动显示”。接下来是一个无序列表(<ul class="code">
),用于容纳数字。每个数字都被包裹在一个列表项(<li tabindex="0" class="digit">
)中,通过tabindex="0"
属性使得这些列表项可以获得焦点。数字本身则被包裹在<span>
标签中。
* {
box-sizing: border-box;
}
body {
min-height: 100vh;
display: grid;
place-items: center;
font-family: "SF Pro Text", "SF Pro Icons", "AOS Icons", "Helvetica Neue", Helvetica, Arial, sans-serif, system-ui;
background: hsl(0 0% 0%);
gap: 2rem;
}
这部分是CSS代码,用于设置全局的样式。*
选择器指定了所有元素应使用border-box模型进行盒子大小计算。body
选择器定义了页面主体部分的样式,其中min-height: 100vh;
将页面高度设置为视口高度。display: grid;
和place-items: center;
则将页面内容居中显示。font-family
属性指定了字体的优先级和备选项,background
属性设置页面背景色为黑色,gap
属性定义了元素之间的间距。
body::before {
--line: hsl(0 0% 95% / 0.25);
content: "";
height: 100vh;
width: 100vw;
position: fixed;
background:
linear-gradient(90deg, var(--line) 1px, transparent 1px 10vmin) 0 -5vmin / 10vmin 10vmin,
linear-gradient(var(--line) 1px, transparent 1px 10vmin) 0 -5vmin / 10vmin 10vmin;
mask: linear-gradient(-15deg, transparent 30%, white);
top: 0;
z-index: -1;
}
这部分是CSS代码,用于创建页面背景中的网格线条效果。body::before
伪元素被用来添加内容,在页面主体之前显示。--line
是一种自定义CSS变量,用于指定线条的颜色和透明度。content: "";
表示伪元素没有实际内容,只是为了生成背景效果。height
和width
属性将伪元素的高度和宽度设置为100vh和100vw,使其与视口的尺寸相同。position: fixed;
将伪元素固定在视口的位置。background
属性使用两个线性渐变背景实现网格线条效果,具体的细节可以参考代码中的注释。mask
属性创建了一个遮罩效果,通过线性渐变设置透明度渐变,产生一种网格线消失的效果。top: 0;
将伪元素定位到页面顶部,z-index: -1;
将其置于最下层。
section {
display: grid;
gap: 4rem;
align-items: center;
justify-content: center;
}
section p {
margin: 0;
font-size: 2.25rem;
color: hsl(0 0% 40%);
text-align: center;
color: #d5d0d0;
background-clip: text;
}
这部分是CSS代码,用于设置包裹数字组合的<section>
元素的样式。display: grid;
将元素以网格布局显示。gap: 4rem;
设置网格项之间的间距为4rem。align-items: center;
和justify-content: center;
使得元素内容在水平和垂直方向上都居中对齐。section p
选择器定义了包裹数字组合的<p>
元素的样式,包括字体大小、颜色和文本居中对齐。
.code {
font-size: 3rem;
display: flex;
flex-wrap: nowrap;
color: hsl(0 0% 100%);
border-radius: 1rem;
background: hsl(0 0% 6%);
justify-content: center;
box-shadow: 0 1px hsl(0 0% 100% / 0.25) inset;
}
.code:hover {
cursor: grab;
}
这部分是CSS代码,用于设置数字组合的样式。.code
类定义了数字组合的样式。font-size: 3rem;
设置数字的字体大小为3rem。display: flex;
将数字组合以弹性盒子形式显示。flex-wrap: nowrap;
设置弹性盒子不换行,保持在同一行显示。color: hsl(0 0% 100%);
将数字的颜色设置为白色。border-radius: 1rem;
设置数字组合的边框圆角为1rem。background: hsl(0 0% 6%);
设置数字组合的背景色为黑色。justify-content: center;
将数字在水平方向上居中显示。box-shadow: 0 1px hsl(0 0% 100% / 0.25) inset;
添加一个内阴影效果,使得数字组合看起来有一定的立体感。.code:hover
伪类设置当鼠标悬停在数字组合上时,光标变为抓取样式。
.digit {
display: flex;
height: 100%;
padding: 5.5rem 1rem;
}
.digit:focus-visible {
outline-color: hsl(0 0% 50% / 0.25);
outline-offset: 1rem;
}
这部分是CSS代码,用于设置数字项(列表项)的样式。.digit
类定义了数字项的样式。display: flex;
将数字项以弹性盒子形式显示。height: 100%;
设置数字项的高度为100%。padding: 5.5rem 1rem;
设置数字项的内边距,垂直方向上为5.5rem,水平方向上为1rem。.digit:focus-visible
伪类设置当数字项获得焦点时,显示轮廓,并设置轮廓的颜色和偏移量。
.digit span {
scale: calc(var(--active, 0) + 0.5);
filter: blur(calc((1 - var(--active, 0)) * 1rem));
transition: scale calc(((1 - var(--active, 0)) + 0.2) * 1s), filter calc(((1 - var(--active, 0)) + 0.2) * 1s);
}
这部分是CSS代码,用于设置数字(<span>
元素)的样式。.digit span
选择器定义了数字的样式。scale
属性通过计算设置数字的缩放比例,缩放比例由变量--active
控制,初始值为0。filter
属性通过计算设置数字的模糊效果,模糊程度由变量--active
控制,初始值为0。transition
属性定义了数字在变化过程中的动画效果,包括缩放比例和模糊程度。
ul {
padding: 0;
margin: 0;
}
.digit:first-of-type {
padding-left: 5rem;
}
.digit:last-of-type {
padding-right: 5rem;
}
这部分是CSS代码,用于设置无序列表(ul
)的样式以及第一个和最后一个数字项的内边距。ul
选择器设置无序列表的内边距和外边距都为0,以消除默认样式。.digit:first-of-type
选择器设置第一个数字项的左侧内边距为5rem,即增加数字组合的左侧间距。.digit:last-of-type
选择器设置最后一个数字项的右侧内
完整代码
html 部分
<section>
<p>滑动显示</p>
<ul class="code">
<li tabindex="0" class="digit">
<span>0</span>
</li>
<li tabindex="0" class="digit">
<span>3</span>
</li>
<li tabindex="0" class="digit">
<span>4</span>
</li>
<li tabindex="0" class="digit">
<span>8</span>
</li>
<li tabindex="0" class="digit">
<span>7</span>
</li>
<li tabindex="0" class="digit">
<span>2</span>
</li>
</ul>
</section>
css 部分
* {
box-sizing: border-box;
}
body {
min-height: 100vh;
display: grid;
place-items: center;
font-family: "SF Pro Text", "SF Pro Icons", "AOS Icons", "Helvetica Neue", Helvetica, Arial, sans-serif, system-ui;
background: hsl(0 0% 0%);
gap: 2rem;
}
body::before {
--line: hsl(0 0% 95% / 0.25);
content: "";
height: 100vh;
width: 100vw;
position: fixed;
background:
linear-gradient(90deg, var(--line) 1px, transparent 1px 10vmin) 0 -5vmin / 10vmin 10vmin,
linear-gradient(var(--line) 1px, transparent 1px 10vmin) 0 -5vmin / 10vmin 10vmin;
mask: linear-gradient(-15deg, transparent 30%, white);
top: 0;
z-index: -1;
}
section {
display: grid;
gap: 4rem;
align-items: center;
justify-content: center;
}
section p {
margin: 0;
font-size: 2.25rem;
color: hsl(0 0% 40%);
text-align: center;
color: #d5d0d0;
background-clip: text;
}
.code {
font-size: 3rem;
display: flex;
flex-wrap: nowrap;
color: hsl(0 0% 100%);
border-radius: 1rem;
background: hsl(0 0% 6%);
justify-content: center;
box-shadow: 0 1px hsl(0 0% 100% / 0.25) inset;
}
.code:hover {
cursor: grab;
}
.digit {
display: flex;
height: 100%;
padding: 5.5rem 1rem;
}
.digit:focus-visible {
outline-color: hsl(0 0% 50% / 0.25);
outline-offset: 1rem;
}
.digit span {
scale: calc(var(--active, 0) + 0.5);
filter: blur(calc((1 - var(--active, 0)) * 1rem));
transition: scale calc(((1 - var(--active, 0)) + 0.2) * 1s), filter calc(((1 - var(--active, 0)) + 0.2) * 1s);
}
ul {
padding: 0;
margin: 0;
}
.digit:first-of-type {
padding-left: 5rem;
}
.digit:last-of-type {
padding-right: 5rem;
}
:root {
--lerp-0: 1; /* === sin(90deg) */
--lerp-1: calc(sin(50deg));
--lerp-2: calc(sin(45deg));
--lerp-3: calc(sin(35deg));
--lerp-4: calc(sin(25deg));
--lerp-5: calc(sin(15deg));
}
.digit:is(:hover, :focus-visible) {
--active: var(--lerp-0);
}
.digit:is(:hover, :focus-visible) + .digit,
.digit:has(+ .digit:is(:hover, :focus-visible)) {
--active: var(--lerp-1);
}
.digit:is(:hover, :focus-visible) + .digit + .digit,
.digit:has(+ .digit + .digit:is(:hover, :focus-visible)) {
--active: var(--lerp-2);
}
.digit:is(:hover, :focus-visible) + .digit + .digit + .digit,
.digit:has(+ .digit + .digit + .digit:is(:hover, :focus-visible)) {
--active: var(--lerp-3);
}
.digit:is(:hover, :focus-visible) + .digit + .digit + .digit + .digit,
.digit:has(+ .digit + .digit + .digit + .digit:is(:hover, :focus-visible)) {
--active: var(--lerp-4);
}
.digit:is(:hover, :focus-visible) + .digit + .digit + .digit + .digit + .digit,
.digit:has(+ .digit + .digit + .digit + .digit + .digit:is(:hover, :focus-visible)) {
--active: var(--lerp-5);
}