- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我正在尝试在 tableView 中显示自定义 View 并让 iOS 使用 UITableViewAutomaticDimension
计算每一行的高度.不幸的是,我的牢房尺寸不合适(高度不好)。
我的自定义 View 定义如下(出于多种原因,我没有为此部分使用 AutoLayout,并且我不想使用它):
class MyView : UIView {
var label: UILabel!
override init(frame: CGRect) {
super.init(frame: frame)
setupView()
}
required init?(coder aDecoder: NSCoder) {
super.init(coder: aDecoder)
setupView()
}
func setupView() {
label = UILabel()
label.numberOfLines = 0
addSubview(label)
}
func setText(text: String) {
label.text = text
setNeedsLayout()
}
override func layoutSubviews() {
super.layoutSubviews()
label.frame = bounds
invalidateIntrinsicContentSize()
}
override var intrinsicContentSize: CGSize {
guard let text = label.text else { return .zero }
let width = bounds.width
let height = text.heightWithConstrainedWidth(width: width, font: label.font)
return CGSize(width: width, height: height)
}
}
UITableViewCell
中。在
UITableView
中使用它.这里我使用自动布局:
class MyCell : UITableViewCell {
var myView: MyView!
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)
setupCell()
}
required init?(coder aDecoder: NSCoder) {
super.init(coder: aDecoder)
setupCell()
}
func setupCell() {
myView = MyView()
contentView.addSubview(myView)
myView.snp.makeConstraints { make in
make.edges.equalTo(contentView)
}
}
}
class ViewController: UIViewController {
var tableView: UITableView!
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
tableView = UITableView()
view.addSubview(tableView)
tableView.snp.makeConstraints { make in
make.edges.equalTo(view)
}
tableView.register(MyCell.self, forCellReuseIdentifier: "MyCell")
tableView.rowHeight = UITableViewAutomaticDimension
tableView.estimatedRowHeight = 30
tableView.dataSource = self
}
}
extension ViewController: UITableViewDataSource {
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return 1
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "MyCell") as! MyCell
cell.myView.setText(text: "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur")
return cell
}
}
intrinsicContentSize
返回的高度也是如此。调用
MyView
是 143。这是结果的屏幕截图:
intrinsictContentSize
的
MyView
使用它的框架,它不应该,但我不知道如何设计一个 UIView 子类,它在给定宽度的情况下具有灵活的高度。考虑到我调用
invalidateIntrinsicContentSize()
,我认为它会起作用。在 MyView 的
layoutSubviews()
.
最佳答案
文档建议覆盖 intrinsicContentSize
可能不是实现您想要实现的目标的最佳方式。正如您所观察到的,从 UIView类引用:
Setting this property allows a custom view to communicate to the layout system what size it would like to be based on its content. This intrinsic size must be independent of the content frame, because there’s no way to dynamically communicate a changed width to the layout system based on a changed height, for example.
bounds
直到其
intrinsicContentSize
之后才设置已被请求,并且在自动布局上下文中不保证在
layoutSubviews()
之前的任何时间都已设置叫做。有一种方法可以解决这个问题,并将宽度传递给您的自定义 View 以在
intrinsicContentSize
中使用。 ,但是,我已将其作为此答案中的第二个选项。
sizeThatFits(_:)
手动布局
// MyView.swift
func setupView() {
label = UILabel()
label.numberOfLines = 0
label.autoresizingMask = [.flexibleWidth] // New
addSubview(label)
}
override func layoutSubviews() {
super.layoutSubviews()
label.sizeToFit() // New
}
override func sizeThatFits(_ size: CGSize) -> CGSize {
return label.sizeThatFits(size) // New
}
.flexibleWidth
自动调整大小掩码)和它的文本内容(因为
numberOfLines
设置为 0 )。如果 MyView 有其他 subview ,则需要计算并返回
sizeThatFits(_:)
中的总大小。 .
systemLayoutSizeFitting(_:withHorizontalFittingPriority:verticalFittingPriority)
在每个单元格上,依次调用
sizeThatFits(_:)
在细胞上。 (见
WWDC 2014 Session 226。)
UILayoutPriorityRequired
.您可以通过覆盖自定义单元子类中的其中一种方法(前者用于自动布局,后者用于手动布局)并使用传入的大小的宽度来计算并返回细胞。
myView
的大小。 :
// MyCell.swift
override func systemLayoutSizeFitting(_ targetSize: CGSize, withHorizontalFittingPriority horizontalFittingPriority: UILayoutPriority, verticalFittingPriority: UILayoutPriority) -> CGSize {
return myView.sizeThatFits(targetSize)
}
intrinsicContentSize
自动布局
// MyView.swift
override func layoutSubviews() {
super.layoutSubviews()
label.frame = bounds
}
// MyView.swift
override var intrinsicContentSize: CGSize {
return CGSize(width: UIViewNoIntrinsicMetric, height: label.intrinsicContentSize.height)
}
preferredMaxLayoutWidth
属性(property)。 UILabel 的头文件为该属性提供了以下注释:
// If nonzero, this is used when determining -intrinsicContentSize for multiline labels
preferredMaxLayoutWidth
设置为适当的宽度。作为
intrinsicContentSize
属性构成自动布局过程的一部分并且仅与自动布局过程相关,并且您的自定义单元子类是执行自动布局的代码的唯一部分,设置布局首选项的适当位置在该类中:
// MyCell.swift
override func systemLayoutSizeFitting(_ targetSize: CGSize, withHorizontalFittingPriority horizontalFittingPriority: UILayoutPriority, verticalFittingPriority: UILayoutPriority) -> CGSize {
myView.label.preferredMaxLayoutWidth = targetSize.width
myView.invalidateIntrinsicContentSize()
return super.systemLayoutSizeFitting(targetSize, withHorizontalFittingPriority: horizontalFittingPriority, verticalFittingPriority: verticalFittingPriority)
}
preferredMaxLayoutWidth
MyView 上的属性并执行。
intrinsicContentSize
基于其多行标签的内容,并且大小在旋转时无效。
关于ios - 如何为具有灵活高度和固定宽度的 View 覆盖 intrinsictContentSize?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41089638/
我正在尝试在 tableView 中显示自定义 View 并让 iOS 使用 UITableViewAutomaticDimension 计算每一行的高度.不幸的是,我的牢房尺寸不合适(高度不好)。
我是一名优秀的程序员,十分优秀!