jjzjj

ios - DrawRect 超出 View 范围

coder 2023-09-15 原文

我正在尝试在自定义 UIView 中绘制一个循环计时器,如下所示:

override func draw(_ rect: CGRect) {
    bgShapeLayer.path = UIBezierPath(arcCenter: CGPoint(x: self.frame.midX , y: self.frame.midY), radius:
    min(frame.width/2, frame.height/2), startAngle: -90.degreesToRadians, endAngle: 270.degreesToRadians, clockwise: true).cgPath
    bgShapeLayer.strokeColor = self.backgroundStrokeColor
    bgShapeLayer.fillColor = self.backgroundFillColor
    bgShapeLayer.lineWidth = self.backgroundLineWidth
    self.layer.addSublayer(bgShapeLayer)        

    timeLeftShapeLayer.path = UIBezierPath(arcCenter: CGPoint(x: self.frame.midX , y: self.frame.midY), radius:
        min(frame.width/2, frame.height/2), startAngle: -90.degreesToRadians, endAngle: 270.degreesToRadians, clockwise: true).cgPath
    timeLeftShapeLayer.strokeColor = self.timeLeftSrtokeColor
    timeLeftShapeLayer.fillColor = self.timeLeftFillColor
    timeLeftShapeLayer.lineWidth = self.timeLeftLineWidth
    self.layer.addSublayer(timeLeftShapeLayer)        

    timeLabel = UILabel(frame: CGRect(x: self.frame.midX-50 ,y: self.frame.midY-25, width: 100, height: 50))
    timeLabel.adjustsFontSizeToFitWidth = true
    timeLabel.textAlignment = .center
    timeLabel.text = self.timeLeft.stringTime
    timeLabel.textColor = self.textColor
    timeLabel.font = self.textFont
    self.addSubview(timeLabel)

    strokeIt.toValue = 100 //From value is set in "startTimer(duration, timerProgress)
    strokeIt.duration = self.timeLeft
    // add the animation to your timeLeftShapeLayer
    timeLeftShapeLayer.add(strokeIt, forKey: nil)
    // define the future end time by adding the timeLeft to now Date() 
}

它在 iPhone 6/7/X 上运行良好,但在较小的 iPhone(如 5/5s/SE)上,计时器超出了 View 范围。像这样:

[![SE][1]][1]

它应该如何(iPhone X):

[![X][2]][2]

我已经尝试找出原因几个小时了,但我什么也没找到:(

有人有想法吗?谢谢! 雷尼尔梅里安:

[![图片][3]][3]

[![2][4]][4]

注意:timer View 以编程方式添加到名为 container 的 super View 中,该 super View 是通过 Interface Builder 添加的,并且具有 Clip To边界 已启用。 container 位于 UITableViewCell

完整代码:

    public var backgroundStrokeColor: CGColor = UIColor.white.cgColor
    public var backgroundFillColor: CGColor = UIColor.clear.cgColor
    public var backgroundLineWidth: CGFloat = 15
    public var timeLeftSrtokeColor: CGColor = UIColor.red.cgColor
    public var timeLeftFillColor: CGColor = UIColor.clear.cgColor
    public var timeLeftLineWidth: CGFloat = 15
    public var textColor: UIColor = UIColor.white
    public var textFont: UIFont = UIFont.systemFont(ofSize: 15.0)

    fileprivate var timeLeft: TimeInterval = 0
    fileprivate var endDate: Date?
    fileprivate var timeLeftShapeLayer: CAShapeLayer?
    fileprivate var bgShapeLayer: CAShapeLayer?
    fileprivate var timeLabel:  UILabel?
    fileprivate var timer = Timer()
    fileprivate let strokeIt = CABasicAnimation(keyPath: "strokeEnd")

    //MARK: - UIView
    override func draw(_ rect: CGRect) {    
        drawBgShape()
        drawTimeLeftShape()
        addTimeLabel()

        strokeIt.toValue = 1 //From value is set in "startTimer(duration, timerProgress)
        strokeIt.duration = self.timeLeft
        // add the animation to your timeLeftShapeLayer
        timeLeftShapeLayer?.add(strokeIt, forKey: nil)
        // define the future end time by adding the timeLeft to now Date()

    }

    //MARK: - Public
    public func startTimer(duration: TimeInterval, timerProgress: Int) {
        self.timeLeft = duration
        endDate = Date().addingTimeInterval(timeLeft)
        timer = Timer.scheduledTimer(timeInterval: 0.1, target: self, selector: #selector(updateTime), userInfo: nil, repeats: true)
        strokeIt.fromValue = timerProgress
    }

    //MARK: - Private
    fileprivate func drawBgShape() {
        //we initialize and add the layer only if there is not initialized
        if(bgShapeLayer == nil){
            bgShapeLayer = CAShapeLayer()
            self.layer.addSublayer(bgShapeLayer!)
        }

        bgShapeLayer?.path = UIBezierPath(arcCenter: CGPoint(x: self.frame.midX , y: self.frame.midY), radius:
            min(frame.width/2, frame.height/2), startAngle: -90.degreesToRadians, endAngle: 270.degreesToRadians, clockwise: true).cgPath
        bgShapeLayer?.strokeColor = self.backgroundStrokeColor
        bgShapeLayer?.fillColor = self.backgroundFillColor
        bgShapeLayer?.lineWidth = self.backgroundLineWidth
    }

    fileprivate func drawTimeLeftShape() {
        //we initialize and add the layer only if there is not initialized
        if(timeLeftShapeLayer == nil){
            timeLeftShapeLayer = CAShapeLayer()
            self.layer.addSublayer(timeLeftShapeLayer!)
        }
        timeLeftShapeLayer?.path = UIBezierPath(arcCenter: CGPoint(x: self.frame.midX , y: self.frame.midY), radius:
            min(frame.width/2, frame.height/2), startAngle: -90.degreesToRadians, endAngle: 270.degreesToRadians, clockwise: true).cgPath
        timeLeftShapeLayer?.strokeColor = self.timeLeftSrtokeColor
        timeLeftShapeLayer?.fillColor = self.timeLeftFillColor
        timeLeftShapeLayer?.lineWidth = self.timeLeftLineWidth
    }

    fileprivate func addTimeLabel() {
        //we initialize and add the UILabel only if there is not initialized
        if(timeLabel == nil){
            timeLabel = UILabel()
            self.addSubview(timeLabel!)
        }

        timeLabel?.frame = CGRect(x: self.frame.midX-50 ,y: self.frame.midY-25, width: 100, height: 50)
        timeLabel?.adjustsFontSizeToFitWidth = true
        timeLabel?.textAlignment = .center
        timeLabel?.text = self.timeLeft.stringTime
        timeLabel?.textColor = self.textColor
        timeLabel?.font = self.textFont
    }

    //MARK: - Actions
    @objc fileprivate func updateTime() {
        if timeLeft > 0 {
            timeLeft = endDate?.timeIntervalSinceNow ?? 0
            timeLabel?.text = self.timeLeft.stringTime
        } else {
            timeLabel?.text = self.timeLeft.stringTime
            timer.invalidate()
        }
    }

UITableViewCell 代码:

override func layoutSubviews() {
    super.layoutSubviews()
    let countDownTimer = CountDownTimer(frame: self.countDownTimerContainer.frame)
    countDownTimer.frame.origin = CGPoint.zero
    countDownTimer.frame = self.countDownTimerContainer.bounds
    let secondsUntilEndDate = abs(self.timerEndDate.timeIntervalSinceNow)//abs = absolute value (-x to x)
    countDownTimer.startTimer(duration: secondsUntilEndDate, timerProgress: self.timerProgress)

    self.countDownTimerContainer.addSubview(countDownTimer)
        self.countDownTimerContainer.setNeedsDisplay()
}

最佳答案

您需要在单元格layoutSubViews 方法中重新调整containerView 的框架

func layoutSubviews() {
    super.layoutSubviews()
    timerView.frame = container.bounds
    containerView.setNeedsDisplay()
}

在你原来的方法稍作调整之后

override func draw(_ rect: CGRect) {
    //we initialize and add the layer only if there is not initialized
    if(bgShapeLayer == nil){
       bgShapeLayer = CAShapeLayer()
       self.layer.addSublayer(bgShapeLayer)  
    }

    bgShapeLayer.path = UIBezierPath(arcCenter: CGPoint(x: self.frame.midX , y: self.frame.midY), radius:
    min(frame.width/2, frame.height/2), startAngle: -90.degreesToRadians, endAngle: 270.degreesToRadians, clockwise: true).cgPath
    bgShapeLayer.strokeColor = self.backgroundStrokeColor
    bgShapeLayer.fillColor = self.backgroundFillColor
    bgShapeLayer.lineWidth = self.backgroundLineWidth

       //we initialize and add the layer only if there is not initialized
    if(timeLeftShapeLayer == nil){
       timeLeftShapeLayer = CAShapeLayer()
       self.layer.addSublayer(timeLeftShapeLayer)  
    }
    timeLeftShapeLayer.path = UIBezierPath(arcCenter: CGPoint(x: self.frame.midX , y: self.frame.midY), radius:
        min(frame.width/2, frame.height/2), startAngle: -90.degreesToRadians, endAngle: 270.degreesToRadians, clockwise: true).cgPath
    timeLeftShapeLayer.strokeColor = self.timeLeftSrtokeColor
    timeLeftShapeLayer.fillColor = self.timeLeftFillColor
    timeLeftShapeLayer.lineWidth = self.timeLeftLineWidth     

       //we initialize and add the UILabel only if there is not initialized
    if(timeLabel == nil){
       timeLabel = timeLabel()
       self.addSubview(timeLabel)
    }
    timeLabel.frame = CGRect(x: self.frame.midX-50 ,y: self.frame.midY-25, width: 100, height: 50)
    timeLabel.adjustsFontSizeToFitWidth = true
    timeLabel.textAlignment = .center
    timeLabel.text = self.timeLeft.stringTime
    timeLabel.textColor = self.textColor
    timeLabel.font = self.textFont

    strokeIt.toValue = 100 //From value is set in "startTimer(duration, timerProgress)
    strokeIt.duration = self.timeLeft
    // add the animation to your timeLeftShapeLayer
    timeLeftShapeLayer.add(strokeIt, forKey: nil)
    // define the future end time by adding the timeLeft to now Date() 
}

为此更改您的手 secret 码

在您的单元格中添加 CountDownTimer 作为 var

var timerView :  CountDownTimer?

修改您的 layoutSubviews 方法

override func layoutSubviews() {
    super.layoutSubviews()
    if(self.timerView == nil) {
        self.timerView = CountDownTimer(frame: self.countDownTimerContainer.bounds)
        self.countDownTimerContainer.addSubview(countDownTimer)
    }

    countDownTimer.frame = self.countDownTimerContainer.bounds
    let secondsUntilEndDate = abs(self.timerEndDate.timeIntervalSinceNow)//abs = absolute value (-x to x)
    countDownTimer.startTimer(duration: secondsUntilEndDate, timerProgress: self.timerProgress)
    self.countDownTimerContainer.setNeedsDisplay()
}

关于ios - DrawRect 超出 View 范围,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48886824/

有关ios - DrawRect 超出 View 范围的更多相关文章

  1. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  2. ruby-on-rails - 渲染另一个 Controller 的 View - 2

    我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>

  3. ruby - 触发器 ruby​​ 中 3 点范围运算符和 2 点范围运算符的区别 - 2

    请帮助我理解范围运算符...和..之间的区别,作为Ruby中使用的“触发器”。这是PragmaticProgrammersguidetoRuby中的一个示例:a=(11..20).collect{|i|(i%4==0)..(i%3==0)?i:nil}返回:[nil,12,nil,nil,nil,16,17,18,nil,20]还有:a=(11..20).collect{|i|(i%4==0)...(i%3==0)?i:nil}返回:[nil,12,13,14,15,16,17,18,nil,20] 最佳答案 触发器(又名f/f)是

  4. ruby-on-rails - 如何在我的 Rails 应用程序 View 中打印 ruby​​ 变量的内容? - 2

    我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby​​中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R

  5. ruby-on-rails - 相关表上的范围为 "WHERE ... LIKE" - 2

    我正在尝试从Postgresql表(table1)中获取数据,该表由另一个相关表(property)的字段(table2)过滤。在纯SQL中,我会这样编写查询:SELECT*FROMtable1JOINtable2USING(table2_id)WHEREtable2.propertyLIKE'query%'这工作正常:scope:my_scope,->(query){includes(:table2).where("table2.property":query)}但我真正需要的是使用LIKE运算符进行过滤,而不是严格相等。然而,这是行不通的:scope:my_scope,->(que

  6. ruby-on-rails - 如何在 Rails View 上显示错误消息? - 2

    我是rails的新手,想在form字段上应用验证。myviewsnew.html.erb.....模拟.rbclassSimulation{:in=>1..25,:message=>'Therowmustbebetween1and25'}end模拟Controller.rbclassSimulationsController我想检查模型类中row字段的整数范围,如果不在范围内则返回错误信息。我可以检查上面代码的范围,但无法返回错误消息提前致谢 最佳答案 关键是您使用的是模型表单,一种显示ActiveRecord模型实例属性的表单。c

  7. ruby - 当使用::指定模块时,为什么 Ruby 不在更高范围内查找类? - 2

    我刚刚被困在这个问题上一段时间了。以这个基地为例:moduleTopclassTestendmoduleFooendend稍后,我可以通过这样做在Foo中定义扩展Test的类:moduleTopmoduleFooclassSomeTest但是,如果我尝试通过使用::指定模块来最小化缩进:moduleTop::FooclassFailure这失败了:NameError:uninitializedconstantTop::Foo::Test这是一个错误,还是仅仅是Ruby解析变量名的方式的逻辑结果? 最佳答案 Isthisabug,or

  8. ruby - 如何验证 IO.copy_stream 是否成功 - 2

    这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下

  9. Ruby 文件 IO 定界符? - 2

    我正在尝试解析一个文本文件,该文件每行包含可变数量的单词和数字,如下所示:foo4.500bar3.001.33foobar如何读取由空格而不是换行符分隔的文件?有什么方法可以设置File("file.txt").foreach方法以使用空格而不是换行符作为分隔符? 最佳答案 接受的答案将slurp文件,这可能是大文本文件的问题。更好的解决方案是IO.foreach.它是惯用的,将按字符流式传输文件:File.foreach(filename,""){|string|putsstring}包含“thisisanexample”结果的

  10. Ruby 从大范围中获取第 n 个项目 - 2

    假设我有这个范围:("aaaaa".."zzzzz")如何在不事先/每次生成整个项目的情况下从范围中获取第N个项目? 最佳答案 一种快速简便的方法:("aaaaa".."zzzzz").first(42).last#==>"aaabp"如果出于某种原因你不得不一遍又一遍地这样做,或者如果你需要避免为前N个元素构建中间数组,你可以这样写:moduleEnumerabledefskip(n)returnto_enum:skip,nunlessblock_given?each_with_indexdo|item,index|yieldit

随机推荐