ios - 将进度条添加到 UIAlertController 并显示更新

标签 ios swift3 queue progress-bar uialertcontroller

我尝试在我的应用程序中添加进度条。我在 How to add Progress bar to UIAlertController? 上找到了问题但它没有显示如何更新进度条。我将代码简化如下,但它没有更新进度条(它只在进度完成后显示)。我看了什么?感谢您的帮助。

override func viewDidLoad() {
    super.viewDidLoad()
    var topViewController = UIApplication.shared.delegate!.window!!.rootViewController!
    while (topViewController.presentedViewController != nil){
        topViewController = topViewController.presentedViewController!
    }
    DispatchQueue.main.async(execute: {
        let alert = UIAlertController(title: "downloading", message: "pls wait", preferredStyle: .alert)
        let progressBar = UIProgressView(progressViewStyle: .default)
        progressBar.setProgress(0.0, animated: true)
        progressBar.frame = CGRect(x: 10, y: 70, width: 250, height: 0)
        alert.view.addSubview(progressBar)
        topViewController.present(alert, animated: true, completion: nil)
        var progress: Float = 0.0
        repeat {
            DispatchQueue.global(qos: .background).async(execute: {
                progress += 0.01
                print (progress)
                DispatchQueue.main.async(flags: .barrier, execute: {
                    progressBar.setProgress(progress, animated: true)
                })
            })
        } while progress < 1.0
    })
}

最佳答案

您的代码中的所有这些调度队列可能会让您感到有点困惑:-)

我试着解释这个问题:

第一个(最外面的) DispatchQueue.main.async 在主 (UI) 线程中执行。 它创建 UIAlertController,将 UIProgressView 填充到其中并显示对话框。然后它在主线程中执行一些时间紧迫的工作(repeat-while-loop)。永远不要这样做,因为它阻塞主线程。 由于主线程被阻塞,UI 控件中不会反射(reflect)任何更新,因此您看不到进度变化。

因此,

  • 我将时间紧迫工作项(重复循环)作为异步工作项放入全局队列(不是主队列)
    • 在该工作项中,我会将进度条更新分派(dispatch)到主队列(在主线程中执行)
    • 最后,当一切都完成后,我关闭 UIAlertController

只有当您的警报 View 通过 viewDidLoadviewWillAppear 等方法显示时,您才需要 first Dispatch,例如从尚未显示 View 的时间点开始。如果您从按钮回调或 viewDidAppear(例如, View 可见)调用它,您可以跳过外部 Dispatch。

开始吧 - 我还加入了一些休眠时间并稍微修改了 GCD 调用以使用尾随闭包:

override func viewDidLoad() {
    super.viewDidLoad()
    var topViewController = UIApplication.shared.delegate!.window!!.rootViewController!
    while (topViewController.presentedViewController != nil){
        topViewController = topViewController.presentedViewController!
    }

    // Usage of GCD here is only necessary because it's called from
    // viewDidLoad. If called by viewDidAppear or some action callback, you 
    // don't need it:
    DispatchQueue.main.async {
        let alert = UIAlertController(title: "downloading", message: "pls wait", preferredStyle: .alert)
        let progressBar = UIProgressView(progressViewStyle: .default)
        progressBar.setProgress(0.0, animated: true)
        progressBar.frame = CGRect(x: 10, y: 70, width: 250, height: 0)
        alert.view.addSubview(progressBar)
        topViewController.present(alert, animated: true, completion: nil)
        var progress: Float = 0.0
        // Do the time critical stuff asynchronously
        DispatchQueue.global(qos: .background).async {
            repeat {
                progress += 0.1
                Thread.sleep(forTimeInterval: 0.25)
                print (progress)
                DispatchQueue.main.async(flags: .barrier) {
                    progressBar.setProgress(progress, animated: true)
                }
            } while progress < 1.0
            DispatchQueue.main.async {
                alert.dismiss(animated: true, completion: nil);
            }
        }
    }
}

关于ios - 将进度条添加到 UIAlertController 并显示更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41898558/

相关文章:

ios - 应用程序在 Objective-C 中运行时如何获取本地通知

iphone - 在 iPhone 中添加自定义 AnnotationView

swift 3 |核心数据 |自定义实体

ios - 同步谷歌日历?

c++ - 使用具有自定义数据结构的 STL 队列

ios - Firebase 和 GeoFire fetch double 将新帖子添加到 Collection View

ios - 用于多部分表单上传的 AFNetworking 2.0 API

ios - XCUIApplication 上的 swipeUp() 破坏了 UITest 中的 XCUIApplication

Java 用队列计算序列

php - 清除 Laravel 5.x/6.x/7.x/8+ 中 Redis 中的 Laravel 排队作业