ios - 我试图在不使用 Storyboard 的情况下使用 swift (Xcode) 将标题部分添加到我的个人资料页面

标签 ios swift xcode

这是我到目前为止所做的,但它一直给我错误:

由于未捕获的异常“NSInternalInconsistencyException”而终止应用程序,原因:“无法使类型 View 出列:带有标识符 ProfileHeader 的 UICollectionElementKindSectionFooter - 必须为标识符注册一个 nib 或类或连接 Storyboard中的原型(prototype)单元格”

我将如何解决这个问题?

private let reuseIdentifier = "Cell"
private let headerIdentifier = "ProfileHeader"

class UserProfileVC: UICollectionViewController, UICollectionViewDelegateFlowLayout {

// MARK: Properties
var USER_NAME: String = "Username"

override func viewDidLoad() {
    super.viewDidLoad()
    collectionView.register(UINib(nibName: headerIdentifier, bundle: nil), forCellWithReuseIdentifier: reuseIdentifier)
    self.navigationItem.title = "Profile"

    self.collectionView!.register(UICollectionViewCell.self, forCellWithReuseIdentifier: reuseIdentifier)
    self.collectionView!.register(ProfileHeader.self, forSupplementaryViewOfKind: UICollectionView.elementKindSectionHeader, withReuseIdentifier: headerIdentifier)
    fetchUsersData()

}

// MARK: UICollectionViewDataSource

override func numberOfSections(in collectionView: UICollectionView) -> Int {
    // #warning Incomplete implementation, return the number of sections
    return 1
}


override func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
    // #warning Incomplete implementation, return the number of items
    return 0
}

func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, referenceSizeForFooterInSection section: Int) -> CGSize {

    return CGSize(width: view.frame.width, height: 200)
}


override func collectionView(_ collectionView: UICollectionView, viewForSupplementaryElementOfKind kind: String, at indexPath: IndexPath) -> UICollectionReusableView {

    //Declare header
    let header = collectionView.dequeueReusableSupplementaryView(ofKind: kind, withReuseIdentifier: headerIdentifier, for: indexPath) as! ProfileHeader
    return header
}

override func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
    let cell = collectionView.dequeueReusableCell(withReuseIdentifier: reuseIdentifier, for: indexPath)

    // Configure the cell

    return cell
}


// MARK - API

func fetchUsersData() {

    guard let currentUser = Auth.auth().currentUser?.uid else { return }
    print("Current user id is \(currentUser)")


    Database.database().reference().child("Users").child(currentUser).child(USER_NAME).observeSingleEvent(of: .value) { (snapshot) in
        guard let username = snapshot.value as? String else {return}
        self.navigationItem.title = username
    }
}
}

最佳答案

ViewForSupplementary为页眉和页脚 View 调用

func collectionView(_ collectionView: UICollectionView, viewForSupplementaryElementOfKind kind: String, at indexPath: IndexPath) -> UICollectionReusableView {
        if kind == UICollectionView.elementKindSectionHeader {
            let header = collectionView.dequeueReusableSupplementaryView(ofKind: kind, withReuseIdentifier: headerIdentifier, for: indexPath) as! ProfileHeader
            return header
        }
        else {
            fatalError("Unexpected element kind")
        }
    }

如果你不想要 viewForSupplementaryElement被要求设置页脚
extension UserProfileVC: UICollectionViewDelegate {
    func collectionView(_ collectionView: UICollectionView,
                          layout collectionViewLayout: UICollectionViewLayout,
                          referenceSizeForFooterInSection section: Int) -> CGSize {
        return CGSize.zero
    }
}

另请删除
collectionView.register(UINib(nibName: headerIdentifier, bundle: nil), forCellWithReuseIdentifier: reuseIdentifier)

这是注册 ProfileHeader作为单元格,在下一条语句中,您还将 ProfileHeader 注册为节标题 self.collectionView!.register(ProfileHeader.self, forSupplementaryViewOfKind: UICollectionView.elementKindSectionHeader, withReuseIdentifier: headerIdentifier)
编辑 2:

由于 OP 面临有趣的理解解决方案的困难,我在下面写全类
private let reuseIdentifier = "Cell"
private let headerIdentifier = "ProfileHeader"

class UserProfileVC: UICollectionViewController, UICollectionViewDelegateFlowLayout {

    // MARK: Properties
    var USER_NAME: String = "Username"

    override func viewDidLoad() {
        super.viewDidLoad()
        collectionView.register(UINib(nibName: headerIdentifier, bundle: nil), forCellWithReuseIdentifier: reuseIdentifier)
        self.navigationItem.title = "Profile"

        self.collectionView!.register(UICollectionViewCell.self, forCellWithReuseIdentifier: reuseIdentifier)
        self.collectionView!.register(ProfileHeader.self, forSupplementaryViewOfKind: UICollectionView.elementKindSectionHeader, withReuseIdentifier: headerIdentifier)
        fetchUsersData()

    }

    // MARK: UICollectionViewDataSource

    override func numberOfSections(in collectionView: UICollectionView) -> Int {
        // #warning Incomplete implementation, return the number of sections
        return 1
    }


    override func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
        // #warning Incomplete implementation, return the number of items
        return 0
    }

    func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, referenceSizeForFooterInSection section: Int) -> CGSize {

        return CGSize.zero
    }

    func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, referenceSizeForHeaderInSection section: Int) -> CGSize {
        return CGSize(width: view.frame.width, height: 200)
    }


    override func collectionView(_ collectionView: UICollectionView, viewForSupplementaryElementOfKind kind: String, at indexPath: IndexPath) -> UICollectionReusableView {

        if kind == UICollectionView.elementKindSectionHeader {
            let header = collectionView.dequeueReusableSupplementaryView(ofKind: kind, withReuseIdentifier: headerIdentifier, for: indexPath) as! ProfileHeader
            return header
        }
        else {
            fatalError("Unexpected element kind")
        }
    }

    override func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
        let cell = collectionView.dequeueReusableCell(withReuseIdentifier: reuseIdentifier, for: indexPath)

        // Configure the cell

        return cell
    }


    // MARK - API

    func fetchUsersData() {

        guard let currentUser = Auth.auth().currentUser?.uid else { return }
        print("Current user id is \(currentUser)")


        Database.database().reference().child("Users").child(currentUser).child(USER_NAME).observeSingleEvent(of: .value) { (snapshot) in
            guard let username = snapshot.value as? String else {return}
            self.navigationItem.title = username
        }
    }
}

关于ios - 我试图在不使用 Storyboard 的情况下使用 swift (Xcode) 将标题部分添加到我的个人资料页面,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62091337/

相关文章:

IOS:AVAudioPlayer管理内存

xcode - 在 Xcode 单元测试中使用 @testable 时为 "No such module"

ios - 在 Phonegap 生成的应用商店中部署 .ipa 文件?

ios - XLForm - 如何左对齐单元格?

通过 Safari 的 iOS SSO : Can't log out/switch user any more

ios - 如何使用点击手势 (iOS8/Swift) 在 pageViewController 上显示/隐藏状态栏

swift - addPeriodicTimeObserver 不是每毫秒调用一次?

swift - 如何修复 4 个水平标签的约束?

ios - 如何在 CALayers 上创建反向蒙版

iOS 'like' facebook应用页面,怎么办?