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

Android笔记(三十二):封装一个毫秒级别倒计时View

效果

倒计时View视频

背景

业务场景需要显示带有毫秒级别的倒计时,于是自己封装一个通用的倒计时组件

源码分析

  1. 核心倒计时逻辑,主要是每隔100毫秒计算一次从开始倒计时到现在的剩余时间,并通过process接口返回出去
  2. Handler每次设置100毫秒的延迟
  3. 将返回出来的时间解析出来
private fun formatTimeToView(remainTime: Long) {val lengthSec = remainTime / 1000val hours = lengthSec / 3600val rem = lengthSec % 3600val minutes = rem / 60val seconds = rem % 60val milliseconds = remainTime % 1000tvMill.text = String.format("%03d", milliseconds)tvHour.text = String.format("%02d", hours)tvMin.text = String.format("%02d", minutes)tvSecond.text = String.format("%02d", seconds)
}

完整源码

class MillCountdownView @JvmOverloads constructor(context: Context, attrs: AttributeSet? = null
) : LinearLayout(context, attrs) {private val root = LayoutInflater.from(context).inflate(R.layout.view_count_down, this, true)private val countDownTask: CountDownRunnableprivate val tvHour: TextViewprivate val tvMin: TextViewprivate val tvSecond: TextViewprivate val tvMill: TextViewinit {background = context.getDrawable(R.drawable.count_down_item_bg)orientation = HORIZONTALgravity = Gravity.CENTER_VERTICALtvHour = root.findViewById(R.id.tvHour)tvMin = root.findViewById(R.id.tvMin)tvSecond = root.findViewById(R.id.tvSecond)tvMill = root.findViewById(R.id.tvMill)countDownTask = CountDownRunnable(1).apply {listener = object : TaskListener {override fun finish() {tvHour.text = "00"tvMin.text = "00"tvSecond.text = "00"tvMill.text = "000"Toast.makeText(context, "倒计时结束", Toast.LENGTH_SHORT).show()}override fun process(remainTime: Long) {if (remainTime < 1) {tvHour.text = "00"tvMin.text = "00"tvSecond.text = "00"tvMill.text = "000"return}formatTimeToView(remainTime)}}}}private fun formatTimeToView(remainTime: Long) {val lengthSec = remainTime / 1000val hours = lengthSec / 3600val rem = lengthSec % 3600val minutes = rem / 60val seconds = rem % 60val milliseconds = remainTime % 1000tvMill.text = String.format("%03d", milliseconds)tvHour.text = String.format("%02d", hours)tvMin.text = String.format("%02d", minutes)tvSecond.text = String.format("%02d", seconds)}/*** 预先展示倒计时文本* @param remainTime 倒计时时间,单位毫秒*/fun preShowRemainSecs(remainTime: Long) {countDownTask.totalCountDownTime = remainTimeformatTimeToView(remainTime)}/*** 开始倒计时* @param remainTime 倒计时时间,单位毫秒*/fun startCountdown(remainTime: Long) {countDownTask.destroy()countDownTask.totalCountDownTime = remainTimecountDownTask.start()}fun destroyCountdown() {countDownTask.destroy()}
}
class CountDownRunnable(@IntRange(from = 1)var totalCountDownTime: Long) : Runnable {private val mHandler = Handler(Looper.getMainLooper())var listener: TaskListener? = nullprivate var startCountDownTime = 0L //开始时当前系统时间private var isTaskExecuting = falseoverride fun run() {if (!isTaskExecuting) {return}val dur = SystemClock.elapsedRealtime() - startCountDownTimeval remainTime = totalCountDownTime - durval mill = remainTime % 1000if (remainTime <= 0) {listener?.finish()isTaskExecuting = falsereturn} else {listener?.process(remainTime)}mHandler.postDelayed(this, 100)}fun start() {startCountDownTime = SystemClock.elapsedRealtime()mHandler.post(this)isTaskExecuting = true}fun resume() {val remainTime = totalCountDownTime - ((SystemClock.elapsedRealtime() - startCountDownTime) / 1000).toInt()if (remainTime <= 0) {listener?.finish()return}mHandler.removeCallbacks(this)isTaskExecuting = truemHandler.post(this)}fun pause() {isTaskExecuting = falsemHandler.removeCallbacks(this)}fun destroy() {isTaskExecuting = falsemHandler.removeCallbacks(this)}
}interface TaskListener {fun finish()fun process(remainTime: Long)
}
<?xml version="1.0" encoding="utf-8"?>
<merge xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="wrap_content"android:layout_height="wrap_content"android:gravity="center_vertical"android:orientation="horizontal"tools:parentTag="LinearLayout"><androidx.appcompat.widget.AppCompatTextViewandroid:id="@+id/tvHour"android:layout_width="wrap_content"android:layout_height="wrap_content"android:minWidth="30dp"android:gravity="center"android:textColor="#000"android:textSize="25sp"android:textStyle="bold"android:layout_marginStart="10dp"android:layout_marginVertical="5dp"tools:text="1" /><androidx.appcompat.widget.AppCompatTextViewandroid:id="@+id/divider1"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginHorizontal="4dp"android:gravity="center"android:text=":"android:textColor="#000"android:textSize="25sp"android:textStyle="bold" /><androidx.appcompat.widget.AppCompatTextViewandroid:id="@+id/tvMin"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginStart="5dp"android:gravity="center"android:textColor="#000"android:minWidth="30dp"android:textSize="25sp"android:textStyle="bold"tools:text="8" /><androidx.appcompat.widget.AppCompatTextViewandroid:id="@+id/divider2"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginHorizontal="4dp"android:gravity="center"android:text=":"android:textColor="#000"android:textSize="25sp"android:textStyle="bold" /><androidx.appcompat.widget.AppCompatTextViewandroid:id="@+id/tvSecond"android:layout_width="wrap_content"android:layout_height="wrap_content"android:gravity="center"android:textColor="#000"android:textSize="25sp"android:textStyle="bold"android:minWidth="30dp"tools:text="3" /><androidx.appcompat.widget.AppCompatTextViewandroid:id="@+id/divider3"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginHorizontal="4dp"android:gravity="center"android:text=":"android:textColor="#000"android:textSize="25sp"android:textStyle="bold" /><androidx.appcompat.widget.AppCompatTextViewandroid:id="@+id/tvMill"android:layout_width="wrap_content"android:minWidth="49dp"android:layout_height="wrap_content"android:layout_marginStart="5dp"android:gravity="center"android:textColor="#a00"android:textSize="25sp"android:textStyle="bold"tools:text="000"android:layout_marginEnd="10dp"/></merge>
http://www.lryc.cn/news/480101.html

相关文章:

  • [产品管理-60]:马斯洛需求层次与产品的情感化设计
  • Python接口自动化测试自学指南(项目实战)
  • ESLint 使用教程(三):12个ESLint 配置项功能与使用方式详解
  • 如何将 EDB 文件导入 Ansys HFSS 和 Ansys Q3D
  • HbuildderX运行到手机或模拟器的Android App基座识别不到设备 mac
  • 智慧流控 力行天地 | 同元软控受邀参加第十三届全国流体传动与控制学术会议
  • Python日志分析与故障定位
  • w029基于springboot的网上购物商城系统研发
  • Uniapp全局文件执行顺序详解
  • 车企死亡加速,买车看好这三条线
  • SpringClud一站式学习之Eureka服务治理(二)
  • 空间解析几何【上】
  • Python 获取PDF的各种页面信息(页数、页面尺寸、旋转角度、页面方向等)
  • 独孤思维:曾经副业赚大钱的人,怎么不见了
  • OpenGL 异常处理-glCreateShader失败
  • 【el-pagination的使用及修改分页组件的整体大小修改默认样式的宽度详细教程】
  • Uniapp的学习
  • C#-万物之父object、装箱拆箱
  • AI大模型重塑软件开发流程:从自动化编码到智能协作的未来展望
  • HTB:GreenHorn[WriteUP]
  • SelfAttention在Ascend上的实现
  • C#设计模式
  • 仪表板展示|DataEase看中国:历年双十一电商销售数据分析
  • 急着骂华为?我劝你别急
  • 虚拟机linux7.9下安装mysql
  • 【Linux】一篇文章轻松搞懂基本指令
  • 深入浅出理解Spring和SpringBoot,剖析自动配置源码
  • Spring配置文件初始化加载(一)
  • 正则表达式 - 简介
  • 【电机控制器】STC8H1K芯片——ADC电压采集