I've currently hit a wall while trying to develop the search view within my application. I've attempted to place a collection view within a table view cell, as well as use the traditional collection view data source delegates (cell for row at etc.) however either of these implementations seems to be far from optimal in a modern application. Our application is using three models for 'Friends', 'Users', and 'Venues'.
We've successfully got one of our models to populate in the search view using a Diffable Data Source and compositional layout, but when I try to add the others I get the following error:
""*** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'Invalid parameter not satisfying: section != NSNotFound' terminating with uncaught exception of type NSException""
Heres the SearchViewController
import UIKit
fileprivate typealias DataSource = UICollectionViewDiffableDataSource<Section, SearchV2VC.DataType>
fileprivate typealias SourceSnapshot = NSDiffableDataSourceSnapshot<Section, SearchV2VC.DataType>
class SearchV2VC: UIViewController {
var collectionView: UICollectionView! = nil
private var dataSource: DataSource!
var friendData = [FollowingInfo]()
var userData = [FriendInfo]()
var venueData = [Venue]()
override func viewDidLoad() {
super.viewDidLoad()
configureHierarchy()
configureDataSource()
}
private func firstLayoutSection() -> NSCollectionLayoutSection {
let itemSize = NSCollectionLayoutSize(widthDimension: .fractionalWidth(1), heightDimension: .fractionalHeight(0.5))
let item = NSCollectionLayoutItem(layoutSize: itemSize)
item.contentInsets.bottom = 15
let groupSize = NSCollectionLayoutSize(widthDimension: .fractionalWidth(0.9), heightDimension: .fractionalWidth(0.5))
let group = NSCollectionLayoutGroup.vertical(layoutSize: groupSize, subitems: [item])
group.contentInsets = .init(top: 0, leading: 15, bottom: 0, trailing: 2)
let section = NSCollectionLayoutSection(group: group)
//section.orthogonalScrollingBehavior = .groupPaging
section.orthogonalScrollingBehavior = .none
return section
}
private func secondLayoutSection() -> NSCollectionLayoutSection {
let itemSize = NSCollectionLayoutSize(widthDimension: .fractionalWidth(1), heightDimension: .fractionalHeight(0.5))
let item = NSCollectionLayoutItem(layoutSize: itemSize)
item.contentInsets.bottom = 15
let groupSize = NSCollectionLayoutSize(widthDimension: .fractionalWidth(0.9), heightDimension: .fractionalWidth(0.5))
let group = NSCollectionLayoutGroup.vertical(layoutSize: groupSize, subitems: [item])
group.contentInsets = .init(top: 0, leading: 15, bottom: 0, trailing: 2)
let section = NSCollectionLayoutSection(group: group)
//section.orthogonalScrollingBehavior = .groupPaging
section.orthogonalScrollingBehavior = .none
return section
}
private func thirdLayoutSection() -> NSCollectionLayoutSection {
let itemSize = NSCollectionLayoutSize(widthDimension: .fractionalWidth(1), heightDimension: .fractionalHeight(2))
let item = NSCollectionLayoutItem(layoutSize: itemSize)
item.contentInsets.bottom = 15
let groupSize = NSCollectionLayoutSize(widthDimension: .fractionalWidth(0.35), heightDimension: .fractionalWidth(0.35))
let group = NSCollectionLayoutGroup.horizontal(layoutSize: groupSize, subitems: [item])
group.contentInsets = .init(top: 0, leading: 15, bottom: 0, trailing: 0)
let section = NSCollectionLayoutSection(group: group)
navigationItem.title = "You a Bitch"
section.orthogonalScrollingBehavior = .continuous
return section
}
}
extension SearchV2VC {
private func createLayout() -> UICollectionViewCompositionalLayout {
return UICollectionViewCompositionalLayout { (sectionNumber, env) -> NSCollectionLayoutSection? in
switch sectionNumber {
case 0: return self.firstLayoutSection()
case 1: return self.secondLayoutSection()
default: return self.thirdLayoutSection()
}
}
}
}
//MARK: - UICollectionViewDataSource Methods
extension SearchV2VC {
private func configureHierarchy() {
//collectionView.delegate = self
collectionView = UICollectionView(frame: view.bounds, collectionViewLayout: createLayout())
collectionView.delegate = self
collectionView.autoresizingMask = [.flexibleWidth, .flexibleHeight]
collectionView.backgroundColor = .white
collectionView.register(UserCell.self, forCellWithReuseIdentifier: UserCell.reuseIdentifier)
collectionView.register(FriendCell.self, forCellWithReuseIdentifier: FriendCell.reuseIdentifier)
collectionView.register(VenueCell.self, forCellWithReuseIdentifier: VenueCell.reuseIdentifier)
view.addSubview(collectionView)
}
}
extension SearchV2VC {
func configureDataSource(){
dataSource = DataSource(collectionView: collectionView, cellProvider: { (collectionView, index, friend) -> UICollectionViewCell in
switch friend{
case .user(let user):
guard let cell = collectionView.dequeueReusableCell(withReuseIdentifier: UserCell.reuseIdentifier, for: index) as? UserCell else {fatalError("Couldn't Create New Cell")}
cell.viewModel = UserCellViewModel(user: user)
return cell
case .friend(let friend):
guard let cell = collectionView.dequeueReusableCell(withReuseIdentifier: FriendCell.reuseIdentifier, for: index) as? FriendCell else {fatalError("Couldn't Create New Cell")}
cell.viewModel = FriendCellViewModel(user: friend)
return cell
case .venue(let venue):
guard let cell = collectionView.dequeueReusableCell(withReuseIdentifier: VenueCell.reuseIdentifier, for: index) as? VenueCell else {fatalError("Couldn't Create New Cell")}
cell.viewModel = VenueCellViewModel(venue: venue)
return cell
}
})
dataSource.apply(snapshotForCurrentState(), animatingDifferences: false)
}
func snapshotForCurrentState() -> NSDiffableDataSourceSnapshot<Section, DataType>{
var snapshot = NSDiffableDataSourceSnapshot<Section, DataType>()
snapshot.appendSections(Section.allSections)
let userItems = userData.map { DataType.user($0) }
snapshot.appendItems(userItems, toSection: Section.users)
let friendItems = friendData.map { DataType.friend($0) }
snapshot.appendItems(friendItems, toSection: Section.friends)
let venueItems = venueData.map { DataType.venue($0) }
snapshot.appendItems(venueItems, toSection: Section.venue)
return snapshot
}
}
extension SearchV2VC: UICollectionViewDelegate {
enum DataType: Hashable {
case user(FriendInfo)
case friend(FollowingInfo)
case venue(Venue)
}
func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
collectionView.deselectItem(at: indexPath, animated: true)
guard let user = dataSource.itemIdentifier(for: indexPath) else { return }
print(user)
}
}
This is what our models look like:
import Foundation
import Firebase
import Mapbox
struct Venue: Hashable {
static func == (lhs: Venue, rhs: Venue) -> Bool {
lhs.venID == rhs.venID
}
func hash(into hasher: inout Hasher) {
hasher.combine(venID)
}
var venID: String?
var latitude: Double?
var longitude: Double?
var name: String?
var description: String?
var image: String?
var address: String?
}
struct FriendInfo: Hashable {
static func == (lhs: FriendInfo, rhs: FriendInfo) -> Bool {
lhs.userID == rhs.userID
}
func hash(into hasher: inout Hasher) {
hasher.combine(userID)
}
var displayName: String?
var userID: String?
var userName: String?
var providerID: String?
var profileImageURL: String?
var isFriend = false
var stats: UserStats?
var isCurrentUser: Bool { return Auth.auth().currentUser?.uid == providerID }
}
struct FollowingInfo: Hashable {
static func == (lhs: FollowingInfo, rhs: FollowingInfo) -> Bool {
lhs.userID == rhs.userID
}
func hash(into hasher: inout Hasher) {
hasher.combine(userID)
}
var displayName: String?
var userID: String?
var profileURL: String?
var userName: String?
}
Lastly here is the Custom Section Class we are using instead of the usual 'enum section' located in the controller, and this is most likely where the problem lies:
import UIKit
struct Section: Hashable {
var id = UUID()
var title: String
var data: [Any]
init(title: String, data: [Any]) {
self.title = title
self.data = data
}
func hash(into hasher: inout Hasher) {
hasher.combine(id)
}
static func == (lhs: Section, rhs: Section) -> Bool {
lhs.id == rhs.id
}
}
extension Section {
static var allSections: [Section] = [
Section(title: "Users", data: [FriendInfo]()),
Section(title: "Friends", data: [FollowingInfo]()),
Section(title: "Venue", data: [Venue]())
]
}
extension Section {
static var users: Section =
Section(title: "Users", data: [FriendInfo]())
}
extension Section {
static var friends: Section =
Section(title: "Friends", data: [FollowingInfo]())
}
extension Section {
static var venue: Section =
Section(title: "Venue", data: [Venue]())
}
Lastly, I should mention we are setting these 3 models with completely different Data via an API call to Firebase. (The data is operating as expected everywhere else in the application) The collection view cells are utilizing a view model hence the 'cell.viewModel = ...' seen in the view controller.
I've been trying to get this functioning as expected for several days now. I have watched every video from Apple, thoroughly inspected all the example projects, and visited nearly every link google has to offer on the topic.
This is my first stack question so I apologize if I missed any crucial information. Surely someone has implemented something like this design, I am open to any ways of achieving this. Thanks!