草庐IT

ios - 改变不同变量的动画回调

coder 2023-09-11 原文

我有一个显示 View 并在指定时间间隔后自动消失的按钮。现在,如果在 View 已经可见时再次按下按钮,那么它应该消失并显示一个新 View ,并且重置新 View 的计时器。

在按下按钮时,我有以下代码:

func showToast() {
    timer?.invalidate()
    timer = nil

    removeToast()

    var appDelegate = UIApplication.sharedApplication().delegate as! AppDelegate
    var toAddView = appDelegate.window!

    toastView = UIView(frame: CGRectMake(0, toAddView.frame.height, toAddView.frame.width, 48))
    toastView.backgroundColor = UIColor.darkGrayColor()
    toAddView.addSubview(toastView)

    timer = NSTimer.scheduledTimerWithTimeInterval(2.0, target: self, selector: Selector("removeToast"), userInfo: nil, repeats: false)

    UIView.animateWithDuration(0.5, animations: { () -> Void in
        self.toastView.frame.origin.y -= 48
    })
}

要删除 toast ,我有以下代码:

func removeToast() {
    if toastView != nil {
        UIView.animateWithDuration(0.5,
            animations: { () -> Void in
                self.toastView.frame.origin.y += 48
            },
            completion: {(completed: Bool) -> Void in
                self.toastView.removeFromSuperview()
                self.toastView = nil
            })
    }
}

现在,尽管我每次都通过执行 timer.invalidate() 重置计时器,但我在 removeToast() 中收到了两次调用,它们删除了新插入的 View 。会不会是 UIView.animate 导致了问题,我不知道如何调试 removeToast() 的两个回调。显示该行为的演示项目是 here

注意: 我确实发现一些帖子说使用 dispatch_after() 而不是计时器,@jervine10 也提出了同样的要求,但这并不能满足我的需求。就好像我使用 dispatch_after 那么很难使 GCD 调用无效。有什么可以用 NSTimers 完成的吗?我认为 NSTimers 就是为此而设计的,但我做错了什么。

最佳答案

很抱歉没有看到您的示例项目,感谢您指引我去做。我可以清楚地看到问题出在哪里,解决方案也非常简单。将删除 toast 更改为:

func removeToast() {

    guard let toastView = self.toastView else {
        return
    }

    UIView.animateWithDuration(0.1,
        animations: { () -> Void in
            toastView.frame.origin.y += 48
        },
        completion: {(completed: Bool) -> Void in
            toastView.removeFromSuperview()

            if self.toastView == toastView {
                self.toastView = nil
            }
        })
}

基本上,问题是您在动画 block 中捕获 self,而不是 toastView。因此,一旦动画 block 异步执行,它们将删除在上一个函数中设置的 toastView

解决方案很简单,也修复了可能的竞争条件,那就是将 toastView 捕获到一个变量中。最后,我们检查实例变量是否等于我们要删除的 View ,我们将其取消。

提示:考虑对toastView

使用weak 引用

关于ios - 改变不同变量的动画回调,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32315097/

有关ios - 改变不同变量的动画回调的更多相关文章

  1. ruby-on-rails - 如何使用 instance_variable_set 正确设置实例变量? - 2

    我正在查看instance_variable_set的文档并看到给出的示例代码是这样做的:obj.instance_variable_set(:@instnc_var,"valuefortheinstancevariable")然后允许您在类的任何实例方法中以@instnc_var的形式访问该变量。我想知道为什么在@instnc_var之前需要一个冒号:。冒号有什么作用? 最佳答案 我的第一直觉是告诉你不要使用instance_variable_set除非你真的知道你用它做什么。它本质上是一种元编程工具或绕过实例变量可见性的黑客攻击

  2. ruby - 通过 ruby​​ 进程共享变量 - 2

    我正在编写一个gem,我必须在其中fork两个启动两个webrick服务器的进程。我想通过基类的类方法启动这个服务器,因为应该只有这两个服务器在运行,而不是多个。在运行时,我想调用这两个服务器上的一些方法来更改变量。我的问题是,我无法通过基类的类方法访问fork的实例变量。此外,我不能在我的基类中使用线程,因为在幕后我正在使用另一个不是线程安全的库。所以我必须将每个服务器派生到它自己的进程。我用类变量试过了,比如@@server。但是当我试图通过基类访问这个变量时,它是nil。我读到在Ruby中不可能在分支之间共享类变量,对吗?那么,还有其他解决办法吗?我考虑过使用单例,但我不确定这是

  3. 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

  4. ruby - 如何在 Rails 4 中使用表单对象之前的验证回调? - 2

    我有一个服务模型/表及其注册表。在表单中,我几乎拥有服务的所有字段,但我想在验证服务对象之前自动设置其中一些值。示例:--服务Controller#创建Action:defcreate@service=Service.new@service_form=ServiceFormObject.new(@service)@service_form.validate(params[:service_form_object])and@service_form.saverespond_with(@service_form,location:admin_services_path)end在验证@ser

  5. 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返回它复制的字节数,但是当我还没有下

  6. ruby - 有人可以帮助解释类创建的 post_initialize 回调吗 (Sandi Metz) - 2

    我正在阅读SandiMetz的POODR,并且遇到了一个我不太了解的编码原则。这是代码:classBicycleattr_reader:size,:chain,:tire_sizedefinitialize(args={})@size=args[:size]||1@chain=args[:chain]||2@tire_size=args[:tire_size]||3post_initialize(args)endendclassMountainBike此代码将为其各自的属性输出1,2,3,4,5。我不明白的是查找方法。当一辆山地自行车被实例化时,因为它没有自己的initialize方法

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

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

  8. ruby-on-rails - 使用 ruby​​ 将多个实例变量转换为散列的更好方法? - 2

    我收到格式为的回复#我需要将其转换为哈希值(针对活跃商家)。目前我正在遍历变量并执行此操作:response.instance_variables.eachdo|r|my_hash.merge!(r.to_s.delete("@").intern=>response.instance_eval(r.to_s.delete("@")))end这有效,它将生成{:first="charlie",:last=>"kelly"},但它似乎有点hacky和不稳定。有更好的方法吗?编辑:我刚刚意识到我可以使用instance_variable_get作为该等式的第二部分,但这仍然是主要问题。

  9. ruby - Rack:如何将 URL 存储为变量? - 2

    我正在编写一个简单的静态Rack应用程序。查看下面的config.ru代码:useRack::Static,:urls=>["/elements","/img","/pages","/users","/css","/js"],:root=>"archive"map'/'dorunProc.new{|env|[200,{'Content-Type'=>'text/html','Cache-Control'=>'public,max-age=6400'},File.open('archive/splash.html',File::RDONLY)]}endmap'/pages/search.

  10. Unity 3D 制作开关门动画,旋转门制作,推拉门制作,门把手动画制作 - 2

    Unity自动旋转动画1.开门需要门把手先动,门再动2.关门需要门先动,门把手再动3.中途播放过程中不可以再次进行操作觉得太复杂?查看我的文章开关门简易进阶版效果:如果这个门可以直接打开的话,就不需要放置"门把手"如果门把手还有钥匙需要旋转,那就可以把钥匙放在门把手的"门把手",理论上是可以无限套娃的可调整参数有:角度,反向,轴向,速度运行时点击Test进行测试自己写的代码比较垃圾,命名与结构比较拉,高手轻点喷,新手有类似的需求可以拿去做参考上代码usingSystem.Collections;usingSystem.Collections.Generic;usingUnityEngine;u

随机推荐