Impressia/Vernissage/Views/TimelineFeedView.swift
2023-01-21 18:01:17 +01:00

150 lines
5.6 KiB
Swift

//
// https://mczachurski.dev
// Copyright © 2022 Marcin Czachurski and the repository contributors.
// Licensed under the MIT License.
//
import SwiftUI
struct TimelineFeedView: View {
@EnvironmentObject private var applicationState: ApplicationState
@State public var accountId: String
@State public var isLocalOnly: Bool
@State private var allItemsLoaded = false
@State private var firstLoadFinished = false
@State private var statusViewModels: [StatusViewModel] = []
private let defaultLimit = 20
var body: some View {
ScrollView {
VStack(alignment: .center) {
if firstLoadFinished == true {
ForEach(self.statusViewModels, id: \.uniqueId) { item in
NavigationLink(destination: StatusView(statusId: item.id,
imageBlurhash: item.mediaAttachments.first?.blurhash,
imageWidth: item.getImageWidth(),
imageHeight: item.getImageHeight())
.environmentObject(applicationState)) {
ImageRowAsync(statusViewModel: item)
}
.buttonStyle(EmptyButtonStyle())
}
LazyVStack {
if allItemsLoaded == false && firstLoadFinished == true {
HStack {
Spacer()
LoadingIndicator()
.task {
do {
try await self.loadMoreStatuses()
} catch {
ErrorService.shared.handle(error, message: "Loading more statuses failed.", showToastr: true)
}
}
Spacer()
}
}
}
}
}
}
.overlay(alignment: .center) {
if firstLoadFinished == false {
LoadingIndicator()
} else {
if self.statusViewModels.isEmpty {
VStack {
Image(systemName: "photo.on.rectangle.angled")
.font(.largeTitle)
.padding(.bottom, 4)
Text("Unfortunately, there are no photos here.")
.font(.title3)
}.foregroundColor(.lightGrayColor)
}
}
}
.task {
do {
try await self.loadStatuses()
} catch {
ErrorService.shared.handle(error, message: "Loading statuses failed.", showToastr: !Task.isCancelled)
}
}.refreshable {
do {
try await self.loadTopStatuses()
} catch {
ErrorService.shared.handle(error, message: "Loading statuses failed.", showToastr: !Task.isCancelled)
}
}
}
private func loadStatuses() async throws {
guard firstLoadFinished == false else {
return
}
let statuses = try await PublicTimelineService.shared.getStatuses(
accountData: self.applicationState.accountData,
local: isLocalOnly,
remote: !isLocalOnly,
limit: self.defaultLimit)
var inPlaceStatuses: [StatusViewModel] = []
for item in statuses {
inPlaceStatuses.append(StatusViewModel(status: item))
}
self.firstLoadFinished = true
self.statusViewModels.append(contentsOf: inPlaceStatuses)
if statuses.count < self.defaultLimit {
self.allItemsLoaded = true
}
}
private func loadMoreStatuses() async throws {
if let lastStatusId = self.statusViewModels.last?.id {
let previousStatuses = try await PublicTimelineService.shared.getStatuses(
accountData: self.applicationState.accountData,
local: isLocalOnly,
remote: !isLocalOnly,
maxId: lastStatusId,
limit: self.defaultLimit)
if previousStatuses.count < self.defaultLimit {
self.allItemsLoaded = true
}
var inPlaceStatuses: [StatusViewModel] = []
for item in previousStatuses {
inPlaceStatuses.append(StatusViewModel(status: item))
}
self.statusViewModels.append(contentsOf: inPlaceStatuses)
}
}
private func loadTopStatuses() async throws {
if let firstStatusId = self.statusViewModels.first?.id {
let newestStatuses = try await PublicTimelineService.shared.getStatuses(
accountData: self.applicationState.accountData,
local: isLocalOnly,
remote: !isLocalOnly,
sinceId: firstStatusId,
limit: self.defaultLimit)
var inPlaceStatuses: [StatusViewModel] = []
for item in newestStatuses {
inPlaceStatuses.append(StatusViewModel(status: item))
}
self.statusViewModels.insert(contentsOf: inPlaceStatuses, at: 0)
}
}
}