I have a table view linked to a NSFetchedResultController
(i.e. loading data and also tracking changes to data is bound to the FRC)
I'm not using AutoLayout
in my cells (due to huge performance drop it introduces in iOS 8). I'm laying out my cells' content manually in cells (using -(void)layoutSubviews
). plus, the height of rows are calculated based on content and are cached/invalidated properly.
If any condition related to my data changes, related cells get updated individually (with the whole -(void)controllerWillChangeContent:...
through -(void)controllerDidChangeContent:...
delegate methods implemented) the animation for row updates is: UITableViewRowAnimationNone
The problem here is that, My cells have transparent backgrounds, and I can see some visual noise (most likely the actual cell getting stretched vertically, I can't say for sure because they are really transient and short lived) during [self.tableView reloadRowsAtIndexPaths:...];
.
I have tried many things to no avail!
Here are the things I have tried that doesn't work:
-(CGFloat)tableView:(UITableView *)tableView estimatedHeightForRowAtIndexPath:(NSIndexPath *)indexPath
, returning the accurate value for estimated heights.self.tableView.rowHeight=UITableViewAutomaticDimension;
, helps a lot but doesn't fix it.[self.tableView beginUpdates];
and enabling after [self.tableView endUpdates];
-(void)tableView:(UITableView *)tableView didEndDisplayingCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath
Things that help a little:
UIKit
counterparts doesn't solve the problem.-(void)layoutSubviews
in my cells shows that the cells with update condition get reloaded 3 times in a row upon each update in contrast with other cells getting updated just the once. I'm not forcing any [cell layoutIfNeeded];
or [cell setNeedsLayout];
anywhere.[self.tableView reloadData];
is not an option for me.At the end, I'm sorry I can't share any actual code, because it's not a test project and the complexity of the implementations renders any code sharing futile unless comprehended as a whole.
Thanks for your time.
If anyone encountered the same problem:
If you are calculating your cell height and its elements and you are dependent on NSIndexPath
for querying any data or conditions, always check that your index path is valid (not nil
). Most of UITableView
methods that return index path (e.g -(NSIndexPath *)indexPathForCell:..
, etc.) may return nil
values and will shoot you down an spiral that is debugging views.
In my case, I had a method to determined if my cell should have a header, and for that I checked wether it is the first cell or a condition has changed since previous cell. the problem was indexPath.row == 0
is true even if the index path is actually nil
(it's a conceptual problem in Objective-C where nil
is 0). so for a brief moment my cells would thought it has a header!
I feel kind of stupid not checking for nil
index paths but I think sharing it might help somebody someday.