#swift #swiftui #nsview #nswindow #nsnotifications
#swift #свифтуи #nsview #нсокно #nsnotifications
Вопрос:
В настоящее время у меня есть LazyVGrid
настройка как таковая:
struct NetworkGrid: View {
var networks: [Network]
let columns = [
GridItem(.flexible()),
GridItem(.flexible()),
GridItem(.flexible()),
GridItem(.flexible())
]
var body: some View {
ScrollView {
LazyVGrid(columns: columns) {
ForEach(networks) { network in
NetworkCard(network: network)
}
}
}
}
}
Я хотел бы установить количество столбцов сетки на основе текущего размера окна, т.Е.
func windowDidResize(_ notification: Notification) {
itemWidth = CGFloat(300)
if window.width <= itemWidth {
GridItem(.flexible()), GridItem(.flexible())
} else if window.width <= itemWidth * 2 {
GridItem(.flexible()), GridItem(.flexible()), GridItem(.flexible())
} else if window.width <= itemWidth * 3 {
GridItem(.flexible()), GridItem(.flexible()), GridItem(.flexible()), GridItem(.flexible())
}
...
}
Как бы я приступил к реализации такого наблюдателя с помощью SwiftUI?
Ответ №1:
Эквивалент SwiftUI для прослушивания размера окна, вероятно, будет использовать GeometryReader
. В вашем примере вы можете считывать размер и динамически определять столбцы на основе его значения ширины:
struct NetworkGrid: View {
var networks: [Network]
func columnsForWidth(width: CGFloat) -> [GridItem] {
print("Columns for width: (width)")
return Array(repeating: GridItem(.flexible()), count: Int(width) / 100)
}
var body: some View {
GeometryReader { geometry in
ScrollView {
LazyVGrid(columns: columnsForWidth(width: geometry.size.width)) {
ForEach(networks) { network in
NetworkCard(network: network)
}
}
}
}
}
}
Комментарии:
1. Спасибо, это работает потрясающе! Похоже, оно не появляется, когда я встраиваю его в другое представление и запускаю с помощью переключателя кнопок. Есть идеи?
2.Консоль будет печатать
Columns for width: 1356.0
Columns for width: 0.0
Columns for width: 1356.0
при вызове представления.3. Возможно, он не отображается при встраивании в список.
4. Вы должны были бы показать мне пример — не совсем уверен, о каком именно сценарии вы говорите.
5. Я также скажу (заранее), что
GeometryReader
это не измеряет размер всего окна — оно измеряет только размер его собственного представления. Таким образом, чтобы получить полный размер окна, это должен быть вид самого высокого уровня.
Ответ №2:
Если вы хотите обновить @State
@ObservedObject
переменные or при изменении размера представления, вы можете использовать onAppear(выполнить:) для обновления переменных с начальным размером представления и onChange(of:perform:) для обнаружения изменений размера представления:
struct MyView: View {
@State private var size: CGSize = .zero
var body: some View {
GeometryReader { geometry in
ZStack {
Text("Hello World")
}.onAppear {
size = geometry.size
}.onChange(of: geometry.size) { newSize in
size = newSize
}
}
}
}