【问题标题】:UITextField inside UIStackView Spacing Issue on width constraintUIStackView 内的 UITextField 宽度约束的间距问题
【发布时间】:2019-03-01 12:20:53
【问题描述】:

我有一个垂直的UIStackView,里面有各种UITextFields(现在只有4个)。当我给 UIStackView 提供间距时,我给出了以下错误;

"<NSLayoutConstraint:0x1c4286720 H:|-(0)-[UIStackView:0x104d3e6f0](LTR)   (active, names: '|':Hello.OtpInputView:0x104d3c5b0 )>",
"<NSLayoutConstraint:0x1c4286770 UIStackView:0x104d3e6f0.right == Hello.OtpInputView:0x104d3c5b0.right   (active)>",
"<NSLayoutConstraint:0x1c4286860 '_UITemporaryLayoutWidth' Hello.OtpInputView:0x104d3c5b0.width == 0   (active)>",
"<NSLayoutConstraint:0x1c4286e50 'UISV-canvas-connection' UIStackView:0x104d3e6f0.leading == Hello.CustomOtpTextField:0x10588d600.leading   (active)>",
"<NSLayoutConstraint:0x1c4286ea0 'UISV-canvas-connection' H:[Hello.CustomOtpTextField:0x105082600]-(0)-|   (active, names: '|':UIStackView:0x104d3e6f0 )>",
"<NSLayoutConstraint:0x1c4286f40 'UISV-fill-equally' Hello.CustomOtpTextField:0x105024600.width == Hello.CustomOtpTextField:0x10588d600.width   (active)>",
"<NSLayoutConstraint:0x1c4287030 'UISV-fill-equally' Hello.CustomOtpTextField:0x105077a00.width == Hello.CustomOtpTextField:0x10588d600.width   (active)>",
"<NSLayoutConstraint:0x1c42871c0 'UISV-fill-equally' Hello.CustomOtpTextField:0x105082600.width == Hello.CustomOtpTextField:0x10588d600.width   (active)>",
"<NSLayoutConstraint:0x1c4286ef0 'UISV-spacing' H:[Hello.CustomOtpTextField:0x10588d600]-(10)-[Hello.CustomOtpTextField:0x105024600]   (active)>",
"<NSLayoutConstraint:0x1c4286fe0 'UISV-spacing' H:[Hello.CustomOtpTextField:0x105024600]-(10)-[Hello.CustomOtpTextField:0x105077a00]   (active)>",
"<NSLayoutConstraint:0x1c42870d0 'UISV-spacing' H:[Hello.CustomOtpTextField:0x105077a00]-(10)-[Hello.CustomOtpTextField:0x105082600]   (active)>"

)

将尝试通过打破约束来恢复

我如何声明 StackView:

stackView.translatesAutoresizingMaskIntoConstraints = false
stackView.axis = .horizontal
stackView.distribution = .fillEqually
stackView.spacing = 10

//textFields is just an array of UITextFields
textFields.enumerated().forEach { (index,field) in
    field.textColor = UIColor.darkGray
    field.font = Font.sourceSansRegular?.withSize(46)
    field.keyboardType = .numberPad
    field.delegate = self
    field.translatesAutoresizingMaskIntoConstraints = false
    field.tag = index
    field.textAlignment = .center
    field.myDelegate = self
    field.sizeToFit()
    stackView.addArrangedSubview(field)
    field.widthAnchor.constraint(equalTo: stackView.widthAnchor, multiplier: (CGFloat(0 / textFields.count)),constant: CGFloat(10 * (textFields.count - 1))).isActive = true
}

self.addSubview(stackView)
stackView.anchor(self.topAnchor, left: self.leftAnchor, bottom: self.bottomAnchor, right: self.rightAnchor, topConstant: 0, leftConstant: 0, bottomConstant: 0, rightConstant: 0, widthConstant: 0, heightConstant: 0)

我想要的是,例如 UIStackView 有 110pt 宽度和 10 个空间,那么 UITextFields 应该有 20pt ((20 * 4) + (3 * 10) 即 110)。因此,所有UITextFields 都应具有相同的宽度(均等填充)

我怎样才能做到这一点?

编辑:我认为我需要为一个 TextField 指定至少一个宽度,所以我添加了

 field.widthAnchor.constraint(equalTo: stackView.widthAnchor, multiplier: (CGFloat(0 / textFields.count)),constant: CGFloat(10 * (textFields.count - 1))).isActive = true

但是这也会产生中断错误。

EDIT2:此堆栈视图位于自定义 UIView 中,我在其中提供锚点

 override init(frame: CGRect) {
    super.init(frame: frame)
    //Where I gave all anchors
    setupUI()
}

还有我如何初始化这个自定义 UIView

    override func viewDidLoad() {
    super.viewDidLoad()

    // Do any additional setup after loading the view.
    self.view.backgroundColor = UIColor.white
    self.view.addSubview(otpField)
    otpField.anchor(self.view.topAnchor, left: self.view.leftAnchor, bottom: nil, right: self.view.rightAnchor, topConstant: 40, leftConstant: 40, bottomConstant: 0, rightConstant: 40, widthConstant: 0, heightConstant: 60)
}

我想在一张图片中实现什么:

【问题讨论】:

  • 它有垂直轴。抱歉,在我的问题中,我忘记更改了。
  • 如果你的意思是 UITextfield 的宽度锚点,我是在将 textfield 添加到 UIStackview 之后写的。
  • 我想要它用于 VERTICAL 并且我正在自定义 UIView 中进行操作,我编辑了我的问题,您可以检查一下吗?
  • 我添加了我如何初始化自定义 UIView。
  • 我还加了一张图片更清楚。

标签: ios swift uitextfield uistackview


【解决方案1】:

如果你已经设置了stackView.distribution = .fillEqually,你只需要为高度添加一个约束,像这样为你的任何 textFiled 添加一个约束

let height = NSLayoutConstraint(item: textField1, attribute: .height, relatedBy: .equal, toItem: nil, attribute: .height, multiplier: 1.0, constant: 40.0)
textField1.addConstraint = height

通过这种方式,所有文本字段的高度都将是 40,总大小将取决于给定的项目数和间距,即 Height = 40Space = 10 然后总高度将是 (Height * Number of textfields) + (Space * (number of textfield - 1))

注意:垂直和水平堆栈视图的概念是相同的,您需要使用高度和宽度。

【讨论】:

  • 给定高度 40 也会打破约束。
  • 这可能是由于与其他 UI 元素的相关性而发生的。
  • 但是我的 Stackview 是垂直的而不是水平的,所以我的问题是宽度而不是高度。
  • 好的,只需将高度约束更改为宽度即可。你能显示你的 ViewController