From 117fddc15ad38aec74af3167ca3b274e66bcb853 Mon Sep 17 00:00:00 2001 From: Simon Mitchell Date: Sun, 1 Jul 2018 11:59:14 +0100 Subject: [PATCH 1/3] Adds a function to call arbitrary code without redrawing the tableView. This allows for mutating `self.data` without redrawing the entire table view for example, and is a working point to being able to mutate `data` and only reload certain rows --- ThunderTable/TableViewController.swift | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/ThunderTable/TableViewController.swift b/ThunderTable/TableViewController.swift index 15881fd..ba11260 100644 --- a/ThunderTable/TableViewController.swift +++ b/ThunderTable/TableViewController.swift @@ -111,9 +111,18 @@ open class TableViewController: UITableViewController { private var _data: [Section] = [] + private var _isBlockingRedrawing: Bool = false + + public func withoutRedrawing(_ closure: () -> Void) { + _isBlockingRedrawing = true + closure() + _isBlockingRedrawing = false + } + open var data: [Section] { set { _data = newValue + guard !_isBlockingRedrawing else { return } tableView.reloadData() } get { From 5265b9997c7e5dcc9ad34284c60cab98755c093c Mon Sep 17 00:00:00 2001 From: Simon Mitchell Date: Tue, 17 Jul 2018 10:28:15 +0100 Subject: [PATCH 2/3] Adds functions to reload particular rows rather than the whole TableView --- ThunderTable.xcodeproj/project.pbxproj | 4 + ThunderTable/TableViewController+Reload.swift | 109 ++++++++++++++++++ ThunderTable/TableViewController.swift | 3 + 3 files changed, 116 insertions(+) create mode 100644 ThunderTable/TableViewController+Reload.swift diff --git a/ThunderTable.xcodeproj/project.pbxproj b/ThunderTable.xcodeproj/project.pbxproj index a5794d7..03d90ae 100644 --- a/ThunderTable.xcodeproj/project.pbxproj +++ b/ThunderTable.xcodeproj/project.pbxproj @@ -60,6 +60,7 @@ B1EC81031FDE86BF00C8EE72 /* SubtitleTableViewCell.xib in Resources */ = {isa = PBXBuildFile; fileRef = B1EC81011FDE86BF00C8EE72 /* SubtitleTableViewCell.xib */; }; B1EC81061FDE873700C8EE72 /* Value1TableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = B1EC81041FDE873700C8EE72 /* Value1TableViewCell.swift */; }; B1EC81071FDE873700C8EE72 /* Value1TableViewCell.xib in Resources */ = {isa = PBXBuildFile; fileRef = B1EC81051FDE873700C8EE72 /* Value1TableViewCell.xib */; }; + B1F8047F20FDEC5200921AC9 /* TableViewController+Reload.swift in Sources */ = {isa = PBXBuildFile; fileRef = B1F8047E20FDEC5200921AC9 /* TableViewController+Reload.swift */; }; /* End PBXBuildFile section */ /* Begin PBXContainerItemProxy section */ @@ -143,6 +144,7 @@ B1EC81011FDE86BF00C8EE72 /* SubtitleTableViewCell.xib */ = {isa = PBXFileReference; lastKnownFileType = file.xib; path = SubtitleTableViewCell.xib; sourceTree = ""; }; B1EC81041FDE873700C8EE72 /* Value1TableViewCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Value1TableViewCell.swift; sourceTree = ""; }; B1EC81051FDE873700C8EE72 /* Value1TableViewCell.xib */ = {isa = PBXFileReference; lastKnownFileType = file.xib; path = Value1TableViewCell.xib; sourceTree = ""; }; + B1F8047E20FDEC5200921AC9 /* TableViewController+Reload.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "TableViewController+Reload.swift"; sourceTree = ""; }; /* End PBXFileReference section */ /* Begin PBXFrameworksBuildPhase section */ @@ -253,6 +255,7 @@ B17BAA361D89639100844421 /* Info.plist */, B1EC80FB1FDE85C000C8EE72 /* Cells */, B17BAA551D8963BB00844421 /* TableViewController.swift */, + B1F8047E20FDEC5200921AC9 /* TableViewController+Reload.swift */, B17BAA571D89643800844421 /* TableSection.swift */, B17BAA591D89645000844421 /* TableRow.swift */, B114F0C220359F76005D52F2 /* TableRow+Actions.swift */, @@ -506,6 +509,7 @@ B17BAA581D89643800844421 /* TableSection.swift in Sources */, B1784D831D8C3A60007358EA /* InputSliderRow.swift in Sources */, B1C2C56C1FFCEE3100D968C5 /* InputDatePickerRow.swift in Sources */, + B1F8047F20FDEC5200921AC9 /* TableViewController+Reload.swift in Sources */, B1EC81021FDE86BF00C8EE72 /* SubtitleTableViewCell.swift in Sources */, B1B679611D89807A00B66FD8 /* TableViewCell.swift in Sources */, B1784D861D8C3A8A007358EA /* InputSliderViewCell.swift in Sources */, diff --git a/ThunderTable/TableViewController+Reload.swift b/ThunderTable/TableViewController+Reload.swift new file mode 100644 index 0000000..8a1e085 --- /dev/null +++ b/ThunderTable/TableViewController+Reload.swift @@ -0,0 +1,109 @@ +// +// TableViewController+Reload.swift +// ThunderTable +// +// Created by Simon Mitchell on 17/07/2018. +// Copyright © 2018 3SidedCube. All rights reserved. +// + +import Foundation + +extension TableViewController { + + /// Finds the index path for a particular row. + /// + /// - Note: The row in question must conform to Equatable for this to succeed. + /// + /// - Parameter row: The row to find the index path for. + /// - Returns: The index path the row is positioned at if it is in `data`. + public func indexPathFor(row: T) -> IndexPath? { + + for (index, section) in data.enumerated() { + + guard let matchingRowIndex = section.rows.index(where: { + guard let matchableRow = $0 as? T else { return false } + return matchableRow == row + }) else { + continue + } + + return IndexPath(row: matchingRowIndex, section: index) + } + + return nil + } + + /// Replaces a row in `data` with another row, whilst optionally reloading other index paths. + /// + /// - Note: This relies on the containing section of the row being of class `TableSection` as apposed to any object conforming to `Section` + /// + /// - Parameters: + /// - row: The row that should be replaced. + /// - otherRow: The row that is replacing the original row. + /// - additionalReloadIndexPaths: Additional index paths that should be reloaded at the same time. + public func replace(row: T, with otherRow: Row, reloading additionalReloadIndexPaths: [IndexPath] = []) { + + guard let indexPath = indexPathFor(row: row) else { return } + guard let tableSection = data[indexPath.section] as? TableSection else { return } + + var rows = tableSection.rows + rows[indexPath.row] = otherRow + tableSection.rows = rows + + withoutRedrawing { + data[indexPath.section] = tableSection + } + + var indexPaths = [indexPath] + indexPaths.append(contentsOf: additionalReloadIndexPaths) + tableView.reloadRows(at: indexPaths, with: .none) + } + + /// Replaces multiple rows with replacement rows. + /// + /// - Note: This relies on the containing section of each row being of class `TableSection` as apposed to any object conforming to `Section`. + /// + /// - Parameters: + /// - rows: The rows to replace. + /// - otherRows: The rows they should be replaced by. + public func replace(rows: [T], with otherRows: [Row]) { + + guard rows.count == otherRows.count else { return } + + let replacement: [(Int, IndexPath)] = rows.enumerated().compactMap({ + guard let indexPath = indexPathFor(row: $0.element) else { return nil } + return ($0.offset, indexPath) + }) + + replacement.forEach { (index, indexPath) in + + guard let tableSection = data[indexPath.section] as? TableSection else { return } + + var rows = tableSection.rows + rows[indexPath.row] = otherRows[index] + tableSection.rows = rows + + withoutRedrawing { + data[indexPath.section] = tableSection + } + } + + tableView.reloadRows(at: replacement.map({ $0.1 }), with: .none) + } + + /// Reloads the cell at the indexPath for a given row. + /// + /// - Parameter row: The row to reload the cell for. + public func redraw(row: T) { + + guard let indexPath = indexPathFor(row: row) else { return } + tableView.reloadRows(at: [indexPath], with: .none) + } + + /// The last available indexPath in the tableView + public var lastIndexPath: IndexPath? { + guard let lastSection = data.last else { return nil } + guard !lastSection.rows.isEmpty else { return nil } + return IndexPath(row: lastSection.rows.count - 1, section: data.count - 1) + } +} diff --git a/ThunderTable/TableViewController.swift b/ThunderTable/TableViewController.swift index ba11260..f33949c 100644 --- a/ThunderTable/TableViewController.swift +++ b/ThunderTable/TableViewController.swift @@ -113,6 +113,9 @@ open class TableViewController: UITableViewController { private var _isBlockingRedrawing: Bool = false + /// A function which allows for mutation of `data` without causing the tableView to reload. + /// + /// - Parameter closure: Code to be run without reloading the table view. public func withoutRedrawing(_ closure: () -> Void) { _isBlockingRedrawing = true closure() From 8193f7aa9381154b9d650141fac534781ae55503 Mon Sep 17 00:00:00 2001 From: Simon Mitchell Date: Thu, 30 May 2019 09:17:09 +0100 Subject: [PATCH 3/3] Improves reload functions, and adds optional animation properties --- .../contents.xcworkspacedata | 7 ++++++ .../xcshareddata/IDEWorkspaceChecks.plist | 8 ++++++ ThunderTable/TableViewController+Reload.swift | 25 +++++++++++-------- 3 files changed, 29 insertions(+), 11 deletions(-) create mode 100644 ThunderTable.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 ThunderTable.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist diff --git a/ThunderTable.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/ThunderTable.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 0000000..919434a --- /dev/null +++ b/ThunderTable.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/ThunderTable.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/ThunderTable.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 0000000..18d9810 --- /dev/null +++ b/ThunderTable.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/ThunderTable/TableViewController+Reload.swift b/ThunderTable/TableViewController+Reload.swift index 8a1e085..7266041 100644 --- a/ThunderTable/TableViewController+Reload.swift +++ b/ThunderTable/TableViewController+Reload.swift @@ -6,7 +6,7 @@ // Copyright © 2018 3SidedCube. All rights reserved. // -import Foundation +import UIKit extension TableViewController { @@ -20,7 +20,7 @@ extension TableViewController { for (index, section) in data.enumerated() { - guard let matchingRowIndex = section.rows.index(where: { + guard let matchingRowIndex = section.rows.firstIndex(where: { guard let matchableRow = $0 as? T else { return false } return matchableRow == row }) else { @@ -41,7 +41,8 @@ extension TableViewController { /// - row: The row that should be replaced. /// - otherRow: The row that is replacing the original row. /// - additionalReloadIndexPaths: Additional index paths that should be reloaded at the same time. - public func replace(row: T, with otherRow: Row, reloading additionalReloadIndexPaths: [IndexPath] = []) { + /// - animation: The animation to use when reloading the rows + public func replace(row: T, with otherRow: Row, reloading additionalReloadIndexPaths: [IndexPath] = [], animation: UITableView.RowAnimation = .none) { guard let indexPath = indexPathFor(row: row) else { return } guard let tableSection = data[indexPath.section] as? TableSection else { return } @@ -56,7 +57,7 @@ extension TableViewController { var indexPaths = [indexPath] indexPaths.append(contentsOf: additionalReloadIndexPaths) - tableView.reloadRows(at: indexPaths, with: .none) + tableView.reloadRows(at: indexPaths, with: animation) } /// Replaces multiple rows with replacement rows. @@ -66,7 +67,8 @@ extension TableViewController { /// - Parameters: /// - rows: The rows to replace. /// - otherRows: The rows they should be replaced by. - public func replace(rows: [T], with otherRows: [Row]) { + /// - animation: The row animation to use when replacing + public func replace(rows: [T], with otherRows: [Row], animation: UITableView.RowAnimation) { guard rows.count == otherRows.count else { return } @@ -88,22 +90,23 @@ extension TableViewController { } } - tableView.reloadRows(at: replacement.map({ $0.1 }), with: .none) + tableView.reloadRows(at: replacement.map({ $0.1 }), with: animation) } /// Reloads the cell at the indexPath for a given row. /// - /// - Parameter row: The row to reload the cell for. - public func redraw(row: T) { + /// - Parameters: + /// - row: The row to reload the cell for. + /// - animation: The animation to use when redrawing + public func redraw(row: T, with animation: UITableView.RowAnimation = .none) { guard let indexPath = indexPathFor(row: row) else { return } - tableView.reloadRows(at: [indexPath], with: .none) + tableView.reloadRows(at: [indexPath], with: animation) } /// The last available indexPath in the tableView public var lastIndexPath: IndexPath? { - guard let lastSection = data.last else { return nil } - guard !lastSection.rows.isEmpty else { return nil } + guard let lastSection = data.last(where: { !$0.rows.isEmpty }) else { return nil } return IndexPath(row: lastSection.rows.count - 1, section: data.count - 1) } }