专栏名称: 全栈修仙之路
专注分享 TS、Vue3、前端架构和源码解析等技术干货。
目录
相关文章推荐
好地讯江苏  ·  苏州狮山挂地1宗,楼面起价15586元/㎡, ... ·  昨天  
度房苏州  ·  苏州北墅居新作,首亮相便圈粉无数! ·  2 天前  
融媒吴江  ·  大反转!吴江直冲26℃! ·  3 天前  
51好读  ›  专栏  ›  全栈修仙之路

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

全栈修仙之路  · 公众号  ·  · 2023-01-29 08:00

正文

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

el:hover{
  color: red
}

这就意味着,如果需要保留 hover 的状态,可能就不得不借助 JS 了, 比如下面是 某某书院 [1] 的首页排行榜效果

Kapture 2022-09-17 at 11.44.48

这里的主要交互有以下几个

  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年后。 这一次,他看见了他遥想过的盛世大夏。 · 不久后胤皇身份曝光,司扶倾得知偶像竟然就在身边,她敬佩万分,只想—— 司扶倾:努力奋斗,报效大夏! 胤皇:以身相许 司扶倾:??? 我一心奋发上进你却想要我? · 全能颜巅女神×杀伐清贵帝王 从全网黑到封神顶流,顺便和男神1v1p>
    li>
  ...
ul>

简单修饰一下

.list{
  list-style: none;
  margin0;
  padding0;
  width400px;
}
.item{
  position: relative;
  padding10px 10px 10px 34px;
  cursor: pointer;
  counter-increment: num;
}
.title::before{
  contentcounter(num) ' ';
  width25px;
  line-height30px;
  text-align: center;
  color#fff;
  position: absolute;
  font-size14px;
  font-family: fantasy;
  left4px;
  background: center/100% 100% url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAADQAAAA8CAMAAAAe9Wm0AAAAbFBMVEUAAACaov2SnPKXnvaXofiyuvuCjPOgqvi4vfiBjPSyuf6Ml/a0u/21vPtte/F/i/WTnfWNmPiyuv6Ik/Zue/GRnPinr/Wfp/Gyuv6Ml/ebo/KTnfOutfqEj/W3vvpue/G/xfd1gvN8iPRodfBFzp+BAAAAF3RSTlMAECA7U2BgdIiTn6Wsvr+/v9Df39/s/o6+GugAAAGPSURBVHjazdZhk4IgEAZgpS6z8iotPEBF8P//x1tJrTxQlrm5uff7MxubLBtF9sQfp+ywjRCJk+zLhPq6SUwuxokhp4/YT1SUVtWam4nSZObImyA2YXHZ022cYsirOzwYoYti7nJzuMOqeHPNvke5jxgP2DRNNqLSMzQQtXjUtv8cBTWiwqOqohcsKikt8ajkHI+4EGjEBWNnNGJ1jUe1Vp9YJLSSeKRkh0eyC0BdEJIBZ5J/h1TA/6SUOgcgjUdaT6gJQA0CjV85jFrq/ZU/UYtA7HFzG5iA1PcOMnbs0b7tU/mcCAhjO/NE52A8jmXKMHYdnukTzE141vhyGWaSTq97ksMQhOnEF8qYFJvXneDO++lU82VznG0giYDppBVzTTvIffdjZSGFVjA2JHeVuRDbopOC6TrJ7GUSx0a1vYGR8u1kQ5krce5hJAUjlRbuRtuyu8GFVopxZ6OtxW5wObU2P3Ewaby+jya6R7UYO+C3xpICbk3NFhptTToaZ6OtzS/WGm39iUdkmdFFv5pvdNPREWjicTcAAAAASUVORK5CYII=);
}
.title{
  margin0;
  padding8px 0;
  font-weight: normal;
}
.sumary{
  margin0;
  overflow: hidden;
  display: -webkit-box;
  -webkit-box-orient: vertical;
  -webkit-line-clamp2;
  color#666;
  font-size14px;
  color20px;
  height0;
}

.item::before{
  content'';
  position: absolute;
  inset0;
  border-radius8px;
  opacity0;
  z-index: -1;
  backgroundlinear-gradient(270deg, rgb(2412362490%rgba(2412362490100%);
}

效果如下

image-20220917131014700

现在加上 hover 的效果






    
.item:hover .title{
  color: rebeccapurple;
}
.item:hover .sumary{
  height40px;
}
.item.item:hover::before{
  opacity1;
}

效果如下

Kapture 2022-09-17 at 13.12.45

很正常的 hover 效果,没什么特别的,那如何在移出后仍然保留最后的状态呢?接着往下看

二、保留 hover 的状态

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

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

el:hover{
  color: red
}
Kapture 2022-09-17 at 13.18.58

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

el{
  transition-delay1s;
}

那么,在鼠标移入和移出时都会有延迟

Kapture 2022-09-17 at 13.20.43

接着,我们在 hover 的时候取消延时

el:hover{
  color: red;
  transition-delay0s;
}

那么,在鼠标移入的时候会迅速响应,移出的时候仍然会有延迟

Kapture 2022-09-17 at 13.24.09

到这里相信大家都明白了吧,如果把延时设置为足够大,比如

el{
  transition-delay9999s;
}

这样鼠标在移出后,需要经历 9999s 后才会变为原状,也就相当于保留了 hover 状态

Kapture 2022-09-17 at 13.28.42

原理就是这样,接下来看看实际应用吧

三、鼠标移出列表后仍然保留上一次的选中态

根据上面的原理,我们可以轻松的实现在 hover 后保留状态,如下

/* 默认 transition */
.item::before,
.item .sumary,
.item .title{
  transition0s 9999s;
}
/* 每一项hover */
.item:hover .title{
  color: rebeccapurple;
  transition: none;
}
.item:hover .sumary{
  height40px;
  transition: none;
}
.item.item:hover::before{
  opacity1;
  transition: none;
}

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

Kapture 2022-09-17 at 13.42.21

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

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

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

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


Kapture 2022-09-17 at 13.54.53







请到「今天看啥」查看全文