Auto scrolling to a cell with a specific value
Solution 1
You can use find the index of the item(which will be its row), and then scroll to that index.
find
function returns the index of a particular element in the array.
if let index = find(items, groupNoToScroll)
{
let indexPath = NSIndexPath(forRow: index, inSection: 0)
tableView.scrollToRowAtIndexPath(indexPath, atScrollPosition: .Top, animated: true)
}
Solution 2
Swift 4:
let indexPath = IndexPath(row: row, section: section)
tableView.scrollToRow(at: indexPath, at: .top, animated: true)
(first, of course, you have to assign values to row and section based on which cell you want to scroll to)
Solution 3
Swift 3.0
let indexPath = NSIndexPath(forRow: 5, inSection: 0)
tableView.scrollToRow(at: indexPath, at: .top, animated: true)
Swift 4.0
let lastRowIndex = self.tblRequestStatus!.numberOfRows(inSection: 0) - 1
let pathToLastRow = IndexPath.init(row: lastRowIndex, section: 0)
tableView.scrollToRow(at: pathToLastRow, at: .none, animated: false)
Solution 4
Do this to find the first element in the array that is equal to groupNoToScroll. Then, go to that row.
var rowToGoTo:Int = 0 //Rather use the Swift find function.
for x in items{
if x == groupNoToScroll{
break
}
rowToGoTo++
}
let lastRow = tableView.indexPathsForVisibleRows()?.last as NSIndexPath
if indexPath.row == lastRow.row {
if scrollToTime == true {
let indexPath = NSIndexPath(row: rowToGoTo, section: 0)
tblBusList.scrollToRow(at: indexPath as IndexPath, at: .top, animated: true)
}
}
But I would recommend doing this in viewDidAppear.
As Isuru pointed out rather use the find function.
Related videos on Youtube
Isuru
Started out as a C# developer. Turned to iOS in 2012. Currently learning SwiftUI. Loves fiddling with APIs. Interested in UI/UX. Want to try fiddling with IoT. Blog | LinkedIn
Updated on July 17, 2022Comments
-
Isuru almost 2 years
I have a list of numbers in a table view like this.
As you can see the numbers are repeated. Let's consider a set of repeated numbers as a group. So there are group of 1s, group of 2s and so on.
What I want to do is when the app starts, I need to scroll to the start position of a specified group automatically. Before explaining further, here is my code so far.
import UIKit class TableViewController: UITableViewController, UITableViewDataSource, UITableViewDelegate { private var scrollToTime = true private var items = [Int]() private var groupNoToScroll = 12 override func viewDidLoad() { super.viewDidLoad() items = [1, 1, 2, 2, 2, 3, 4, 4, 4, 4, 4, 5, 5, 6, 7, 7, 8, 8, 8, 9, 10, 10, 10, 11, 11, 11, 11, 12, 12, 12, 12, 12, 12, 13, 13, 14, 14, 14, 14, 15, 15, 16, 17, 17, 18, 18, 18, 19, 19, 19, 19, 20, 21, 22, 22, 23, 23, 23] } // MARK: - UITableViewDataSource override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int { return items.count } override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell { let cell = tableView.dequeueReusableCellWithIdentifier("Cell", forIndexPath: indexPath) as UITableViewCell cell.textLabel?.text = String(items[indexPath.row]) return cell } // MARK: - UITableViewDelegate override func tableView(tableView: UITableView, willDisplayCell cell: UITableViewCell, forRowAtIndexPath indexPath: NSIndexPath) { let lastRow = tableView.indexPathsForVisibleRows()?.last as NSIndexPath if indexPath.row == lastRow.row { if scrollToTime == true { let indexPath = NSIndexPath(forRow: groupNoToScroll, inSection: 0) tableView.scrollToRowAtIndexPath(indexPath, atScrollPosition: .Top, animated: true) scrollToTime = false } } } }
I've assigned the value 12 for the variable
groupNoToScroll
. This means when the app starts I want the table view to be auto scrolled to the start of the cells of the 12s group.But currently what my code does is scroll to the 12th cell, not the cell that has the value 12. My question is how can I check with the cells' values and scroll to the number which I specify?
-
Hermann Klecker over 9 yearsYou will have to iterate through your array and get the index of the first appearance of
groupNoToScroll
. -
Hermann Klecker over 9 yearsHowever, it is generally not a good idea overriding tableView:willDisplayCell: . What are you really up to?
-
-
Isuru over 9 yearsThank you for the answer. While the logic of your answer is identical to @rakeshbs's answer, I accepted his because of usage of the Swift function
find
which I thought a more elegant and a cleaner way of doing it. But thanks again :) -
Isuru over 9 yearsWow that's a neat function. Thank you so much. That's surprisingly simple and cleaner syntax. Very cool.
-
Petri over 9 yearsThat makes perfect sense, I would too. I forgot about the find function.
-
rakeshbs over 9 yearsI don't agree with using the
willDisplayCell
function though. I would do it after callingreloadData
. stackoverflow.com/questions/16071503/… -
Hermann Klecker over 9 yearsif lastRow.row is the very row that corresponds to the actual index, which may be the case when the size of each group is one, then you run into an infinite loop here. scrollToRowAtIndexPath can cause willDisplayCell to be invoked.
-
Isuru over 9 years@HermannKlecker Okay, I'm doing the auto scrolling in the
viewDidAppear
method because I want the scrolling to happen every time when I leave this view and come back. -
Isuru over 9 years@rakeshbs Is it necessary to call
reloadData()
every time before doing the auto scroll? If so can you please tell me the reason? -
rakeshbs over 9 yearsNo I didn't say so. I just said do it after loading the data into the
UITableView
. Since you are doing it inviewDidAppear
make sure that theUITableView
has finished loading the data before calling the scroll function. -
Isuru over 9 yearsHello again, I'm facing a small situation related to this solution. If the
groupNoToScroll
I specify doesn't exist in theitems
array. So it doesn't scroll. Is there a way to make it like if the specified value doesn't exist, scroll to the next closest value? -
rakeshbs over 9 yearsyou have to define a function to find the next closest value, and then scroll to it. there is no built in way to do that