Swift 功能实现之滑杆

  今天要搞一个日期滑杆,滑杆上显示当前日期的前15天和后15天。支持范围选取。项目中用到,就写了一下,这里记录一下,写个小demo。

首先,功能动态图:

Swift 功能实现之滑杆_第1张图片

实现

这东西也没有什么可拓展写的,所以可以直接上代码了。

  • 首先处理处理几个工具类:DateTool
import UIKit

class DateTool: NSObject {
    
    //当前时间
    class func getMidTimeText(_ dateCompentCount: Int) -> String {
        let currentDate = getTimeDateWith(dateCompentCount)
        return dateTurnString(date: currentDate)
    }
    
    //左侧时间
    class func getLeftTimeText(_ dateCompentCount: Int) -> String {
        let leftDate: Date = getTimeDateWith(dateCompentCount)
        return dateTurnString(date: leftDate)
    }
    
    //右侧时间
    class func getRightTimeText(_ dateCompentCount: Int) -> String {
        let rightDate: Date = getTimeDateWith(dateCompentCount)
        return dateTurnString(date: rightDate)
    }
    
    
    
    //data转时间
    class func dateTurnString(date: Date) -> String {
        let formatter = DateFormatter()
        formatter.dateFormat = "MM.dd"
        let dateStr = formatter.string(from:date)
        let returnStr = cleanString(dateStr: dateStr)
        return returnStr
    }
    
    //获取距离当前时间指定天数的时间
    class func getTimeDateWith(_ dateCompentCount: Int) -> Date {
        let currentDate = Date()
        var newcomponent = DateComponents()
        newcomponent.day = dateCompentCount
        let date = Calendar.current.date(byAdding: newcomponent, to: currentDate) ?? currentDate
        return date
    }

    
    //时间字符串清零
    class func cleanString(dateStr: String) -> String {
        let strArr = dateStr.components(separatedBy: ".")
        let monthStr: String = strArr.first ?? "0"
        let dayStr: String = strArr.last ?? "0"
        let monCount: Int = Int(monthStr) ?? 0
        let dayCount: Int = Int(dayStr) ?? 0
        let newMonStr = String(monCount)
        let newDayStr = String(dayCount)
        let returnStr = newMonStr + "." + newDayStr
        return returnStr
    }   
}
  • label扩展
import UIKit

extension UILabel {
    
    func setUpLabel(frame: CGRect, text: String, textColor: UIColor, textFont: CGFloat) {
        self.frame = frame
        self.text = text
        self.textColor = textColor
        self.font = UIFont.systemFont(ofSize: textFont)
        self.textAlignment = .center
    }
    
}
  • 工具创建完事,实现controller: DateSliderViewController
import UIKit

let WIDTH: CGFloat = UIScreen.main.bounds.width
let SLIDEW: CGFloat = 14
let SLIDEH: CGFloat = 20

let PRICEBGX: CGFloat = 25
let PRICEBGY: CGFloat = 200
let PRICEBGW: CGFloat = WIDTH - 50
let PRICEBGH: CGFloat = 20
let PRICEMAX: CGFloat = WIDTH * 0.5 + (PRICEBGW * 0.5 - SLIDEW * 0.5 - 5)
let PRICEMIN: CGFloat = WIDTH * 0.5 - (PRICEBGW * 0.5 - SLIDEW * 0.5 - 5)
let RECTWIDTH: CGFloat = PRICEMAX - PRICEMIN
let PROGRESSH: CGFloat = 2
let PROGRESSY: CGFloat = PRICEBGY + (PRICEBGH - 2) / 2




class DateSliderViewController: UIViewController {
    
    
    let leftSlide = UIImageView()
    let rigthSlide = UIImageView()
    
    let leftValue = UILabel()
    let rigthValue = UILabel()
    let midValue = UILabel()
    
    let progressBackView = UIView()
    
    let leftResultLabel = UILabel()
    let rightResultLabel = UILabel()

    override func viewDidLoad() {
        super.viewDidLoad()
        
        let imageView = UIImageView.init(frame: CGRect(x: PRICEBGX, y: PRICEBGY, width: PRICEBGW, height: PRICEBGH))
        imageView.image = UIImage.init(named: "WechatIMG3492")
        self.view.addSubview(imageView)
        
        let leftRect = CGRect(x: PRICEBGX , y: imageView.frame.maxY + 2, width: 30, height: 15)
        let leftText = DateTool.getLeftTimeText(-15)
        leftValue.setUpLabel(frame: leftRect, text: leftText, textColor: UIColor.gray, textFont: 10)
        self.view.addSubview(leftValue)
        
        let rightRect = CGRect(x: imageView.frame.maxX - 20, y: imageView.frame.maxY + 2, width: 30, height: 15)
        let rightText = DateTool.getRightTimeText(15)
        rigthValue.setUpLabel(frame: rightRect, text: rightText, textColor: UIColor.gray, textFont: 10)
        self.view.addSubview(rigthValue)
        
        let midRect = CGRect(x: imageView.frame.midX - 20, y: imageView.frame.maxY + 2, width: 40, height: 16)
        let midText = DateTool.getMidTimeText(0)
        midValue.setUpLabel(frame: midRect, text: midText, textColor: UIColor.black, textFont: 12)
        self.view.addSubview(midValue)
        
        
        //左滑块
        leftSlide.frame = CGRect(x: imageView.center.x - SLIDEW * 0.5 , y: imageView.frame.minY - SLIDEH, width: SLIDEW, height: SLIDEH)
        leftSlide.image = UIImage.init(named: "shangbashou")
        self.view.addSubview(leftSlide)
        
        //右滑块
        rigthSlide.frame = CGRect(x: imageView.center.x - SLIDEW * 0.5, y: imageView.frame.maxY, width: SLIDEW, height: SLIDEH)
        rigthSlide.image = UIImage.init(named: "xiabashou")
        self.view.addSubview(rigthSlide)
        
        //进度条颜色
        progressBackView.frame = CGRect(x: leftSlide.center.x, y: PROGRESSY, width: rigthSlide.center.x - leftSlide.center.x, height: PROGRESSH)
        progressBackView.backgroundColor = UIColor.gray
        self.view.addSubview(progressBackView)

        
        //添加滑动手势
        let leftPanRecognizer = UIPanGestureRecognizer.init(target: self, action: #selector(leftSlideMove(pan:)))
        leftPanRecognizer.maximumNumberOfTouches = 1
        leftPanRecognizer.minimumNumberOfTouches = 1
        leftSlide.isUserInteractionEnabled = true
        leftSlide.addGestureRecognizer(leftPanRecognizer)
        
        let rightPanRecognizer = UIPanGestureRecognizer.init(target: self, action: #selector(rightSlideMove(pan:)))
        rightPanRecognizer.maximumNumberOfTouches = 1
        rightPanRecognizer.minimumNumberOfTouches = 1
        rigthSlide.isUserInteractionEnabled = true
        rigthSlide.addGestureRecognizer(rightPanRecognizer)
        
        leftResultLabel.frame = CGRect(x: 20, y: 400, width: 100, height: 50)
        leftResultLabel.text = DateTool.getMidTimeText(0)
        
        rightResultLabel.frame = CGRect(x: WIDTH - 120, y: 400, width: 100, height: 50)
        rightResultLabel.text = DateTool.getMidTimeText(0)
        self.view.addSubview(leftResultLabel)
        self.view.addSubview(rightResultLabel)
    }
    
    
    @objc func leftSlideMove(pan: UIPanGestureRecognizer) {
        let point = pan.translation(in: leftSlide)
        var x: CGFloat = leftSlide.center.x + point.x
        if x > rigthSlide.center.x {
            x = rigthSlide.center.x
        } else if(x < PRICEMIN) {
            x = PRICEMIN
        }
        leftSlide.center = CGPoint(x: x, y: leftSlide.center.y)
        pan.setTranslation(.zero, in: self.view)
        updataData()
    }
    
    
    @objc func rightSlideMove(pan: UIPanGestureRecognizer) {
        let point = pan.translation(in: rigthSlide)
        var x: CGFloat = rigthSlide.center.x + point.x
        if x > PRICEMAX {
            x = PRICEMAX
        } else if(x < leftSlide.center.x) {
            x = leftSlide.center.x
        }
        rigthSlide.center = CGPoint(x: x, y: rigthSlide.center.y)
        pan.setTranslation(.zero, in: self.view)
        updataData()
    }
    
    func updataData() {
        let progressBackViewRect = CGRect(x: leftSlide.center.x, y: PROGRESSY, width: rigthSlide.center.x - leftSlide.center.x, height: PROGRESSH)
        progressBackView.frame = progressBackViewRect
        
        let leftText = coordinateTurnToNumber(axis: leftSlide.center.x)
        leftResultLabel.text = leftText
        
        let rigthText = coordinateTurnToNumber(axis: rigthSlide.center.x)
        rightResultLabel.text = rigthText
        
    }
    
    
    //坐标转换成数字
    func coordinateTurnToNumber(axis: CGFloat) -> String {
        //将滑条分成30份,取每份的长度
        let everyLenght = RECTWIDTH / 30.0
        //滑动的距离
        let lenght = axis - PRICEMIN
        let count = Int(lenght / everyLenght)
        
        //左面的日期date
        let minDate = DateTool.getTimeDateWith(-15)
        //滑动位置的日期
        var newComponent = DateComponents()
        newComponent.day = count
        let moveDate = Calendar.current.date(byAdding: newComponent, to: minDate) ?? Date()
        
        let returnstr = DateTool.cleanString(dateStr: DateTool.dateTurnString(date: moveDate))
        return returnstr
    }
   
    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }
 
}

最后

ok 最后附上小demo链接:https://github.com/Sufviay/DateSlide,供参考。

你可能感兴趣的:(Swift 功能实现之滑杆)