【前端不得不会的各种特效】01.滑动显示效果的数字选择器代码实现
  LyJBRSvfEdxZ 2023年11月02日 21 0

前言

本文将介绍一种滑动显示效果,通过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: "";表示伪元素没有实际内容,只是为了生成背景效果。heightwidth属性将伪元素的高度和宽度设置为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);
}

GIF效果展示

在这里插入图片描述

【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 2023年11月08日 0

暂无评论

推荐阅读
LyJBRSvfEdxZ