2016-04-29 15 views
2

此代碼塊試圖解決問題的核心問題。如果在將一個單元格出隊後(通過configure)更改了約束條件,以便更改單元格高度,那麼最終會出現損壞的約束警告(Unable to simultaneously satisfy constraints ...)。但是,它顯示正確。UITableView在更改出現排隊後影響單元格高度的約束時發生UITableView,最終以破壞的約束結束

import UIKit 
class ViewController: UIViewController { 

    @IBOutlet var tableView: UITableView! 

    override func viewDidLoad() { 
     super.viewDidLoad() 
     tableView.delegate = self 
     tableView.dataSource = self 
     tableView.registerClass(Cell.self, forCellReuseIdentifier: "cell") 
     tableView.estimatedRowHeight = 55.0 
     tableView.rowHeight = UITableViewAutomaticDimension 
    } 

} 

extension ViewController: UITableViewDelegate, UITableViewDataSource { 
    func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int { 
     return array.count 
    } 

    func numberOfSectionsInTableView(tableView: UITableView) -> Int { 
     return 1 
    } 

    func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell { 
     let cell = tableView.dequeueReusableCellWithIdentifier("cell") as! Cell 
     cell.configure(array[indexPath.row]) 
     return cell 
    } 

    func tableView(tableView: UITableView, willDisplayCell cell: UITableViewCell, forRowAtIndexPath indexPath: NSIndexPath) { 
    } 
} 

enum Position { 
    case Top 
    case Middle 
    case Bottom 
} 

class Cell: UITableViewCell { 

    private var topConstraint: NSLayoutConstraint! 
    private var bottomConstraint: NSLayoutConstraint! 

    private let label = UILabel() 
    private var previous: Position? 

    override init(style: UITableViewCellStyle, reuseIdentifier: String?) { 
     super.init(style: style, reuseIdentifier: reuseIdentifier) 

     label.backgroundColor = UIColor.redColor().colorWithAlphaComponent(0.5) 
     label.textColor = UIColor.blackColor() 
     label.translatesAutoresizingMaskIntoConstraints = false 
     contentView.addSubview(label) 
     label.leadingAnchor.constraintEqualToAnchor(contentView.leadingAnchor, constant: 30).active = true 
     label.trailingAnchor.constraintEqualToAnchor(contentView.trailingAnchor, constant: -30).active = true 
     topConstraint = label.topAnchor.constraintEqualToAnchor(contentView.topAnchor) 
     topConstraint.active = true 
     bottomConstraint = label.bottomAnchor.constraintEqualToAnchor(contentView.bottomAnchor) 
     bottomConstraint.active = true 
    } 

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

    func configure(position: Position) { 
     if previous != nil {print("previous \(previous!) new \(position)")} 
     topConstraint.constant = position == .Top ? 30 : 0 
     bottomConstraint.constant = position == .Bottom ? -30 : 0 
     label.text = String(position) 
     previous = position 
    } 
} 

let array: [Position] = [.Top, .Middle, .Middle, .Middle, .Bottom, 
         .Top, .Middle, .Middle, .Middle, .Middle, .Bottom, 
         .Top, .Middle, .Middle, .Bottom, 
         .Top, .Middle, .Middle, .Middle, .Middle, .Middle, .Middle, .Bottom, 
         .Top, .Middle, .Middle, .Middle, .Bottom, 
         .Top, .Bottom, 
         .Top, .Middle, .Middle, .Middle, .Middle, .Middle, .Middle, .Bottom, 
         .Top, .Middle, .Middle, .Middle, .Bottom, 
         .Top, .Middle, .Middle, .Bottom, 
         .Top, .Middle, .Middle, .Middle, .Middle, .Middle, .Middle, .Bottom, 
] 

錯誤消息是...

2016-04-28 22:56:29.831 test-constraint-change[51059:3776208] Unable to simultaneously satisfy constraints. 
Probably at least one of the constraints in the following list is one you don't want. 
Try this: 
    (1) look at each constraint and try to figure out which you don't expect; 
    (2) find the code that added the unwanted constraint or constraints and fix it. 
(
"<NSLayoutConstraint:0x7c117c70 V:|-(30)-[UILabel:0x7c148770'Middle'] (Names: '|':UITableViewCellContentView:0x7c149b70)>", 
"<NSLayoutConstraint:0x7c115bd0 UILabel:0x7c148770'Middle'.bottom == UITableViewCellContentView:0x7c149b70.bottom>", 
"<NSLayoutConstraint:0x7c14a3f0 'UIView-Encapsulated-Layout-Height' V:[UITableViewCellContentView:0x7c149b70(20.5)]>" 
) 

Will attempt to recover by breaking constraint <NSLayoutConstraint:0x7c115bd0 UILabel:0x7c148770'Middle'.bottom == UITableViewCellContentView:0x7c149b70.bottom> 

Make a symbolic breakpoint at UIViewAlertForUnsatisfiableConstraints to catch this in the debugger. 
The methods in the UIConstraintBasedLayoutDebugging category on UIView listed in <UIKit/UIView.h> may also be helpful. 
+0

你得到的錯誤是什麼?請向我們展示確切的堆棧跟蹤和錯誤消息。 – ozgur

+0

完成,但3個約束沒有任何技術上的錯誤。當時,他們應該強制細胞高度變大(細胞高度是動態的)。該單元正在從「中」向「頂」類型發展。 –

回答

12

設置你的約束優先級999就像現在,它會正確顯示。

+0

感謝您的回覆......我確信這會做到這一點。但感覺有點哈克。希望找到一個乾淨的方式來做到這一點。 (我也可以忽略這些警告) –

+0

這是您最乾淨的解決方案,並且與自動佈局的工作方式有關。使用自動調整表格單元格,系統隱式添加高度約束('UIView-Encapsulated-Layout-Height')。設置優先級允許自動佈局優雅地解決這些衝突。請參閱:https://www.raywenderlich.com/87975/dynamic-table-view-cell-height-ios-8-swift –

+0

好的,我認爲這隻與動態高度有關。我們也注意到與寬度完全相同的問題(不同的場景)。儘可能多的解決方案,仍然感覺像一個自動佈局的錯誤。你怎麼看? –

相關問題