1
0
mirror of https://github.com/yattee/yattee.git synced 2024-12-14 14:20:32 +05:30
yattee/Shared/Videos/VerticalCells.swift

114 lines
3.3 KiB
Swift
Raw Normal View History

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