当前位置: 首页 > article >正文

CSS transition 小技巧!如何保留 hover 的状态

前言

通常情况下,hover 是无法保存状态的。鼠标移入触发额外样式,一旦移出就还原了

el:hover{
   color: red
}

这就意味着,如果需要保留 hover 的状态,可能就不得不借助 JS 了,比如下面是某某书院的首页排行榜效果。
在这里插入图片描述
这里的主要交互有以下几个

  1. 鼠标滑过触发选中态
  2. 鼠标移出列表后仍然保留上一次的选中态(重点)
  3. 默认列表的第一项为选中态

目前官网的实现也是通过 JS 实现的,事实上,仅仅通过 CSS 也是可以完全做到的,需要用到 transition 延时的一些小技巧。

一、鼠标滑过触发选中态
假设列表 HTML 是这样的

<ul class="list">
    <li class="item">
      <h3 class="title">将军,夫人喊你种田了</h3>
      <p class="sumary">只是在休息室里打了个盹儿,一睁眼,竟然穿成了古代目不识丁的乡下胖丫头。好吃懒做不说,还在村里横行霸道。十里八乡没人愿意娶她,好不容易买了个金龟婿,大婚之日竟让人逃了。恶霸老爹一怒之下去道上掳了个夫君给她。就是……爹你掳的是不是有点不太对呀?* 婚后的苏胖丫很忙。
 忙着改造恶霸爹爹与恶霸弟弟。忙着抢救貌美如花的神将夫君。忙着养育三个小小恶霸小豆丁。一不小心,将自己忙成了大燕最位高权重的一品女侯!</p>
    </li>
  <li class="item">
      <h3 class="title">被夺一切后她封神回归</h3>
      <p class="sumary">【甜爽燃,团宠,玄学】 司扶倾一睁眼,不仅被夺了气运,人人还让她滚出娱乐圈。重活一次,她只想咸鱼躺,谁知现在圈内人只知拉踩营销,没点真本事,不好好磨炼演技,这样下去还能行?怎么也得收拾收拾。司扶倾捏了捏手腕,动了。
后来,网上疯狂骂她不自量力倒贴郁曜,造谣她私生活不检点,而—— 国际天后:今天我能站在这里,多亏了倾倾 top1男顶流:离我妹妹远点@郁曜 就连国际运动会官方:恭喜司扶倾拿下第13枚个人金牌,等一个退圈 当天,全网瘫痪。· 
史书记载,胤皇年少成名,八方征战,平天下,安宇内,是大夏朝最年轻的帝王,他完美强大,心怀天下,却因病死于27岁,一生短暂,无妻无妾,无子无孙,是无数人的白月光男神。无人知晓,他再睁开眼,来到了1500年后。这一次,他看见了他遥想过的盛世大夏。· 不久后胤皇身份曝光,司扶倾得知偶像竟然就在身边,她敬佩万分,只想——
 司扶倾:努力奋斗,报效大夏!胤皇:以身相许 司扶倾:???我一心奋发上进你却想要我?· 全能颜巅女神×杀伐清贵帝王 从全网黑到封神顶流,顺便和男神1v1</p>
    </li>
  ...
</ul>

简单修饰一下

.list{
  list-style: none;
  margin: 0;
  padding: 0;
  width: 400px;
}
.item{
  position: relative;
  padding: 10px 10px 10px 34px;
  cursor: pointer;
  counter-increment: num;
}
.title::before{
  content: counter(num) ' ';
  width: 25px;
  line-height: 30px;
  text-align: center;
  color: #fff;
  position: absolute;
  font-size: 14px;
  font-family: fantasy;
  left: 4px;
  background: center/100% 100% 
url();
}
.title{
  margin: 0;
  padding: 8px 0;
  font-weight: normal;
}
.sumary{
  margin: 0;
  overflow: hidden;
  display: -webkit-box;
  -webkit-box-orient: vertical;
  -webkit-line-clamp: 2;
  color: #666;
  font-size: 14px;
  color: 20px;
  height: 0;
}

.item::before{
  content: '';
  position: absolute;
  inset: 0;
  border-radius: 8px;
  opacity: 0;
  z-index: -1;
  background: linear-gradient(270deg, rgb(241, 236, 249) 0%, rgba(241, 236, 249, 0) 100%);
}

效果如下
在这里插入图片描述

现在加上 hover 的效果

.item:hover .title{
  color: rebeccapurple;
}
.item:hover .sumary{
  height: 40px;
}
.item.item:hover::before{
  opacity: 1;
}

效果如下
在这里插入图片描述
很正常的 hover 效果,没什么特别的,那如何在移出后仍然保留最后的状态呢?

二、保留 hover 的状态
实现 hover 保留状态需要用到这样一个小技巧。

比如,给一个元素添加 hover 样式

el:hover{
  color: red
}

在这里插入图片描述

如果我们给这个元素加一个延时


el{
 transition-delay: 1s;
}

那么,在鼠标移入和移出时都会有延迟
在这里插入图片描述

接着,我们在 hover 的时候取消延时
在这里插入图片描述

el:hover{
  color: red;
  transition-delay: 0s;
}

那么,在鼠标移入的时候会迅速响应,移出的时候仍然会有延迟
在这里插入图片描述
到这里相信大家都明白了吧,如果把延时设置为足够大,比如

 el{
 transition-delay: 9999s;
}

在这里插入图片描述
这样鼠标在移出后,需要经历 9999s 后才会变为原状,也就相当于保留了 hover 状态。
原理就是这样
三、鼠标移出列表后仍然保留上一次的选中态
根据上面的原理,我们可以轻松的实现在 hover 后保留状态,如下

/* 默认 transition */
.item::before,
.item .sumary,
.item .title{
  transition: 0s 9999s;
}
/* 每一项hover */
.item:hover .title{
  color: rebeccapurple;
  transition: none;
}
.item:hover .sumary{
  height: 40px;
  transition: none;
}
.item.item:hover::before{
  opacity: 1;
  transition: none;
}

需要注意的是,由于是 transition,所以所有的状态变化都是需要支持过渡属性的,比如隐藏 sumary 这里用的是 height: 0 而不是 display:none,还有选中的背景色变化,由于 background-image 不支持过渡,所以换成了::before,然后单独用 opacity 控制等等一些细节,效果如下
在这里插入图片描述

这样在鼠标离开后,之前状态仍然是保留的。但是我们只需要保留上一次的,而不是所有的,如何处理呢?

这里需要换一种思路,可以这么做,鼠标在移入整个列表的时候就清除所有的状态,这样就只有当前 hover 的选项才会保留下来,有点类似于 JS 中的思维,先把所有的.current 都移除,再给当前项添加.current,实现如下

/* 清除所有hover */
.list:hover .title{
  transition: none;
  color: #333;
}
.list:hover .sumary{
  transition: none;
  height: 0;
}
.list:hover .item::before{
  transition: none;
  opacity: 0;
}

这样就实现了鼠标移出列表后仍然保留上一次的选中态的功能,有点像单选框的效果,只不过是 hover 触发的,效果如下
在这里插入图片描述

四、默认列表的第一项为选中态

下面来实现最后一个功能。

这个相对而言比较容易,需要用到:first-child 伪类,可以匹配到第一个元素。

不过需要考虑的是优先级的问题,这个是默认状态,权限应该是最低的,其他 hover 样式都应该可以覆盖它,所以可以放在最前面,如下

/* 初始状态(第一个选中) */
.item:first-child .sumary{
  height: 40px;
}
.item:first-child .title{
  color: rebeccapurple;
}
.item:first-child::before{
  opacity: 1;
}

/* 清除所有hover */

/* 每一项hover */

这样就完美实现了文章开头的效果
在这里插入图片描述
由于是 CSS 实现,多个列表也是完全复用的
在这里插入图片描述
完整代码请查看 保留 hover 的状态

五、总结一下

以上就是通过纯 CSS 实现保留鼠标滑过样式的全部技巧了,主要还是对 transition-delay 的灵活运用,下面总结一下

  1. 实现原理的利用 transition-delay,让 “还原” 的时间足够长,这样就实现了保留 hover 状态的效果了
  2. 单选效果可以在鼠标移入整个列表的时候就清除所有的状态,这样就只有当前 hover 的选项才会保留下来,有点类似于 JS 中的思维
  3. 需要注意所有属性必须是支持 transition 的,比如 display:none 就不支持 transition,需要用其他样式代替

当然,整个实现对于 CSS 以及选择器要求是相当高的,实际项目过程中可能并不如 JS 实现来的快,但是,CSS 能够实现的又何必动用 JS 呢?在我看来,JS 就应该回归本职,专心处理数据逻辑交互,视觉方面全部交给 CSS 就行了,只是现在 CSS 还不够强大,实现需要用到很多奇技淫巧,但是,CSS 现在已经在变得足够强大,比如:has 伪类,相信未来 CSS 会越来越美好。


http://www.kler.cn/a/4362.html

相关文章:

  • vue3+elementPlus之后台管理系统(从0到1)(day1)
  • windows11下 podman-desktop 复制插件文件 到 RabbitMQ 容器内,并启用
  • NSIS 创建一键安装程序
  • 从字符串使用看Golang和Rust对内存使用的区别
  • C# 并发和并行的区别--16
  • 2025 年前端开发学习路线图完整指南
  • 图解redis的AOF持久化
  • thinkphp+vue水果购物商城网站
  • 二维前缀和求子矩阵
  • VSQT 联编无转到槽选项--VS2019中使用QT建立信号和槽函数连接
  • Pytorch线性模型实现——up主:刘二大人《PyTorch深度学习实践》
  • Spring —— Spring简单的读取和存储对象Ⅰ
  • php使用yield处理大数据文件
  • Python中 join() 函数的使用方法
  • vue+springboot贫困山区儿童衣物捐赠网站系统
  • 基于Spring、Spring MVC、MyBatis的共享单车管理系统
  • Studio One没有声音怎么办 Studio One工程没有声音
  • Go底层原理:一起来唠唠GMP调度(一)
  • 【华为OD机试 2023最新 】 区块链文件转储系统(C++ 100%)
  • IO的类型(BIO、NIO、AIO)
  • GameFramework框架详解之 Config全局配置
  • Nacos配置中心优雅配置JSON数据格式
  • 第二类斯特林数
  • 多少个X 蓝桥杯模拟
  • dp-一和零
  • 『Linux从入门到精通』第 ⑧ 期 - 项目自动化构建工具——make/Makefile