IGListKit/Example/IGListKitExamples/SectionControllers/WorkingRangeSectionController.swift
Ryan Nystrom f6e088acd8 Working Range example
Summary:
Adding a working range example to the example app. This example:

- Displays a list of images downloaded from unsplash.it
- Create 20 uniquely-random sized objects
- When section controllers enter the range, create a data task to download the image
  - When finished, store image in `downloadedImage`
  - Set in cell if cell is visible
  - Don't create task if already downloaded or task created
- Cancel task when section controller is destroyed
- Show a spinner cell while downloading the image

Fixes #84

- [x] All tests pass. Demo project builds and runs.
- [x] I have reviewed the [contributing guide](https://github.com/Instagram/IGListKit/blob/master/CONTRIBUTING.md)
Closes https://github.com/Instagram/IGListKit/pull/89

Differential Revision: D4050210

Pulled By: rnystrom

fbshipit-source-id: 628a777fa819dccd9d9f4f58646cdf72fb4bc65c
2016-10-19 23:29:12 -07:00

96 lines
3.3 KiB
Swift

/**
Copyright (c) 2016-present, Facebook, Inc. All rights reserved.
The examples provided by Facebook are for non-commercial testing and evaluation
purposes only. Facebook reserves all rights not expressly granted.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
FACEBOOK BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
import UIKit
import IGListKit
class WorkingRangeSectionController: IGListSectionController, IGListSectionType, IGListWorkingRangeDelegate {
var height: Int?
var downloadedImage: UIImage?
var task: URLSessionDataTask?
var urlString: String? {
guard let height = height,
let width = collectionContext?.containerSize.width
else { return nil }
return "https://unsplash.it/" + width.description + "/" + height.description
}
deinit {
task?.cancel()
}
override init() {
super.init()
workingRangeDelegate = self
}
func numberOfItems() -> Int {
return 2
}
func sizeForItem(at index: Int) -> CGSize {
let width: CGFloat = collectionContext?.containerSize.width ?? 0
let height: CGFloat = CGFloat(index == 0 ? 55 : (self.height ?? 0))
return CGSize(width: width, height: height)
}
func cellForItem(at index: Int) -> UICollectionViewCell {
let cellClass: AnyClass = index == 0 ? LabelCell.self : ImageCell.self
let cell = collectionContext!.dequeueReusableCell(of: cellClass, for: self, at: index)
if let cell = cell as? LabelCell {
cell.label.text = urlString
} else if let cell = cell as? ImageCell {
cell.setImage(image: downloadedImage)
}
return cell
}
func didUpdate(to object: Any) {
self.height = object as? Int
}
func didSelectItem(at index: Int) {}
//MARK: IGListWorkingRangeDelegate
func listAdapter(_ listAdapter: IGListAdapter, sectionControllerWillEnterWorkingRange sectionController: IGListSectionController) {
guard downloadedImage == nil,
task == nil,
let urlString = urlString,
let url = URL(string: urlString)
else { return }
let section = collectionContext?.section(for: self) ?? 0
print("Downloading image \(urlString) for section \(section)")
task = URLSession.shared.dataTask(with: url, completionHandler: { data, response, err in
if let data = data, let image = UIImage(data: data) {
DispatchQueue.main.async {
self.downloadedImage = image
if let cell = self.collectionContext?.cellForItem(at: 1, sectionController: self) as? ImageCell {
cell.setImage(image: image)
}
}
} else {
print("Error downloading \(urlString): \(err)")
}
})
task?.resume()
}
func listAdapter(_ listAdapter: IGListAdapter, sectionControllerDidExitWorkingRange sectionController: IGListSectionController) {}
}