SwiftUI HStack с элементами разного размера
Ищете способ разделить HStack на неровные элементы, один из которых занимает половину экрана, а два других - 1/4 экрана (см. Приложение).
Код:
struct MyCategoryRow: View {
var body: some View {
VStack(alignment: .leading, spacing: 0) {
HStack(spacing: 0) {
ForEach(0...2, id: \.self) { block in
ColoredBlock()
}.frame(minWidth: 0, maxWidth: .infinity, alignment: Alignment.topLeading)
}
}
}
Таким образом, приведенный выше код приводит к созданию HStack с 3 цветными блоками одинаковой ширины. Я попытался установить ширину с помощью UIScreen.main.bounds.width, но этот подход не адаптируется к изменению ориентации. Я также пробовал использовать GeometryReader вот так:
GeometryReader { g in
ColoredBlock().frame(width: block == 0 ? g.size.width * 2 : g.size.width / 2)
}
Но это тоже не сработало, похоже, что факт, что каждый ColoredBlock займет 1/3 HStack, решен заранее.
Обновление с помощью layoutPriority попытки:
struct MyCategoryRow: View {
var body: some View {
VStack(alignment: .leading, spacing: 0) {
HStack(spacing: 0) {
ForEach(0...2, id: \.self) { block in
ColoredBlock(background: (block == 0) ? Constants.pastel1 : Constants.pastel2).layoutPriority(block == 0 ? 1.0 : 0.5)
}.frame(minWidth: 0, maxWidth: .infinity, alignment: Alignment.topLeading)
}
}
}
struct ColoredBlock: View {
let background: Color
var body: some View {
GeometryReader { geometry in
VStack(spacing: 0){
HStack {
Text("Some text").padding(.horizontal, 15).padding(.top, 15)
}.frame(alignment: .leading)
VStack(alignment: .leading){
Text("Some more text").font(.subheadline).fontWeight(.light)
}.padding(.vertical, 10).padding(.horizontal, 15)
}.background(self.background)
.cornerRadius(15)
}
}
}
2 ответа
В итоге я использовал GeometryReader в MyCategoryRow вместо ColoredBlock и передавал ширину в ColoredBlock
HStack {
ForEach(0...2, id: \.self) { block in
ColoredBlock(background: (block == 0) ? Constants.pastel1 : Constants.pastel2, width: block == 0 ? geometry.size.width / 2 : geometry.size.width / 4, height: 150.0)
}
}
Это работает, хотя теперь мне приходится указывать высоту, которую я бы предпочел не кодировать жестко. Если у кого-то есть лучшее решение, дайте мне знать!
Я думаю, что вам нужно.layoutPriority. Установите значение 1,0 для первого блока и 0,5 для второго и третьего блоков.