微信小程序纯CSS实现无限弹幕滚动效果


Posted in HTML / CSS onSeptember 23, 2022

啥也不说,先上效果图

微信小程序纯CSS实现无限弹幕滚动效果

实现背景

产品: 我们需要一个弹幕滚动效果,类似于微博热搜那种实时评论,但是我们的数据是固定的20条,你让他一直循环滚动

好的(os:要求真多)

实现思路

你不是说要微博热搜弹幕那种效果吗?这好办啊,马上去微博看热搜(又看到那么多塌房的的明星),看了几十分钟,表面上是多研究研究,实则上班期间光明正大的摸鱼,感觉真好!!

  • 先看看微博的效果

分析一下,有以下几个功能点

  • 弹幕实时更新
  • 从右到左匀速运动
  • 点击某一个弹幕跳转到详情,返回时继续移动

一共三个功能点,对比我的需求只有一个功能点相差,我只需要把弹幕实时更新改成弹幕循环滚动。

如何实现循环的感觉呢?

如果要实现一个循环的感觉就必须要把数组拷贝一份,把他们一排排列,然后这样子让他去滚动,当滚动结尾的时候,展示第二个拷贝数组的开头。animate动画帧的编写就改变transform:translateX的值,从0单个数组所渲染的dom长度,第一次的动画执行完成后,刚好是第二个拷贝数组的开头,然后又让动画帧执行从0单个数组所渲染的dom长度,由于每一次动画执行完刚好是第二个拷贝数组的开头,所以执行第二次动画时不会有闪动的感觉

如何实现动画执行完又继续执行呢?

设置animation-iteration-count: infinite;,循环执行,这样子就会无限的从0单个数组所渲染的dom长度,无限的循环套娃,就会有循环滚动的感觉了,大概图如下

微信小程序纯CSS实现无限弹幕滚动效果

代码实现

  • 先拷贝一份数组让他们一排排列,注意最外层的容器需要设置overflow: hidden
<view class="z-bullet-test">
    <view class="z-list">
        <view class="z-container">
            <view class="z-single" wx:for="{{10}}" wx:key="index">
                {{index}}
            </view>
        </view>
        <view class="z-container">
            <view class="z-single" wx:for="{{10}}" wx:key="index">
                {{index}}
            </view>
        </view>
    </view>
</view>
.z-bullet-test {
  width: 690rpx;
  height: 88rpx;
  display: flex;
  align-items: center;
  background-color: saddlebrown;
  margin: 0 auto;
  margin-top: 40rpx;
  overflow: hidden;
}
.z-container {
  display: flex;
  flex-wrap: nowrap;
}

.z-single {
  padding: 10rpx 40rpx;
  color: white;
  background-color: cadetblue;
  margin-right: 20rpx;
}

.z-list {
  display: flex;
  flex-wrap: nowrap;
}

效果如下

微信小程序纯CSS实现无限弹幕滚动效果

  • 让外层的容器滚动起来,只需要给z-list添加动画即可
.z-list {
  display: flex;
  flex-wrap: nowrap;
  animation: rowScrollTest 2s linear infinite;
}

@keyframes rowScrollTest {
  0% {
    /* 为0时,ios会闪动 */
    transform: translateX(1);
  }

  100% {
    transform: translateX(-433.85px); // 单个数组所渲染的dom长度
  }
}

效果如下

微信小程序纯CSS实现无限弹幕滚动效果

通过以下几句代码,并且没有使用任何的js代码就把无限循环滚动实现好了,是不是觉得事情到这儿结束了呢???

并没有

如何动态计算单个数组所渲染的dom长度

上面实现的代码中,对于单个数组所渲染的dom长度是通过开发者工具找到dom长度,并在代码中写死的,但是在开发中你并不能确定这个长度到底为多少,这个时候我们需要去动态获取这个dom长度

这个时候就有同学说了,那还不简单直接通过js去获取dom长度不就可以了吗?

想法很不错,先给你点个赞,但是重点来了!!!注意看,通过js获取dom长度可以得到一个长度变量,但是你如何动态赋值给@keyframes动画帧呢??

@keyframes rowScrollTest {
  0% {
    /* 为0时,ios会闪动 */
    transform: translateX(1);
  }

  100% {
    transform: translateX(domWidth); // 单个数组所渲染的dom长度
  }
}

思考一下?这个domWidth应该如何去动态的赋值呢?

动态赋值动画帧中的css参数

遇到这个问题时,感觉有点奇怪,这个动态的赋值一般是给css属性,如width,height,还没有给动画帧中的css属性动态赋值过,

于是照着给css属性动态赋值的方法,我把@keyframes动画帧函数先写到了行内style上,并把css中的@keyframes动画帧函数删除

<view class="z-list" style="@keyframes rowScrollTest {0% { transform: translateX(1); } 100% { transform: translateX(-433.85px); }}">
    <!-- 数组内容 -->
</view>

满心欢喜的打开开发者工具,一看~~ 嘿! 动都不动了

原因是因为style属性只能识别出css的属性名,我在这里写的动画帧函数,在它看来就是一串没有用的字符串,所以就根本不会编译成css的动画帧函数,就肯定不会有作用了

如何解决呢?

传统的给css属性动态赋值的方法走不通,应该怎么办呢?99%解决问题的办法:点开goole然后再输入栏输入这个问题,按下enter键,然后就去疯狂的寻找就可以了。

经过一番查找,嘿还真找到了一个东西,css自定义属性(变量),这是个啥呢?

MDN解释:自定义属性(有时候也被称作CSS 变量或者级联变量)是由 CSS 作者定义的,它包含的值可以在整个文档中重复使用。由自定义属性标记设定值(比如: --main-color: black; ),由 var() 函数来获取值(比如: color: var(--main-color);)复杂的网站都会有大量的 CSS 代码,通常也会有许多重复的值。

那我们是不是可以动态赋值给这个变量,并在css代码的@keyframes动画帧函数中去使用这个变量的值,当作translateX的值呢,立马试一下

<view class="z-list" style="--domWidth--:-433.85px;">
    <!-- 数组内容 -->
</view>
@keyframes rowScrollTest {
  0% {
    /* 为0时,ios会闪动 */
    transform: translateX(1);
  }

  100% {
    transform: translateX(var(--domWidth--)); // 单个数组所渲染的dom长度
  }
}

激动的打开开发者工具,见证奇迹的时刻到来了!!!!!!!!! 它动了,它在疯狂的滚动着

接下来是不是只需要将-433.85改成js获取的单个数组所渲染的dom长度就可以了呢?,于是我满心欢喜的去写js代码了,先定义一个变量domWidth: 0,然后再去onLoad中去获取z-container的长度并赋值给domWidth,动态的获取单个数组所渲染的dom长度并赋值给@keyframes动画帧函数,就大功告成了!!(其实并没有,不要看到这儿就不看了!

<view class="z-list" style="--domWidth--:-{{domWidth}}px;">
    <!-- 数组内容 -->
</view>
data: {
    domWidth: 0
},
onLoad () {
    wx.createSelectorQuery()
      .selectAll(".z-container")
      .boundingClientRect((res) => {
        this.setData({ domWidth: res[0].width });
      })
      .exec();
}

这个时候我轻松的点开开发者工具,心里想着这个问题被我轻松解决掉了,还剩很多开发时间,可以去摸鱼了,但是它不动了,一点都不动了!

为什么呢? 我立马点开控制台,找到了这个变量的值,发现他是有值的,说明问题不出在js获取长度,

我仔细一想瞬间明白了 domWidth的值默认为0,animation属性会在被渲染出来时立即执行,这个时候的domWidth还为0,动画帧函数就相当于是从0到0,肯定就不会滚动了,你可以把domWidth随便设置一个初始值,你就会发现它会动了

那为什么css属性为0时,动态设置会变化呢? 我觉得是因为微信小程序本身对于setData这个函数,根据改变的data去进行重新渲染时,就没有包含对**css自定义属性(变量)** 的对比,所以会导致渲染失效,这个值一直都是初始值

解决animation执行时domWidth为0的问题

这个时候有一个简单的解决办法

  • 先判断如果domWidth为0,就再渲染一个由这个数组循环成的静止dom,并不渲染下面两个数组组成的滚动弹幕dom
  • 然后去获取这个静止dom的长度,获取完成了之后,再去渲染下面的滚动弹幕dom,并把静止dom给销毁掉
  • 那么当animation在初始化执行时,domWidth的长度就已经是单个数组所渲染的dom长度,就可以顺利的去执行动画了
<view class="z-bullet-test">
    <block wx:if="{{domWidth === 0}}">
        <view class="z-container">
            <view class="z-single" wx:for="{{10}}" wx:key="index">
                {{index}}
            </view>
        </view>
    </block>
    <block wx:else>
        <view class="z-list" style="--domWidth--:-{{domWidth}}px;">
            <view class="z-container">
                <view class="z-single" wx:for="{{10}}" wx:key="index">
                    {{index}}
                </view>
            </view>
            <!--  -->
            <view class="z-container">
                <view class="z-single" wx:for="{{10}}" wx:key="index">
                    {{index}}
                </view>
            </view>
        </view>
    </block>
</view>

这样子就真的大功告成了!!

微信小程序纯CSS实现无限弹幕滚动效果

如何实现多行弹幕滚动?

多行弹幕滚动的话,你就多复制几行就可以了,并且可以给每一行设置一个animation-delay属性,让他们不要同时执行animation,去岔开时间来执行,实现一个比较乱的弹幕效果。

如何实现动画的停止与开始呢?

对于一个循环动画,你如果要动态的设置它开始或停止,可以通过animation-play-state属性设置runningpaused来设置动画的开始或停止

总结

这篇文章,我把自己对于实现一个无限弹幕滚动需求的过程一一描述了一遍,把自己遇到的问题和思考的过程通过文字表述了出来,写的可能不是很好,希望可以勉励自己的同时也能帮助到看到这篇文章的朋友,如果你觉得我的废话太多,可以直接gitee地址去复制代码自己看一下,实现的代码量很少,也比较简单。

到此这篇关于微信小程序纯CSS实现无限弹幕滚动效果的文章就介绍到这了,更多相关CSS弹幕滚动内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章,希望大家以后多多支持三水点靠木!

 
HTML / CSS 相关文章推荐
CSS3 background-image颜色渐变的实现代码
Sep 13 HTML / CSS
利用CSS3的特性改变文本选中时的颜色
Sep 11 HTML / CSS
纯css3实现效果超级炫的checkbox复选框和radio单选框
Sep 01 HTML / CSS
HTML5标签小集
Aug 02 HTML / CSS
HTML5之SVG 2D入门6—视窗坐标系与用户坐标系及变换概述
Jan 30 HTML / CSS
HTML5组件Canvas实现图像灰度化(步骤+实例效果)
Apr 22 HTML / CSS
html5 worker 实例(二) 图片变换效果
Jun 24 HTML / CSS
在HTML5中使用MathML数学公式的简单讲解
Feb 19 HTML / CSS
HTML5高仿微信聊天、微信聊天表情|对话框|编辑器功能
Apr 23 HTML / CSS
Canvas高级路径操作之拖拽对象的实现
Aug 05 HTML / CSS
Html5监听手机摇一摇事件的实现
Nov 07 HTML / CSS
HTML5 SEO优化的一些建议
Aug 27 HTML / CSS
使用 CSS 构建强大且酷炫的粒子动画效果
详解CSS中postion和opacity及cursor的特性
Aug 14 #HTML / CSS
html网页引入svg图片的4种方式
HTML静态页面获取url参数和UserAgent的实现
Aug 05 #HTML / CSS
CSS使用Flex和Grid布局实现3D骰子
Aug 05 #HTML / CSS
css中:last-child不生效的解决方法
Aug 05 #HTML / CSS
CSS浮动引起的高度塌陷问题
Aug 05 #HTML / CSS
You might like
PHP文件上传实例详解!!!
2007/01/02 PHP
php横向重复区域显示二法
2008/09/25 PHP
劣质的PHP代码简化
2010/02/08 PHP
一些PHP Coding Tips(php小技巧)[2011/04/02最后更新]
2011/05/02 PHP
PHP+memcache实现消息队列案例分享
2014/05/21 PHP
Yii使用find findAll查找出指定字段的实现方法
2014/09/05 PHP
jQuery中的read和JavaScript中的onload函数的区别
2014/08/27 Javascript
jQuery实现自定义下拉列表
2015/01/05 Javascript
JavaScript实现数字数组按照倒序排列的方法
2015/04/06 Javascript
谷歌showModalDialog()方法不兼容出现对话窗口的解决办法
2016/02/15 Javascript
举例讲解如何判断JavaScript中对象的类型
2016/04/22 Javascript
Bootstrap组件学习之导航、标签、面包屑导航(精品)
2016/05/17 Javascript
[js高手之路]HTML标签解释成DOM节点的实现方法
2017/08/31 Javascript
vue组件父子间通信详解(三)
2017/11/07 Javascript
vue之浏览器存储方法封装实例
2018/03/15 Javascript
vuejs2.0运用原生js实现简单拖拽元素功能
2020/08/21 Javascript
Jquery的Ajax技术使用方法
2019/01/21 jQuery
vue2.0实现的tab标签切换效果(内容可自定义)示例
2019/02/11 Javascript
微信小程序 弹窗输入组件的实现解析
2019/08/12 Javascript
javascript实现拖拽碰撞检测
2020/03/12 Javascript
[52:37]完美世界DOTA2联赛循环赛 Forest vs DM BO2第一场 10.29
2020/10/29 DOTA
Falsk 与 Django 过滤器的使用与区别详解
2019/06/04 Python
浅谈pytorch池化maxpool2D注意事项
2020/02/18 Python
Python如何测试stdout输出
2020/08/10 Python
工程师必须了解的LRU缓存淘汰算法以及python实现过程
2020/10/15 Python
C#里面可以避免一个类被其他类继承么?如何?
2013/09/26 面试题
4s客服专员岗位职责
2013/12/01 职场文书
奥巴马的演讲稿
2014/05/15 职场文书
大学活动总结模板
2014/07/10 职场文书
建筑结构施工求职信
2014/07/11 职场文书
责任书格式
2015/01/29 职场文书
手术室护士个人总结
2015/02/13 职场文书
社区义诊通知
2015/04/24 职场文书
2015年教师节感言
2015/08/03 职场文书
财务人员廉洁自律心得体会
2016/01/13 职场文书
家电创业计划书
2019/08/05 职场文书