带有约束的嵌套集合视图的意外行为(Swift 4) [英] Unexpected behavior with nested collection views with constraints (Swift 4)

查看:17
本文介绍了带有约束的嵌套集合视图的意外行为(Swift 4)的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我在一个包含水平分页集合视图的表格视图中有一个单元格.

I have a cell in a tableview that contains a horizontal paging collection view.

在此集合视图的每个页面内都有一个垂直集合视图.

Inside each page of this collection view there is a vertical collection view.

为了避免scroll-in-scroll"问题,我在垂直集合视图中禁用了垂直滚动.

In order to avoid a "scroll-in-scroll" problem, I disabled vertical scrolling in the vertical collection views.

垂直集合视图的单元格计数不是静态的,可以是任意数字.

The vertical collection view's cell count is not static and can be any number.

所以这会产生一个问题:集合视图(您必须询问我以澄清哪个)不再适合表格视图的单元格.

So this creates a problem: the collection view (you'll have to ask me for clarification about which one) won't fit properly into the table view's cell anymore.

为了解决这个问题,我通过更改情节提要中的约束来更改表格视图的单元格高度......但问题仍然存在.

To solve this, I changed the table view's cell height by changing the constraints in the storyboard... but the problem remains.

这里有一张图片来帮助说明问题:

Here is an image to help illustrate the problem:

这是我的代码:

class myTBViewController: UITableViewController , UICollectionViewDelegate , UICollectionViewDataSource , UICollectionViewDelegateFlowLayout {
func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
    if collectionView.tag == 10 {
    return 2
    } else {
        return 30
    }
}


func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
    if collectionView.tag == 10 {
    let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "mainPages", for: indexPath) as! mainPages

        cell.backgroundColor = UIColor.yellow
        cell.listCV.clipsToBounds = true
    return cell
    } else {

        let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "showAdvCell", for: indexPath) as! showAdvCell

        cell.backgroundColor = UIColor.red

        return cell

    }
}

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

    if collectionView.tag == 10 {
    let collectionViewHeight = collectionView.frame.height
    let itemsHeight        = collectionView.contentSize.height

    let topInset = ( collectionViewHeight - itemsHeight ) / 4

    return UIEdgeInsetsMake(topInset ,0, 0 , 0)

    } else {

        let collectionViewHeight = collectionView.frame.height
        let itemsHeight        =  CGFloat(80)

        let topInset = ( collectionViewHeight - itemsHeight ) / 4

        return UIEdgeInsetsMake(topInset ,0, 0 , 0)


    }
}

var headerSegment = UISegmentedControl()

override func viewDidLoad() {
    super.viewDidLoad()

    headerSegment.frame = CGRect(x: 0, y: 200, width: UIScreen.main.bounds.width, height: 30)
    let items = ["فروشگاه ها", "دیوار"]
    headerSegment = UISegmentedControl(items: items)
    headerSegment.selectedSegmentIndex = 0
    headerSegment.addTarget(self, action: #selector(selectPage), for: UIControlEvents.valueChanged)
    let frame = CGRect(x: 0, y: 0, width: UIScreen.main.bounds.width, height: 200)
    let headerImageView = UIImageView(frame: frame)
    let image: UIImage = UIImage(named: "1")!
    headerImageView.image = image

    let frame3 = CGRect(x: 0, y: 0, width: 80, height: 80)
    let headerImageView2 = UIImageView(frame: frame3)
    let image2: UIImage = UIImage(named: "1")!
    headerImageView2.image = image2
    headerSegment.addSubview(headerImageView2)
    headerSegment.bringSubview(toFront: headerImageView2)
    headerImageView2.center = CGPoint(x: UIScreen.main.bounds.width / 2, y: 200)
    headerSegment.frame = CGRect(x: 0, y: 200, width: UIScreen.main.bounds.width + 10 , height: 40)
    headerSegment.center = CGPoint(x: UIScreen.main.bounds.width / 2, y: 200)
    let mainView = UIView(frame: CGRect(x: 0, y: 0, width: UIScreen.main.bounds.width, height: 230))
    let headerSegmentView = UIView(frame : CGRect(x: 0, y: 200, width: UIScreen.main.bounds.width, height: 40))
    headerSegmentView.center = headerSegment.center
    headerSegmentView.backgroundColor = UIColor.white
    mainView.addSubview(headerImageView)
    mainView.bringSubview(toFront: headerImageView)
    mainView.addSubview(headerSegmentView)
    mainView.bringSubview(toFront: headerSegmentView)
    mainView.addSubview(headerSegment)
    mainView.bringSubview(toFront: headerSegment)
    mainView.addSubview(headerImageView2)
    mainView.bringSubview(toFront: headerImageView2)
    self.tableView.tableHeaderView = mainView

    headerImageView2.layer.borderWidth = 2
    headerImageView2.layer.masksToBounds = false
    headerImageView2.layer.borderColor = UIColor.init(white: 255/255, alpha: 0.3).cgColor
    headerImageView2.layer.cornerRadius = headerImageView2.frame.height/2
    headerImageView2.clipsToBounds = true

}

@objc func selectPage() {

    let cell = self.tableView.dequeueReusableCell(withIdentifier: "myTBViewCell") as! myTBViewCell

    switch headerSegment.selectedSegmentIndex {
    case 0:
        print("0")
        cell.myTBCollectionView.scrollToItem(at:IndexPath(item: 0, section: 0), at: .top, animated: false)
    case 1:
        print("1")
        cell.myTBCollectionView.scrollToItem(at:IndexPath(item: 1, section: 0), at: .top, animated: false)
    default:
        print("0")
        cell.myTBCollectionView.scrollToItem(at:IndexPath(item: 0, section: 0), at: .top, animated: false)

    }

}



override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
}
override func numberOfSections(in tableView: UITableView) -> Int {
    // #warning Incomplete implementation, return the number of sections
    return 1
}

override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    // #warning Incomplete implementation, return the number of rows
    return 1
}


override func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {

    tableView.estimatedRowHeight = 200
    tableView.rowHeight = UITableViewAutomaticDimension

    return 600
}

override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = self.tableView.dequeueReusableCell(withIdentifier: "myTBViewCell", for: indexPath) as! myTBViewCell

    return cell
}


}

推荐答案

我也有类似的问题,最后通过创建CollectionView的高度约束解决了.

I have a similar Problem , finally, I have solved it by creating a height Constraint of CollectionView.

比基于 CollectionView ContentSize 设置 CollectionView 高度约束.

than setting CollectionView height Constraint based on CollectionView ContentSize.

self.collectionViewHeight.constant = self.recentCollectionView.collectionViewLayout.collectionViewContentSize.height

另一种方法是覆盖 tableViewCell 的 systemLayoutSizeFitting 方法

the alternative approach is to override systemLayoutSizeFitting method of tableViewCell

例如:

override func systemLayoutSizeFitting(_ targetSize: CGSize, withHorizontalFittingPriority horizontalFittingPriority: UILayoutPriority, verticalFittingPriority: UILayoutPriority) -> CGSize {
        super.systemLayoutSizeFitting(targetSize, withHorizontalFittingPriority: horizontalFittingPriority, verticalFittingPriority: verticalFittingPriority)
        self.tagCollectionView.layoutIfNeeded()
        return self.tagCollectionView.collectionViewLayout.collectionViewContentSize
    }

此外,您不需要在表格视图委托 heightForRowAtIndexPath 中返回固定的 600 高度

in addition you don't need to return fixed 600 height in your table view delegate heightForRowAtIndexPath

这篇关于带有约束的嵌套集合视图的意外行为(Swift 4)的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆