UICollection NumberOfItemsInSections在向vars添加常量时出现问题

时间:2017-01-02 04:02:22

标签: ios swift uicollectionview

我有两个不同的单元格类型,我试图使用PayNowCell类创建第一个索引路径,让它保持显示。而其他单元格属于CheckOutCell类。现在问题出在我的func numberofitemsinsection上。目前我正在使用return checkout.count但是在加载视图时它缺少PayNowCell。如果我让它返回checkout.count + 1,总是有PayNowCell可用;我的程序崩溃给我错误索引超出范围。数组结帐是全局变量。有人可以解释原因并提供修复吗?被困在这一段时间了。代码如下。

class CheckoutController: UICollectionViewCell, UICollectionViewDataSource, UICollectionViewDelegate, UICollectionViewDelegateFlowLayout  {

    var inventoryTabController: InventoryTabController?

    override init(frame: CGRect){
        super.init(frame: frame)
        setupViews()

        NotificationCenter.default.addObserver(forName: .arrayValueChanged, object: nil, queue: OperationQueue.main) { [weak self] (notif) in
            self?.collectionView.reloadData()
        }

    }

    deinit {
        NotificationCenter.default.removeObserver(self)
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

    func loadItems() -> [Item]? {
        return NSKeyedUnarchiver.unarchiveObject(withFile: Item.ArchiveURL.path) as? [Item]
    }

    func saveItems() {
        let isSuccessfulSave = NSKeyedArchiver.archiveRootObject(checkout, toFile: Item.ArchiveURL.path)
        if !isSuccessfulSave {
            print("Failed to save items...")
        }
    }

    func addItem(item: Item) {
        items.append(item)
        collectionView.reloadData()
    }

    func editItem(item: Item, index: Int) {
        items[index] = item
        collectionView.reloadData()
    }

    func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        // Identify which segue is occuring.
        if segue.identifier == "ShowDetail" {
            let itemDetailViewController = segue.destination as! AddItemController

            // Get the cell that generated this segue.
            if let selectedItemCell = sender as? InventoryCell {
                let indexPath = collectionView.indexPath(for: selectedItemCell)!
                let selectedItem = items[indexPath.row]
                itemDetailViewController.item = selectedItem
            }
        }
        else if segue.identifier == "AddItem" {
            print("Adding new meal.")
        }
    }

    lazy var collectionView: UICollectionView = {
        let layout = UICollectionViewFlowLayout()
        let cv = UICollectionView(frame: .zero, collectionViewLayout: layout)

        cv.dataSource = self
        cv.delegate = self

        cv.backgroundColor = UIColor.rgb(r: 247, g: 247, b: 247)
        return cv
    }()

    let cellId = "cellId"
    let paynow = "paynow"
    func setupViews() {
        backgroundColor = .brown

        addSubview(collectionView)

        addConstraintsWithFormat("H:|[v0]|", views: collectionView)
        addConstraintsWithFormat("V:|[v0]|", views: collectionView)
        collectionView.indicatorStyle = UIScrollViewIndicatorStyle.white

        collectionView.register(PayNowCell.self, forCellWithReuseIdentifier: paynow)

        collectionView.register(CheckoutCell.self, forCellWithReuseIdentifier: cellId)
    }

    func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
        return checkout.count //init number of cells
    }

    func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {

        if indexPath.item == 0 {
        let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "paynow", for: indexPath) as! PayNowCell //init cells
        return cell
        }else{
        let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "cellId", for: indexPath) as! CheckoutCell //init cells
        print("Printing this \(checkout.count)")
        cell.item = checkout[indexPath.item]
        return cell
        }

    }

    func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
        //let item = items[indexPath.item]
        //inventoryController?.showItemDetailForItem(item: item, index: indexPath.item)
        print("selected")
        print(indexPath.item)
        collectionView.reloadData()
    }

    func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, sizeForItemAt indexPath: IndexPath) -> CGSize {
        if indexPath.item == 0 {
            return CGSize(width:frame.width, height: 100) //each cell dimensions
        }else{
            return CGSize(width:frame.width, height: 150) //each cell dimensions
        }
    }

    func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, minimumLineSpacingForSectionAt section: Int) -> CGFloat {
        return 0
    }

2 个答案:

答案 0 :(得分:1)

最好将您的收藏视图重新排列为2个部分:

  1. 第0部分:PayNow单元格(仅1个单元格)
  2. 第1部分:结帐单元格(使用结帐数组列表)
  3. 然后,您不会对indexPath.item问题感到困惑。

    func numberOfSections(in collectionView: UICollectionView) -> Int {
        return 2
    }
    
    func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
        return section == 0 ? 1 : checkout.count
    }
    
    func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
    
        if indexPath.section == 0 {
            let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "paynow", for: indexPath) as! PayNowCell //init cells
            return cell
        } else {
            let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "cellId", for: indexPath) as! CheckoutCell //init cells
            print("Printing this \(checkout.count)")
            cell.item = checkout[indexPath.item]
            return cell
        }
    
    }
    

答案 1 :(得分:0)

您需要从cellForRow功能中减去1。由于您要添加PayNowCell,因此您要添加一个额外的indexPath,但是目前,您正在使用dataSource函数给出的默认indexPath。该索引将始终高于您的项目数。通过从indexPath中减去1,您将与itemsArray同步,同时考虑到PayNowCell。

cell.item = checkout[indexPath.item - 1]