Swift约束以编程方式创建奇怪的行为

时间:2015-03-14 14:23:34

标签: ios xcode swift autolayout constraints

我尝试以编程方式布局所有约束,因为我必须根据用户输入添加和删除框。当我运行一个在ViewDidLoad中应用所有约束的函数时,它运行得很好。但如果我再次运行它,将它们全部删除后,我的2个标签就会消失在导航栏后面。我无法弄清楚原因!请记住,如果我从未运行下面粘贴的功能,我的屏幕是空白的,因为我没有从故事板启动的限制。我的目标是弄清楚为什么我的标签和按钮会在第二次运行同一段代码时消失。

以下是我在下面发布的代码最初在ViewDidLoad中运行时的图片:

Constraints run during ViewDidLoad

下面是按下下一个按钮后再次运行同一段代码的图片。这只是一个测试,将来我需要在更改后重置所有约束:

Constraints run again later (when next button is pressed)

以下是调试视图层次结构的图片: enter image description here

创建约束的代码如下。请记住,我只有5个元素,我的主视图,滚动视图,aglAltitudeLabel是标签,nextButton是按钮,aglAltitude是文本输入:

    //Remove all pre-existing contraints
    var constraints:NSArray = aglAltitudeLabel.constraints()
    aglAltitudeLabel.removeConstraints(constraints)
    constraints = aglAltitude.constraints()
    aglAltitude.removeConstraints(constraints)
    constraints = nextButton.constraints()
    nextButton.removeConstraints(constraints)
    constraints = scrollView.constraints()
    scrollView.removeConstraints(constraints)
    constraints = view.constraints()
    self.view.removeConstraints(constraints)
    constraints = calculateButton.constraints()
    calculateButton.removeConstraints(constraints)

    //Make scrollview fit normal View, accounting for Navigation bar
    self.view.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("|[first(\(self.view.frame.width))]|", options: nil, metrics: nil, views: ["first": scrollView]))
    self.view.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("V:|-64-[first(\(self.view.frame.height-64))]|", options: nil, metrics: nil, views: ["first": scrollView]))

    //Set the heights for the label and button
    let labelHeight = NSLayoutConstraint(item: aglAltitudeLabel, attribute: NSLayoutAttribute.Height, relatedBy: NSLayoutRelation.Equal, toItem: nil, attribute: NSLayoutAttribute.NotAnAttribute, multiplier: 1.0, constant: 20)
    scrollView.addConstraint(labelHeight)
    let buttonHeight = NSLayoutConstraint(item: nextButton, attribute: NSLayoutAttribute.Height, relatedBy: NSLayoutRelation.Equal, toItem: nil, attribute: NSLayoutAttribute.NotAnAttribute, multiplier: 1.0, constant: 30)
    scrollView.addConstraint(buttonHeight)

    //Setup the vertical layout
    scrollView.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("V:|-8-[first(30)]-8-|", options: nil, metrics: nil, views: ["first": aglAltitude]))


    //Setup the horizontal layout
    scrollView.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("|-8-[first(97)]-8-[second(>=100)]-8-[third(46)]-8-|", options: nil, metrics: nil, views: ["first": aglAltitudeLabel, "second": aglAltitude, "third": nextButton]))

    //Align baselines - this also occurs with the option above of AlignAllBaselines and leaving this code out
    let aglLabelBaseline = NSLayoutConstraint(item: aglAltitudeLabel, attribute: NSLayoutAttribute.Baseline, relatedBy: NSLayoutRelation.Equal, toItem: aglAltitude, attribute: NSLayoutAttribute.Baseline, multiplier: 1.0, constant: 0)
    scrollView.addConstraint(aglLabelBaseline)
    let nextButtonBaseline = NSLayoutConstraint(item: nextButton, attribute: NSLayoutAttribute.Baseline, relatedBy: NSLayoutRelation.Equal, toItem: aglAltitude, attribute: NSLayoutAttribute.Baseline, multiplier: 1.0, constant: 0)
    scrollView.addConstraint(nextButtonBaseline)

    self.view.layoutSubviews()

感谢您的帮助。我确定我做错了55件事。

编辑:如果我从这里更改代码的最后一部分:

    //Align baselines - this also occurs with the option above of AlignAllBaselines and leaving this code out
    let aglLabelBaseline = NSLayoutConstraint(item: aglAltitudeLabel, attribute: NSLayoutAttribute.Baseline, relatedBy: NSLayoutRelation.Equal, toItem: aglAltitude, attribute: NSLayoutAttribute.Baseline, multiplier: 1.0, constant: 0)
    scrollView.addConstraint(aglLabelBaseline)
    let nextButtonBaseline = NSLayoutConstraint(item: nextButton, attribute: NSLayoutAttribute.Baseline, relatedBy: NSLayoutRelation.Equal, toItem: aglAltitude, attribute: NSLayoutAttribute.Baseline, multiplier: 1.0, constant: 0)
    scrollView.addConstraint(nextButtonBaseline)

对此:

    //Align baselines - this also occurs with the option above of AlignAllBaselines and leaving this code out
    let aglLabelBaseline = NSLayoutConstraint(item: aglAltitudeLabel, attribute: NSLayoutAttribute.Top, relatedBy: NSLayoutRelation.Equal, toItem: scrollView, attribute: NSLayoutAttribute.Top, multiplier: 1.0, constant: 14)
    self.view.addConstraint(aglLabelBaseline)
    let nextButtonBaseline = NSLayoutConstraint(item: nextButton, attribute: NSLayoutAttribute.Top, relatedBy: NSLayoutRelation.Equal, toItem: scrollView, attribute: NSLayoutAttribute.Top, multiplier: 1.0, constant: 8)
    self.view.addConstraint(nextButtonBaseline)

它完全符合我的预期。为什么不同?

1 个答案:

答案 0 :(得分:1)

如果不知道您的要求是什么,很难给您提供好的建议。这是我如何做到这一点的一个例子。我在代码中添加了所有视图,并为视图添加了背景颜色,以便我可以更好地查看其布局。我假设滚动视图只添加一次,并且从未删除过,因此它对self.view的约束只应在viewDidLoad中添加一次。如果您不需要更改它们,则其他视图的高度也相同(请注意,我将高度约束添加到视图本身,而不是其超级视图)。唯一需要删除和重新添加的是滚动视图及其子视图之间的约束。

class ViewController: UIViewController {

    var aglAltitudeLabel = UILabel()
    var aglAltitude = UITextField()
    var nextButton = UIButton.buttonWithType(.System) as UIButton
    var scrollView = UIScrollView()
    var viewsDict: NSDictionary!

    override func viewDidLoad() {
        super.viewDidLoad()
        aglAltitudeLabel.text = "AGL Altitude"
        aglAltitudeLabel.backgroundColor = UIColor.lightGrayColor()
        aglAltitudeLabel.setTranslatesAutoresizingMaskIntoConstraints(false)
        aglAltitude.setTranslatesAutoresizingMaskIntoConstraints(false)
        aglAltitude.backgroundColor = UIColor.yellowColor()
        aglAltitudeLabel.addConstraint(NSLayoutConstraint(item: aglAltitudeLabel, attribute: .Height, relatedBy: .Equal, toItem: nil, attribute: .NotAnAttribute, multiplier: 1, constant: 30))
        scrollView.setTranslatesAutoresizingMaskIntoConstraints(false)
        nextButton.setTranslatesAutoresizingMaskIntoConstraints(false)
        nextButton.setTitle("Next", forState: .Normal)
        nextButton.backgroundColor = UIColor.lightGrayColor()
        nextButton.addConstraint(NSLayoutConstraint(item: nextButton, attribute: .Height, relatedBy: .Equal, toItem: nil, attribute: .NotAnAttribute, multiplier: 1, constant: 30))
        self.view.addSubview(scrollView)
        scrollView.addSubview(aglAltitude)
        scrollView.addSubview(aglAltitudeLabel)
        scrollView.addSubview(nextButton)
        scrollView.backgroundColor = UIColor(red: 1, green: 1, blue: 0.8, alpha: 1)

        viewsDict = ["scrollView": scrollView, "aglAltitudeLabel": aglAltitudeLabel, "aglAltitude":aglAltitude, "nextButton":nextButton] as NSDictionary

        self.view.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("|[scrollView]|", options: nil, metrics: nil, views: viewsDict))
        self.view.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("V:|-64-[scrollView]|", options: nil, metrics: nil, views: viewsDict))

        self.redoConstraints()
    }


    @IBAction func redoConstraints() {

        scrollView.removeConstraints(scrollView.constraints())

        scrollView.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("V:|-8-[aglAltitude(30)]-8-|", options: nil, metrics: nil, views: viewsDict))

        scrollView.addConstraints(NSLayoutConstraint.constraintsWithVisualFormat("|-8-[aglAltitudeLabel(97)]-8-[aglAltitude(>=100)]-8-[nextButton(46)]-8-|", options: .AlignAllBottom , metrics: nil, views: viewsDict))
    }
}

请注意,我从滚动视图的约束中取出了固定的宽度和高度。没有必要,它会导致它在旋转时无法正常工作。