Vue组件实现卡片动画倒计时示例详解

目录
  • 前言
  • 需求拆解
  • 组件设计思路
  • 具体开发
    • animate-clock.vue
    • animate-card
  • 项目中使用
  • 后记

前言

最近有朋友在做投票的项目,里面有用到一个倒计时的组件,还想要个动画效果。cv大法浸染多年的我,首先想到的是直接找个现有的组件。

通过一通搜索,看上的只有一个 vue2-flip-countdown,但是当我要修改大小和颜色的时候发现改不了,从而直接把源码拉到项目里面,改起来也挺麻烦。

而且,在搜索大法运行几个周天以后,其实心理已经有了一个倒计时开发整体思路,便决定自己封装一个倒计时出来,最终实现效果如下:

需求拆解

  • 开发一个开箱即用的组件,实现倒计时的效果,显示模式为 01日01时01分01秒。
  • 以终止时间为入参,计算倒计时显示的各个数据。
  • 方便的控制显示的文案,以及主题颜色,大小。

组件设计思路

  • 组件分为两个部分

    • animate-clock,控制组件数据结构,例如:自定义文案、将中文时间单位改成英文,是否换行等等
    • animate-card,动画卡片,即组件里面的具体数字部分,并加上动画。
  • clock 组件中,对传入的 props 进行处理,最核心的为 终止时间(terminalTime),根据终止时间计算天、时、分、秒。
  • 通过定时任务,把计算出来的 天、时、分、秒 数据传入 card 组件,触发视图更新。
  • card 组件中,当数据发生改变的时候触发动画效果。
  • 【升级】在此基础上进行扩展,变为一个倒计时通用解决方案:
    • 反向倒计时
    • 只有数字的倒计时
    • 只有分、秒的倒计时
    • 中文、英文字符串倒计时
    • 动画抽奖
    • 专注时间计时器
    • ...

具体开发

animate-clock.vue

首先设计视图部分:

<template>
  <div class="animate-clock">
    <!-- <p>{{days}}{{hours}}{{minites}}{{seconds}}</p> -->
    <span>距离结束还剩</span>
    <animate-card :val="days" :size="16" :self-disabled="disabled" />
    <span>天</span>
    <animate-card :val="hours" :size="16" :self-disabled="disabled" />
    <span>时</span>
    <animate-card :val="minites" :size="16" :self-disabled="disabled" />
    <span>分</span>
    <animate-card :val="seconds" :size="16" :self-disabled="disabled" />
    <span>秒</span>
  </div>
</template>
<style lang="scss" scoped>
.animate-clock {
  width: 100%;
  text-align: center;
  font-size: 16px;
  font-weight: bold;
  padding: 40px 0 ;
}
</style>

很简单的结构,现在版本为截图所示的一行结构,可以看到,完全可以通过业务组件中通过传入 props 的形式,修改每一个部分的文案,而且 样式也可以随时控制。

js 部分主要做这么几件事:

  • 接收 props,并声明 data
  • 声明一个 工具函数,用来 处理 小于 10 的数字,前面增加 0
  • 声明主要业务函数,被定时任务调用的更新数据方法
<script>
import animateCard from './animate-card.vue'
export default {
  components: { animateCard },
  props: {
    terminalTime: String,
  },
  data() {
    return {
      days: ['0', '0'],
      hours: ['0', '0'],
      minites: ['0', '0'],
      seconds: ['0', '0'],
      setIntVal: null,
      disabled: false,
    }
  },
  mounted() {
    // 先调用一次
    this.updateClock()
    // 箭头函数不修改当前作用域下的 this 指向
    this.setIntVal = setInterval(() => {
      this.updateClock()
    }, 1000)
  },
  methods: {
     /**
     * 更新计时器
     * @result void
     */
    updateClock() {
      let now = new Date().getTime()
      let stopTime = 0
      // 错误入参 处理逻辑
      try {
        stopTime = new Date(this.terminalTime).getTime()
      } catch (err) {
        console.error(err)
        return false
      }
      // 终止逻辑
      const remainingTime = stopTime - now
      if (remainingTime < 1000) {
        clearInterval(this.setIntVal)
        this.setIntVal = null
        // 计时器 清零
        this.days = this.hours = this.minites = this.seconds = ['0', '0']
        this.disabled = true
        console.log('时间到!')
        return false
      }
      // 计算 日、时、分、秒
      let days = parseInt(remainingTime / (24 * 60 * 60 * 1000))
      let hours = parseInt(
        (remainingTime - 24 * 60 * 60 * 1000 * days) / (60 * 60 * 1000)
      )
      let minites = parseInt(
        (remainingTime - 24 * 60 * 60 * 1000 * days - 60 * 60 * 1000 * hours) /
          (60 * 1000)
      )
      let seconds = parseInt(
        (remainingTime -
          24 * 60 * 60 * 1000 * days -
          60 * 60 * 1000 * hours -
          60 * 1000 * minites) /
          1000
      )
      // 更新 data
      this.days = this.toStringAndUnshiftZero(days)
      this.hours = this.toStringAndUnshiftZero(hours)
      this.minites = this.toStringAndUnshiftZero(minites)
      this.seconds = this.toStringAndUnshiftZero(seconds)
    },
    /**
     * 转化数字为数组,并在 头部填充 0
     * @params num: numnber
     * @result string[]
     */
    toStringAndUnshiftZero(num) {
      const val = num.toString().split('')
      if (num < 10) {
        val.unshift('0')
      }
      return val
    },
  },
}
</script>

这一块根本没有什么技术含量,主要是异常数据的处理,和停止逻辑。 其实计时器清零理论上是不需要出现的,但是在测试过程中发现,会出现最后一帧为 01 的情况,就直接清零了。

animate-card

这个组件一开始考虑的很复杂,想着监听数据的变化,触发一个动画的方法,然后这个方法支持重写,提高组件的扩展性,但是时间不允许,后面又觉得不太必要。

最后选择的方案很简单,却提供了一个可以实现 css 能实现的所有效果的思路。

主要思路是通过 vue 的 transition-group 机制,将 0-9 所有的卡片都渲染好,隐藏起来,通过 v-show 来触发绑定在 transition-group 上的动画效果,从而实现动态监听数据变化的效果。

需要注意的是因为宿主项目中 引入了 animate.css,所以就直接使用 animate 的动画效果了。

有需要的,可以翻看文档 【animate.css 官方文档】进行配置。

如果直接CV这套代码的话,没有动画效果。

代码如下:

<template>
  <div class="aimate-card">
    <div class="card-group" v-for="(item,idx) in val" :key="idx" :style="{'font-size': size+'px'}">
      <transition-group enter-active-class="animate__animated animate__bounceIn" leave-active-class="animate__animated animate__fadeOutDown">
        <div class="card-item" :class="{'disabled': selfDisabled}" v-for="num in 10" :key="num" v-show="item== num-1">{{num-1}}</div>
      </transition-group>
    </div>
  </div>
</template>
<script>
export default {
  props: {
    val: {
      type: Array,
      default: () => ['0', '0'],
    },
    size: {
      type: Number,
      default: 16,
    },
    selfDisabled: {
      type: Boolean,
      default: false,
    },
  },
  mounted() {
    console.log(this.selfDisabled)
  },
}
</script>
<style lang="scss" scoped>
.aimate-card {
  width: auto;
  display: inline-block;
  height: 100%;
  .card-group {
    display: inline-block;
    position: relative;
    width: 40px;
    padding: 5px;
    height: 100%;
    vertical-align: middle;
    .card-item {
      position: absolute;
      background: #3a7fe4;
      color: #fff;
      width: 30px;
      height: 40px;
      top: -20px;
      line-height: 40px;
    }
    .disabled {
      background: #ccc !important;
    }
  }
}
</style>

看完代码以后很容易发现,我设计的样式其实一点都不好看,可以对 card-item 写一些前端比较炫的效果。而且动画效果也可以自定义。

项目中使用

粘贴上面两个 vue 文件

在业务页面中 引入并使用

使用方法如下:

<div class="vote-clock">
    <animate-clock :terminalTime="'2023-07-11 23:27:00'" />
</div>

CV大法只支持 terminalTime 这一个入参。

后记

这个组件很简单,但是也有很多可以琢磨的地方,更重要的是整理出一个开发通用组件的思路。

另外,这套代码其实只是一个基础结构,扩展性还是很强的,尤其是前面设计思路中第5条中提到的其他功能,在此基础上可以很快速的进行开发,感兴趣的道友可以简单琢磨一下。

大家在工作中不可避免的会遇到多种多样的需要复用的代码块,有的道友会选择提出为一个公共组件或者公共代码块,有的道友则选择使用CV大法直接复用,也有的可能会另辟蹊径在第一个场景的基础上优化为第二套代码。

我以为这三种情况其实就是一个递进的关系,首先CV大法好,发功后发现场景不是完全一致,进行些许优化,进而搞出一个兼容多个可能存在的场景的通用解决方案。

在设计一个组件的时候,需要尽可能的考虑多种场景,并考虑一下后续的升级方案。想兼容所有的场景肯定不可能,那么就要知道当前组件的边界在哪里。所以,在设计组件的时候不能只着眼于当前业务,更要考虑到其他场景。最简单的例如:一个Button组件,除了保证全局的按钮风格一致的前提下,也要考虑到按钮禁用状态,大按钮,带图标按钮,按钮组这些方面的东西。

以上就是我关于做这个小组件之后进行的一些浅显的思考,共勉之,更多关于Vue卡片动画倒计时的资料请关注我们其它相关文章!

(0)

相关推荐

  • vue实现秒杀倒计时组件

    本文实例为大家分享了vue实现秒杀倒计时组件的具体代码,供大家参考,具体内容如下 下面是使用Vue实现秒杀倒计时组件 开发思路 1.请求服务器获取这一刻的服务器时间(统一以服务器时间为基准) 2.获取用户当前电脑时间与服务器时间比对,获取时间差.以当前电脑时间-减去时间差为最终时间(定义为服务器当前时间) 3.设置秒杀开始时间 4.秒杀时间与服务器当前时间比对,获取时间差(共X秒) 5.根据X秒计算出离秒杀开始时间还有x天x小时x分钟x秒 代码实现 下面代码只展示第4.第5步骤 组件CountD

  • vue实现验证码倒计时按钮

    本文实例为大家分享了vue实现验证码倒计时按钮的具体代码,供大家参考,具体内容如下 1.点击"发送验证码"按钮后进行逻辑判断: ▶️ 如果邮箱已输入且格式正确:按钮变为"已发送" ,此时为不可点击状态 并开始120s倒计时: ▶️ 如果邮箱未输入或格式不正确:显示错误的提示信息. 2.120s倒计时结束后按钮变为"重新发送验证码" . html: <div v-bind:class="{ 'text_email': isActiv

  • vue实现时间倒计时功能

    本文实例为大家分享了vue实现时间倒计时功能的具体代码,供大家参考,具体内容如下 需求: 做一个剩余支付时间倒计时的效果 效果图: 代码: <template> <div>剩余支付时间:{{count}}</div> </template> <script> export default { data() { return { count: '', //倒计时 seconds: 864000 // 10天的秒数 } }, mounted() {

  • vue实现倒计时功能

    本文实例为大家分享了vue实现倒计时功能的具体代码,供大家参考,具体内容如下 通过父组件传入的结束时间减去当前日期得到剩余时间 1.子组件部分 <div class="itemend"> <p class="itemss">倒计时<span>{{day}}</span>天<span>{{hour}}</span>时<span>{{minute}}</span>分<s

  • vue实现同时设置多个倒计时

    本文实例为大家分享了vue实现同时设置多个倒计时的具体代码,供大家参考,具体内容如下 html如下: <div class="home"> <tbody> <tr v-for="(item, index) in bargainGoods" :key="index"> <td v-text="item.down + Djs_timeList(item.countDown)"><

  • vue实现指定日期之间的倒计时

    本文实例为大家分享了vue实现指定日期之间倒计时的具体代码,供大家参考,具体内容如下 效果图如下 此处使用moment.js日期处理类库 使用方法如下 npm install moment 或者 yarn add moment html <div class="time-down"> <div class="back">{{dayNum}}</div> <div class="font-14 date"&

  • Vue组件实现卡片动画倒计时示例详解

    目录 前言 需求拆解 组件设计思路 具体开发 animate-clock.vue animate-card 项目中使用 后记 前言 最近有朋友在做投票的项目,里面有用到一个倒计时的组件,还想要个动画效果.cv大法浸染多年的我,首先想到的是直接找个现有的组件. 通过一通搜索,看上的只有一个 vue2-flip-countdown,但是当我要修改大小和颜色的时候发现改不了,从而直接把源码拉到项目里面,改起来也挺麻烦. 而且,在搜索大法运行几个周天以后,其实心理已经有了一个倒计时开发整体思路,便决定自

  • vue组件生命周期钩子使用示例详解

    目录 组件生命周期图 组件生命周期钩子 1.beforeCreate 2.created 3.beforeMount 4.mounted 5.beforeUpdate 6.updated 7.activated 8.deactivated 9.beforeDestroy 10.destroyed 11.errorCaptured 组件生命周期图 组件生命周期钩子 所有的生命周期钩子自动绑定 一.组件的生命周期:一个组件从创建到销毁的整个过程 二.生命周期钩子:在一个组件生命周期中,会有很多特殊的

  • 使用VitePress搭建及部署vue组件库文档的示例详解

    目录 安装vitepress 目录结构 文档首页 配置 导航栏配置 侧边栏 部署到GitHub Pages 每个组件库都有它们自己的文档.所以当我们开发完成我们自己的组件库必须也需要一个组件库文档.如果你还不了解如何搭建自己的组件库可以看这里->从零搭建Vue3组件库.看完这篇文章你就会发现原来搭建和部署一个组件库文档是那么的简单.当然部署也不需要你有自己的服务器,你只要有github即可.由于我们的组件库还没有完成,所以下面就以element-plus作为示例来搭建一个文档吧. 安装vitep

  • react 组件实现无缝轮播示例详解

    目录 正文 无缝轮播 实现思路 构思使用时代码结构 Carousel组件 CarouselItem组件 完善组件 完成小圆点 正文 需求是做一个无缝轮播图,我说这不是有很多现成的轮子吗?后来了解到他有一个特殊的需求,他要求小圆点需要在轮播图外面,因为现在大部分插件都是将小圆点写在轮播图内部的,这对于不了解插件内部结构的小伙伴确实不知道如何修改. 很久没有写插件的我准备写一个插件(react) 无缝轮播 无缝轮播从最后一张到第一张的过程中不会原路返回,它就像轮子似的,从结束到开始是无缝连接的,非常

  • vue整合项目中百度API示例详解

    目录 官网介绍 申请密钥 官方示例 项目实战 创建地图 获取经纬度 创建Map实例 两个坐标点之间的距离 查询地点信息 Vue项目中整合百度API获取地理位置的方法 组件中使用 vue-baidu-map 百度地图官方vue组件 官网介绍 百度地图 JavaScript API 是一套由 JavaScript 语言编写的应用程序接口 可帮助您在网站中,构建功能丰富交互性强的地图应用 支持PC端和移动端,基于浏览器的地图应用开发,且支持HTML5特性的地图开发 官网传送门 百度地图JavaScri

  • Android Flutter实现3D动画效果示例详解

    目录 前言 AnimatedWidget 简介 3D 旋转动画的实现 总结 前言 上一篇我们介绍了 Animation 和 AnimationController 的使用,这是最基本的动画构建类.但是,如果我们想构建一个可复用的动画组件,通过外部参数来控制其动画效果的时候,上一篇的方法就不太合适了.在 Flutter 中提供了 AnimatedWidget 组件用于构建可复用的动画组件.本篇我们用 AnimatedWidget 来实现组件的3D 旋转效果,如下图所示. AnimatedWidge

  • vue electron实现无边框窗口示例详解

    目录 一.前言 二.实现方案 1.创建无边框窗口 2.创建windows窗口控件组件 三.后记 一.前言 无边框窗口是不带外壳(包括窗口边框.工具栏等),只含有网页内容的窗口.对于一个产品来讲,桌面应用带边框的很少,因为丑(我们的UI觉得--与我无关-.-).因此我们就来展开说下,在做无边框窗口时候需要注意的事项以及我踩过的坑. 二.实现方案 1.创建无边框窗口 要创建无边框窗口,只需在 BrowserWindow的 options 中将 frame 设置为 false: const { Bro

  • Flutter封装组动画混合动画AnimatedGroup示例详解

    目录 一.来源 二.AnimatedGroup使用示例: 三.AnimatedGroup源码 最后 一.来源 项目中遇到混合动画的情况,每次实现都需要生命一堆属性,让代码变得杂乱,难以维护. 参考 iOS 组动画 CAAimationGroup, 随花半天时间封装一个混合动画组件 AnimatedGroup. 此组件基于极简.高扩展.高适用的封装原则,基本满足当前项目开发. 二.AnimatedGroup使用示例: // // AnimatedGroupDemo.dart // flutter_

  • vue实现At人文本输入框示例详解

    目录 知识前置 需求分析 实现 创建能够输入文本的文本框 添加at功能 后记 知识前置 基于vue手把手教你实现一个拥有@人功能的文本编辑器(其实就是微信群聊的输入框) Selection 对象,表示用户选择的文本范围或插入符号的当前 developer.mozilla.org/zh-CN/docs/… contenteditable 是一个枚举属性,表示元素是否可被用户编辑. developer.mozilla.org/zh-CN/docs/… 需求分析 文本框能够输入文本(太简单了) 能够a

  • vue组件之间的数据传递方法详解

    (1)props属性: 在父组件中,可以通过子组件标签属性的形式将数据或者函数传给子组件,子组件通过props去读取父组件传过来的数据 用法 父组件传数据给子组件: 一般的属性值都是用来给子组件展示的 子组件传数据给父组件 属性值为函数类型的,一般是用来子组件向父组件传递数据,子组件通过调用父组件传过来的函数,可以修改父组件的状态数据 缺点: 隔层组件间传递: 必须逐层传递(麻烦) 兄弟组件间: 必须借助父组件(麻烦) 注意: //子组件获取父组件传过来的值 props: { obj: {//o

随机推荐