I have a UICollectionView that I have created programmatically. I would like for the collection view to behave in the following way:
1. User touches cell
2. Cell background color changes
3. User releases touch
4. Cell background color changes
This should be a quick color change that happens just before the selector related to the tap action is executed in which the viewcontroller containing the collection view is popped off the stack.
I have been looking at this question: UICollectionView cell change background while tap
in which there is the following summary of methods to use for this purpose:
// Methods for notification of selection/deselection and highlight/unhighlight events.
// The sequence of calls leading to selection from a user touch is:
//
// (when the touch begins)
// 1. -collectionView:shouldHighlightItemAtIndexPath:
// 2. -collectionView:didHighlightItemAtIndexPath:
//
// (when the touch lifts)
// 3. -collectionView:shouldSelectItemAtIndexPath: or - collectionView:shouldDeselectItemAtIndexPath:
// 4. -collectionView:didSelectItemAtIndexPath: or -collectionView:didDeselectItemAtIndexPath:
// 5. -collectionView:didUnhighlightItemAtIndexPath:
I am assuming I only need to implement one of the above methods from 'when touch begins' and 'when touch ends.' But no matter what I do, it appears that a background color changes and then remains changed. Here is an example of something I attempted which did not work:
- (void)collectionView:(UICollectionView *)collectionView didSelectItemAtIndexPath:(NSIndexPath *)indexPath
{
//pop vc
}
- (void)collectionView:(UICollectionView *)collectionView didHighlightItemAtIndexPath:(NSIndexPath *)indexPath
{
UICollectionViewCell* cell = [collectionView cellForItemAtIndexPath:indexPath];
cell.contentView.backgroundColor = [UIColor redColor];
}
- (void)collectionView:(UICollectionView *)collectionView didDeselectItemAtIndexPath:(NSIndexPath *)indexPath
{
UICollectionViewCell *cell = [collectionView cellForItemAtIndexPath:indexPath];
cell.contentView.backgroundColor = [UIColor greenColor];
}
This results in the cell background color being changed only to red. I also looked at this question: UICollectionView Select and Deselect issue and tried implementing [UICollectionView selectItemAtIndexPath:animated:scrollPosition:] and calling it inside of didSelectItemAtIndexPath, but this did not work either. Collection view data source and delegate are set.
Simple binary logic solution. Works with Swift 3 and 4:
Swift 3 version
Add the following two methods to your view controller class:
See here for help in setting up a basic collection view in Swift.
The problem is that you are changing the color on highlight and changing it back on deselect instead that on unhighlight
You should simply change this:
to this:
Also, if you don't want to wait a bit before getting your highlight happen you should set the
delaysContentTouches
property of the collection view toNO
Edit: also ensure that you call
inside the -didSelectItemAtIndexPath method
Here is my solution. And I'm sure it really works.
I provide three methods to highlight a cell (selectedBackgroundView, tint
cell.contentView
and tint a special area).How to use:
1. just inherit
BaseCollectionViewCell
and do nothing;2. inherit and set
specialHighlightedArea = UIView()
, andcontentView.addSubView(specialHighlightedArea)
, then layout it or add constraint to use Auto Layout;3. if you don't need highlight effect, just write a method named 'shouldHighlightItemAtIndexPath' defined by
UICollectionViewDelegate
and make it return false, or setcell.shouldTintBackgroundWhenSelected = false
and setspecialHighlightedArea = nil
and remove it from superView.Edit: Answer in Swift 3