yattee/Shared/Videos/VerticalCells.swift

111 lines
3.1 KiB
Swift
Raw Normal View History

2021-08-01 23:01:24 +00:00
import Defaults
import SwiftUI
2022-11-27 10:42:16 +00:00
struct VerticalCells<Header: View>: View {
2021-08-01 23:01:24 +00:00
#if os(iOS)
@Environment(\.verticalSizeClass) private var verticalSizeClass
#endif
@Environment(\.loadMoreContentHandler) private var loadMoreContentHandler
2022-12-12 00:18:29 +00:00
@Environment(\.listingStyle) private var listingStyle
var items = [ContentItem]()
2022-03-27 10:49:57 +00:00
var allowEmpty = false
2023-05-21 15:17:56 +00:00
var edgesIgnoringSafeArea = Edge.Set.horizontal
2021-08-01 23:01:24 +00:00
2022-11-27 10:42:16 +00:00
let header: Header?
2023-05-25 12:28:29 +00:00
@State private var gridSize = CGSize.zero
init(
items: [ContentItem],
allowEmpty: Bool = false,
edgesIgnoringSafeArea: Edge.Set = .horizontal,
@ViewBuilder header: @escaping () -> Header? = { nil }
) {
2022-11-27 10:42:16 +00:00
self.items = items
self.allowEmpty = allowEmpty
2023-05-21 15:17:56 +00:00
self.edgesIgnoringSafeArea = edgesIgnoringSafeArea
2022-11-27 10:42:16 +00:00
self.header = header()
}
2023-05-25 12:28:29 +00:00
init(
items: [ContentItem],
allowEmpty: Bool = false
) where Header == EmptyView {
2022-11-27 10:42:16 +00:00
self.init(items: items, allowEmpty: allowEmpty) { EmptyView() }
}
2021-08-01 23:01:24 +00:00
var body: some View {
2021-09-30 16:53:26 +00:00
ScrollView(.vertical, showsIndicators: scrollViewShowsIndicators) {
LazyVGrid(columns: adaptiveItem, alignment: .center) {
2022-11-27 10:42:16 +00:00
Section(header: header) {
ForEach(contentItems) { item in
ContentItemView(item: item)
.onAppear { loadMoreContentItemsIfNeeded(current: item) }
}
2021-09-29 23:29:18 +00:00
}
2021-08-01 23:01:24 +00:00
}
2021-09-30 16:53:26 +00:00
.padding()
2021-08-01 23:01:24 +00:00
}
2022-08-08 17:28:02 +00:00
.animation(nil)
2023-05-21 15:17:56 +00:00
.edgesIgnoringSafeArea(edgesIgnoringSafeArea)
2021-09-26 22:28:42 +00:00
#if os(macOS)
.background(Color.secondaryBackground)
2021-09-26 22:28:42 +00:00
.frame(minWidth: 360)
#endif
2021-08-01 23:01:24 +00:00
}
2022-03-27 10:49:57 +00:00
var contentItems: [ContentItem] {
2022-12-14 16:20:24 +00:00
items.isEmpty ? (allowEmpty ? items : ContentItem.placeholders) : items.sorted { $0 < $1 }
2022-03-27 10:49:57 +00:00
}
func loadMoreContentItemsIfNeeded(current item: ContentItem) {
let thresholdIndex = items.index(items.endIndex, offsetBy: -5)
if items.firstIndex(where: { $0.id == item.id }) == thresholdIndex {
loadMoreContentHandler()
}
}
2021-08-02 21:10:22 +00:00
var adaptiveItem: [GridItem] {
2022-12-12 00:18:29 +00:00
if listingStyle == .list {
return [.init(.flexible())]
}
return [GridItem(.adaptive(minimum: adaptiveGridItemMinimumSize, maximum: adaptiveGridItemMaximumSize))]
2021-08-01 23:01:24 +00:00
}
2021-09-18 20:36:42 +00:00
var adaptiveGridItemMinimumSize: Double {
2021-08-01 23:01:24 +00:00
#if os(iOS)
2021-08-16 13:39:31 +00:00
return verticalSizeClass == .regular ? 320 : 800
2021-08-01 23:01:24 +00:00
#elseif os(tvOS)
2021-11-11 21:07:13 +00:00
return 600
2021-08-01 23:01:24 +00:00
#else
2021-08-16 13:39:31 +00:00
return 320
2021-08-01 23:01:24 +00:00
#endif
}
2021-11-11 21:07:13 +00:00
var adaptiveGridItemMaximumSize: Double {
#if os(tvOS)
return 600
#else
return .infinity
#endif
}
2021-08-01 23:01:24 +00:00
var scrollViewShowsIndicators: Bool {
#if !os(tvOS)
true
#else
false
#endif
}
}
2021-10-23 16:49:45 +00:00
struct VeticalCells_Previews: PreviewProvider {
2021-08-01 23:01:24 +00:00
static var previews: some View {
2021-11-11 21:07:13 +00:00
VerticalCells(items: ContentItem.array(of: Array(repeating: Video.fixture, count: 30)))
2021-09-29 11:45:00 +00:00
.injectFixtureEnvironmentObjects()
2021-08-01 23:01:24 +00:00
}
}