Swift 技术 视频播放器滚动条(源码)

news2025/1/14 0:49:42

一直觉得自己写的不是技术,而是情怀,一个个的教程是自己这一路走来的痕迹。靠专业技能的成功是最具可复制性的,希望我的这条路能让你们少走弯路,希望我能帮你们抹去知识的蒙尘,希望我能帮你们理清知识的脉络,希望未来技术之巅上有你们也有我。

整个的原理在视频中已经说的很清楚了,不懂的话直接看视频是很明朗的.
在这里插入图片描述

在这里插入图片描述

前言

2023.09.03自定义封装一个视频拖拽的滚动条.需要满足一下的需求:
1.能拖拽快进
2.能点击快进
3.能预加载进度
4.显示当前播放进度
5.显示节点,点击跳到对应的节点

在这里插入图片描述

思路

该拖拽进度条底部是用来显示预加载进度的,使用系统自带的UIProgressView控件.最外面能拖拽播放进度的使用系统自带的UISlider控件.节点通过for循环创建按键出来的节点.这样有一个问题,当播放到节点的位置的时候,节点会挡住拖拽进度条的位置.所以拖拽进度条的原点,就用一张透明的图片(相当于不要了),然后创建一个按键在最前面,用一张圆形图片,挡住原来UISlider控件的原点,跟随滑动,播放的个过程

在这里插入图片描述
可以看到上面的图片就是用一个自定义的按键挡住了原来UISlider控件的原点.

重点

下面说一下整个封装的几个关键地方.

点击滚动条
在这里插入图片描述
算出point值的公式:
在这里插入图片描述

let point = ((pointTapped.x - positionOfSlider.x) * CGFloat(slider.maximumValue) / widthOfSlider)

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
全部代码

//
//  PlayProgressView.swift
//  SwiftDemol
//
//  Created by 冯汉栩 on 2023/8/21.
//

import UIKit

class PlayProgressView: UIView {
    
    // 预加载进度
    lazy var progress: UIProgressView = {
        let progress = UIProgressView()
        progress.progressTintColor = UIColor(red: 169.0/255.0, green: 169.0/255.0, blue: 169.0/255.0, alpha: 1.0)//深灰色
        progress.trackTintColor = UIColor(red: 228.0/255.0, green: 228.0/255.0, blue: 228.0/255.0, alpha: 1.0)//浅灰色
        return progress
    }()
    
    // 进度条 clearn_point   slider_pointer_normal
    lazy var slider: UISlider = { [weak self] in
        let slider  = UISlider()
        slider.minimumTrackTintColor = UIColor(red: 74.0/255.0, green: 155.0/255.0, blue: 234.0/255.0, alpha: 1.0)//蓝色
        slider.maximumTrackTintColor = UIColor.clear // 透明颜色
        slider.setThumbImage(UIImage(named: "slider_pointer_normal"), for: .normal)//clearn_point(透明) slider_pointer_normal(白色)
        slider.setThumbImage(UIImage(named: "slider_pointer_normal"), for: .highlighted)
        slider.setThumbImage(UIImage(named: "slider_pointer_normal"), for: .selected)
        let tapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(sliderTapped(gestureRecognizer:)))
        slider.addGestureRecognizer(tapGestureRecognizer)
        return slider
    }()
    //自定义point点
    lazy private var draggableButton: UIButton = {
        let button = UIButton()
        button.frame = CGRect(x: 0, y: 0, width: 10, height: 20)
        //button.backgroundColor = .blue.withAlphaComponent(0.5)
        button.setImage(UIImage(named: "slider_pointer_normal"), for: .normal)
        button.tag = 111
        return button
    }()
    
    // 修改播放进度
    var changePointBlock:((_ point: Float) ->Void)?
    // point数组
    var pointList = [Float]()
    // 当前播放点的移动位置
    var pointX:Float = 0.0 {
        didSet {
            let newX = (self.frame.width - draggableButton.frame.width) * CGFloat(pointX)
            if newX.isNaN { return }
            draggableButton.frame = CGRect(x: newX, y: draggableButton.frame.origin.y, width: draggableButton.frame.width, height: draggableButton.frame.height)
            
            slider.value = pointX
        }
    }
    
    override init(frame: CGRect) {
        super.init(frame: frame)
        buildUI()
    }
    
    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
        
    }
    
    private func buildUI() {
        backgroundColor = .clear // .green.withAlphaComponent(0.5)
        
        addSubview(progress)
        progress.snp.makeConstraints { make in
            make.centerY.equalToSuperview()
            make.left.equalToSuperview().offset(2+1)//2是跟随slider的2+偏移1
            make.right.equalToSuperview().offset(-2)
            make.height.equalTo(4)
        }
        
        addSubview(slider)
        slider.snp.makeConstraints { make in
            make.centerY.equalToSuperview().offset(-1)
            make.left.equalToSuperview().offset(2)
            make.right.equalToSuperview().offset(-2)
            make.height.equalTo(20)
        }
        
        
        addSubview(draggableButton)
        let panGesture = UIPanGestureRecognizer(target: self, action: #selector(handlePanGesture(_:)))
        draggableButton.addGestureRecognizer(panGesture)
    }
    
    override func layoutSubviews() {
        super.layoutSubviews()
        
        //严重的bug是否创建过
        if pointList.isEmpty { return }
        //如果有节点就移除
        for subview in self.subviews {
            if subview is UISlider {
                for sliderSubView in subview.subviews {
                    if sliderSubView is UIButton {
                        sliderSubView.removeFromSuperview()
                    }
                }
            }
        }
        //布局节点
        for i in 0..<pointList.count {
            let button = UIButton()
            slider.addSubview(button)
            button.tag = i
            button.setImage(UIImage(named: "point_none"), for: .normal)
            button.addTarget(self, action: #selector(pointNoneClick), for: .touchUpInside)
            button.snp.makeConstraints { make in
                make.width.equalTo(16)
                make.height.equalTo(20)
                make.centerY.equalToSuperview().offset(1)
                make.left.equalToSuperview().offset(Float(progress.bounds.size.width)*pointList[i]-16*0.5)
            }
        }
  
    }
    
    // MARK: - slider -- 点击滚动条
    @objc private func sliderTapped(gestureRecognizer: UIGestureRecognizer) {
        let pointTapped: CGPoint = gestureRecognizer.location(in: self)
        let positionOfSlider: CGPoint = slider.frame.origin
        let widthOfSlider: CGFloat = slider.frame.size.width
        let point = ((pointTapped.x - positionOfSlider.x) * CGFloat(slider.maximumValue) / widthOfSlider)
  
        //UI--修改进度条的值
        slider.setValue(Float(point), animated: false)
        
        //换算x的值
        let newX = (self.frame.width - draggableButton.frame.width) * CGFloat(point)
        //修改自定义point按键的frame
        draggableButton.frame = CGRect(x: newX, y: draggableButton.frame.origin.y, width: draggableButton.frame.width, height: draggableButton.frame.height)
        
        //闭包出去修改AVPlayer的值 0 -- 100
        changePointBlock?(Float(point))
    }
    
    // MARK: - 按键 -- 节点按键
    @objc private func pointNoneClick(button:UIButton) {
        //换算x的值
        let newX = (self.frame.width - draggableButton.frame.width) * CGFloat(pointList[button.tag])
        //修改自定义point按键的frame
        draggableButton.frame = CGRect(x: newX, y: draggableButton.frame.origin.y, width: draggableButton.frame.width, height: draggableButton.frame.height)
        //UI--修改进度条的值
        slider.setValue(Float(pointList[button.tag]), animated: false)
        //闭包出去修改AVPlayer的值 0 -- 100
        changePointBlock?(pointList[button.tag])
    }
    
    // MARK: - 自定义按键 -- 拖拽
    @objc func handlePanGesture(_ gesture: UIPanGestureRecognizer) {
        let translation = gesture.translation(in: self)
        //translation.x 拖拽的值
        var newX = draggableButton.frame.origin.x + translation.x
        switch gesture.state {
        case .changed:
            //更新按钮的 x 坐标,限制在 view 的范围内
            newX = max(0, min(newX, self.frame.width - draggableButton.frame.width))//意思是:newX取值范围0 ~ self.frame.width - draggableButton.frame.width 之间
            //修改自定义point按键的frame
            draggableButton.frame = CGRect(x: newX, y: draggableButton.frame.origin.y, width: draggableButton.frame.width, height: draggableButton.frame.height)
            gesture.setTranslation(.zero, in: self)
            //修改进度
            //换算成进度0.0-1.0
            let progress = newX / (self.frame.width - draggableButton.frame.width)
            //UI--修改进度条的值
            slider.setValue(Float(progress), animated: false)
            //闭包出去修改AVPlayer的值 0 -- 100
            changePointBlock?(Float(progress))
        default:
            break
        }
    }
    
}

使用

创建+返回值出来
在这里插入图片描述
约束
在这里插入图片描述
赋值给它
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/968640.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

数学建模--非多项式拟合法的Python实现

目录 1.算法异同区别 2.算法核心步骤 3.算法核心代码 4.算法效果展示 1.算法异同区别 #*************************************************************************************************************# 方法区别探究 1.对于多项式拟合你需要大致知道这些点的分布&#xf…

对Excel表中归类的文件夹进行自动分类

首先把excel表另存为.txt文件&#xff08;注意&#xff1a;刚开始可能是ANSI格式&#xff0c;需要转成UTF-8格式&#xff09;&#xff1b;再新建一个.txt文件&#xff0c;重命名成.bat文件(注意&#xff1a;直接创建的如果是是UTF-8格式&#xff0c;最好转成ANSI格式&#xff0…

WEBGL(5):绘制线

1 实现代码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"widthdevice-width, …

零基础教程:使用yolov8训练自己的目标检测数据集

1.前言 Ultralytics YOLOv8 是一款前沿、最先进&#xff08;SOTA&#xff09;的模型&#xff0c;基于先前 YOLO 版本的成功&#xff0c;引入了新功能和改进&#xff0c;进一步提升性能和灵活性。YOLOv8 设计快速、准确且易于使用&#xff0c;使其成为各种物体检测与跟踪、实例…

1782_Adobe Reader X实现pdf分页保存

全部学习汇总&#xff1a; GitHub - GreyZhang/windows_skills: some skills when using windows system. 看了一本pdf电子书&#xff0c;觉得其中几页很值得分享。如果分享整本书当然是不错的选择&#xff0c;但是分享整本书很可能会导致一个结局——内容太多别人不会去看&…

rz命令无法正常使用?

使用rz命令上传文件时出现如下问题&#xff1a; 这里用的是mobaxterm终端 改用xshell,secureCRT即可正常使用&#xff1a;

Ubutnu允许ssh连接使用root与密码登录

文章目录 1. 修改sshd_config2. 设置root密码3. 重启SSH服务 1. 修改sshd_config 修改/etc/ssh/sshd_config文件&#xff0c;找到 #Authentication&#xff0c;将 PermitRootLogin 参数修改为 yes。如果 PermitRootLogin 参数被注释&#xff0c;请去掉首行的注释符号&#xff…

人大金仓国产数据库在线下载体验测试版本安装方法

在线下载人大金仓数据库体验测试版本方法 下载地址如下&#xff1a; https://www.kingbase.com.cn/rjcxxz/index.htm #下载windows版本地址参考 https://blog.csdn.net/qq_57052445/article/details/130427207https://www.kingbase.com.cn/sqwjxz/index.htm此处需要下载获取…

2023开学礼《乡村振兴战略下传统村落文化旅游设计》许少辉八一新书山东师范大学图书馆

2023开学礼《乡村振兴战略下传统村落文化旅游设计》许少辉八一新书山东师范大学图书馆

openGauss学习笔记-59 openGauss 数据库管理-相关概念介绍

文章目录 openGauss学习笔记-59 openGauss 数据库管理-相关概念介绍59.1 数据库59.2 表空间59.3 模式59.4 用户和角色59.5 事务管理 openGauss学习笔记-59 openGauss 数据库管理-相关概念介绍 59.1 数据库 数据库用于管理各类数据对象&#xff0c;与其他数据库隔离。创建数据…

1781_通用型归并排序函数的C语言实现

全部学习汇总&#xff1a; GitHub - GreyZhang/c_basic: little bits of c. 近期在学习C语言数据结构&#xff0c;看到了排序。我看得是一本国外的书籍&#xff0c;直接网上寻找到的一个英文版。内容比较简洁&#xff0c;讲解也算是调理清晰。 关于排序算法&#xff0c;书中只…

DT vray(焦散 间接照明 图像照明)

折射属性 焦散 材质覆盖 基于图像的照明 双面材质 间接照明 开启GI 光子映射 边上 Irradiance Maps 渲染平滑几何体

项目文章 |首次发现太平洋亚历山大藻(甲藻)快速生长中H3K4me3修饰

发表单位&#xff1a;中国海洋大学海洋遗传育种教育部重点实验室 发表日期&#xff1a;2022年11月30日 期刊 &#xff1a;Frontiers in Marine Science &#xff08;IF2022: 5.247&#xff09; 2022年11月30日中国海洋大学海洋遗传育种教育部重点实验室在Frontiers in Marin…

java八股文面试[多线程]—— as-if-serial

什么是as-if-serial 我不管你编译器和执行器怎么处理指令&#xff0c;怎样的进行指令重排&#xff0c;我要求的单线程所执行的结果不能受影响&#xff0c;我不管你年轻时候犯了什么错&#xff0c;你在人生的过程是怎样来的&#xff0c;我只管你结果是不是成功的。在计算机中&a…

Linux系统中调试GDB调试方法入门分享

本篇讲解使用GDB调试Linux应用程序&#xff0c;以下以 hellowld.c 为例介绍 GDB 的调试入门&#xff1a; 编写代码 #include <stdio.h>int main(int argc, char **argv) {int i;int result 0;if(1 > argc){printf("Helloworld.\n");}printf("Hello W…

DEtection TRansformer (DETR) 与 You Only Look Once (YOLO)

曾经想过计算机如何分析图像&#xff0c;识别并定位其中的物体吗&#xff1f;这正是计算机视觉领域的目标检测所完成的任务。DEtection TRansformer&#xff08;DETR&#xff09;和You Only Look Once&#xff08;YOLO&#xff09;是目标检测的两种重要方法。YOLO已经赢得了作为…

Ubuntu18.04版本下配置ORB-SLAM3和数据集测试方法

文章目录 环境说明必要配置一、Pangolin源码和库文件下载依赖安装和编译安装 二、Eigen3源码和库文件下载编译安装 三、Opencv源码和库文件下载编译安装 四、DBoW2 和 g2o五、boost源码和库文件下载编译安装 六、libssl-dev七、ORB-SLAM3源码和库文件下载编译安装 数据集测试参…

Python 字典排序超级简单

再Python中不可避免地要对字典进行排序&#xff0c;有时候字典里放着还是数组&#xff0c;对数组的某个位置元素进行排序&#xff0c;这样有点不容易 转换下思路&#xff0c;可以将字典放在Pandas中的DataFrame中&#xff0c;这样就可以迅速排序了。 import pandas as pd# 原…

8月琐碎但值得的事情

8月份结束了&#xff0c;最近心态比较好&#xff0c;慢点就慢点&#xff0c;没有那么着急了&#xff0c;可能是因为着急也没啥办法&#xff0c; 8月是比较开心的一个月&#xff0c;可能是做的事情更有盼头了&#xff0c;可能是看了喜欢的书&#xff0c;可能是我变瘦了&#xff…

LMD-恶意软件检测工具

LMD是Linux恶意软件扫描器&#xff0c;以GNU GLPv2许可发布。 官方地址&#xff1a;https://www.rfxn.com 下载软件包命令&#xff1a; wget https://www.rfxn.com/downloads/maldetect-current.tar.gz tar命令解包后进入其目录。 安装命令如下&#xff1a; ./install.sh …