mirror of
https://github.com/Instagram/IGListKit
synced 2026-05-06 15:08:50 +00:00
Summary: Changed the criteria for starting search in the OS X Example. Closes https://github.com/Instagram/IGListKit/issues/371 Doesn't look like there are tests for the example app. Please ping me if you'd like me to add some. Also, please ping me if you'd like me to add this change to the changelog -- I'm assuming bug fixes to sample apps don't get included there. - [x] All tests pass. Demo project builds and runs. - [x] I added tests, an experiment, or detailed why my change isn't tested. - [x] I added an entry to the `CHANGELOG.md` for any breaking changes, enhancements, or bug fixes. - [x] I have reviewed the [contributing guide](https://github.com/Instagram/IGListKit/blob/master/.github/CONTRIBUTING.md) Closes https://github.com/Instagram/IGListKit/pull/377 Differential Revision: D4374006 Pulled By: jessesquires fbshipit-source-id: 6e20f61aece769240aa63c8949d3c06a0ecd26c8
150 lines
4.5 KiB
Swift
150 lines
4.5 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 Cocoa
|
|
import IGListKit
|
|
|
|
final class UsersViewController: NSViewController {
|
|
|
|
@IBOutlet weak var tableView: NSTableView!
|
|
|
|
//MARK: Data
|
|
|
|
var users = [User]() {
|
|
didSet {
|
|
computeFilteredUsers()
|
|
}
|
|
}
|
|
|
|
var searchTerm = "" {
|
|
didSet {
|
|
computeFilteredUsers()
|
|
}
|
|
}
|
|
|
|
private func computeFilteredUsers() {
|
|
guard !searchTerm.characters.isEmpty else {
|
|
filteredUsers = users
|
|
return
|
|
}
|
|
|
|
filteredUsers = users.filter({ $0.name.localizedCaseInsensitiveContains(self.searchTerm) })
|
|
}
|
|
|
|
fileprivate func delete(user: User) {
|
|
guard let index = self.users.index(where: { $0.pk == user.pk }) else { return }
|
|
|
|
self.users.remove(at: index)
|
|
}
|
|
|
|
//MARK: -
|
|
//MARK: Diffing
|
|
|
|
var filteredUsers = [User]() {
|
|
didSet {
|
|
// get the difference between the old array of Users and the new array of Users
|
|
let diff = IGListDiff(oldValue, filteredUsers, .equality)
|
|
|
|
// this difference is used here to update the table view, but it can be used
|
|
// to update collection views and other similar interface elements
|
|
// this code can also be added to an extension of NSTableView ;)
|
|
tableView.beginUpdates()
|
|
tableView.insertRows(at: diff.inserts, withAnimation: .slideDown)
|
|
tableView.removeRows(at: diff.deletes, withAnimation: .slideUp)
|
|
tableView.reloadData(forRowIndexes: diff.updates, columnIndexes: .zero)
|
|
diff.moves.forEach { move in
|
|
self.tableView.moveRow(at: move.from, to: move.to)
|
|
}
|
|
tableView.endUpdates()
|
|
}
|
|
}
|
|
|
|
//MARK: -
|
|
|
|
private func loadSampleUsers() {
|
|
guard let file = Bundle.main.url(forResource: "users", withExtension: "json") else { return }
|
|
|
|
do {
|
|
self.users = try UsersProvider(with: file).users
|
|
} catch {
|
|
NSAlert(error: error).runModal()
|
|
}
|
|
}
|
|
|
|
// MARK: Interface
|
|
|
|
override func viewDidLoad() {
|
|
super.viewDidLoad()
|
|
|
|
loadSampleUsers()
|
|
}
|
|
|
|
override func viewDidAppear() {
|
|
super.viewDidAppear()
|
|
|
|
view.window?.titleVisibility = .hidden
|
|
}
|
|
|
|
@IBAction func shuffle(_ sender: Any?) {
|
|
users = users.shuffled
|
|
}
|
|
|
|
@IBAction func search(_ sender: NSSearchField) {
|
|
searchTerm = sender.stringValue
|
|
}
|
|
|
|
@IBAction func delete(_ sender: Any?) {
|
|
guard tableView.selectedRowIndexes.count > 0 else { return }
|
|
|
|
tableView.selectedRowIndexes.forEach({ self.delete(user: self.filteredUsers[$0]) })
|
|
}
|
|
|
|
}
|
|
|
|
extension UsersViewController: NSTableViewDataSource {
|
|
|
|
func numberOfRows(in tableView: NSTableView) -> Int {
|
|
return filteredUsers.count
|
|
}
|
|
|
|
}
|
|
|
|
extension UsersViewController: NSTableViewDelegate {
|
|
|
|
private struct Storyboard {
|
|
static let cellIdentifier = "cell"
|
|
}
|
|
|
|
func tableView(_ tableView: NSTableView, viewFor tableColumn: NSTableColumn?, row: Int) -> NSView? {
|
|
guard let cell = tableView.make(withIdentifier: Storyboard.cellIdentifier, owner: tableView) as? NSTableCellView else {
|
|
return nil
|
|
}
|
|
|
|
cell.textField?.stringValue = filteredUsers[row].name
|
|
|
|
return cell
|
|
}
|
|
|
|
@available(OSX 10.11, *)
|
|
func tableView(_ tableView: NSTableView, rowActionsForRow row: Int, edge: NSTableRowActionEdge) -> [NSTableViewRowAction] {
|
|
let delete = NSTableViewRowAction(style: .destructive, title: "Delete") { action, row in
|
|
guard row < self.filteredUsers.count else { return }
|
|
|
|
self.delete(user: self.filteredUsers[row])
|
|
}
|
|
|
|
return [delete]
|
|
}
|
|
|
|
}
|