In iOS9, you can create your own NSProgress object, and observe, for example, fractionCompleted. Then you can addChild:
private var observerContext = 0
class ViewController: UIViewController {
private var progress: NSProgress!
override func viewDidLoad() {
super.viewDidLoad()
progress = NSProgress()
progress.addObserver(self, forKeyPath: "fractionCompleted", options: .New, context: &observerContext)
downloadFiles()
}
deinit {
progress?.removeObserver(self, forKeyPath: "fractionCompleted")
}
private func downloadFiles() {
let filenames = ["as17-134-20380.jpg", "as17-140-21497.jpg", "as17-148-22727.jpg"]
let baseURL = NSURL(string: "http://example.com/path")!
progress.totalUnitCount = Int64(filenames.count)
progress.completedUnitCount = 0
for filename in filenames {
let url = baseURL.URLByAppendingPathComponent(filename)
let childProgress = Alamofire.request(.GET, url.absoluteString)
.response() { request, response, data, error in
// process response
}
.progress
progress.addChild(childProgress, withPendingUnitCount: 1)
}
}
override func observeValueForKeyPath(keyPath: String?, ofObject object: AnyObject?, change: [String : AnyObject]?, context: UnsafeMutablePointer<Void>) {
if context == &observerContext {
if keyPath == "fractionCompleted" {
let percent = change![NSKeyValueChangeNewKey] as! Double
print("\(percent)")
}
} else {
super.observeValueForKeyPath(keyPath, ofObject: object, change: change, context: context)
}
}
}
If you need support for iOS 7/8, too, you can call becomeCurrentWithPendingUnitCount and resignCurrent:
for filename in filenames {
let url = baseURL.URLByAppendingPathComponent(filename)
progress.becomeCurrentWithPendingUnitCount(1)
Alamofire.request(.GET, url.absoluteString)
.response() { request, response, data, error in
// process response
}
progress.resignCurrent()
}
If you are using AFNetworking, it's the same process (i.e. the same viewDidLoad, observeValueForKeyPath, and deinit methods as above), but rather than retrieving the Alamofire progress property, you instead use the AFHTTPSessionManager method downloadProgressForTask to get the NSProgress associated with the NSURLSessionTask. For example:
private func getFiles() {
let filenames = ["as17-134-20380.jpg", "as17-140-21497.jpg", "as17-148-22727.jpg"]
let manager = AFHTTPSessionManager()
manager.responseSerializer = AFHTTPResponseSerializer()
let baseURL = NSURL(string: "http://example.com/path")!
progress.totalUnitCount = Int64(filenames.count)
progress.completedUnitCount = 0
for filename in filenames {
let url = baseURL.URLByAppendingPathComponent(filename)
let task = manager.GET(url.absoluteString, parameters: nil, progress: nil, success: { task, responseObject in
// do something with responseObject
print(url.lastPathComponent! + " succeeded")
}, failure: { task, error in
// do something with error
print(error)
})
if let downloadTask = task, let childProgress = manager.downloadProgressForTask(downloadTask) {
progress.addChild(childProgress, withPendingUnitCount: 1)
}
}
}
Or, if using download tasks:
private func downloadFiles() {
let filenames = ["as17-134-20380.jpg", "as17-140-21497.jpg", "as17-148-22727.jpg"]
let manager = AFHTTPSessionManager()
let baseURL = NSURL(string: "http://example.com/path")!
progress.totalUnitCount = Int64(filenames.count)
progress.completedUnitCount = 0
let documents = try! NSFileManager.defaultManager().URLForDirectory(.DocumentDirectory, inDomain: .UserDomainMask, appropriateForURL: nil, create: false)
for filename in filenames {
let url = baseURL.URLByAppendingPathComponent(filename)
let task = manager.downloadTaskWithRequest(NSURLRequest(URL: url), progress: nil, destination: { (temporaryURL, response) -> NSURL in
return documents.URLByAppendingPathComponent(url.lastPathComponent!)
}, completionHandler: { response, url, error in
guard error == nil else {
print(error)
return
}
if let name = url?.lastPathComponent {
print("\(name) succeeded")
}
})
if let childProgress = manager.downloadProgressForTask(task) {
progress.addChild(childProgress, withPendingUnitCount: 1)
}
task.resume()
}
}