mirror of
https://github.com/Dimillian/IceCubesApp.git
synced 2025-01-12 17:15:30 +00:00
Merge branch 'main' into iOS-18
This commit is contained in:
commit
cf38a77105
2 changed files with 131 additions and 80 deletions
|
@ -47,6 +47,7 @@ import SwiftUI
|
||||||
|
|
||||||
// Internal source of truth for a timeline.
|
// Internal source of truth for a timeline.
|
||||||
private(set) var datasource = TimelineDatasource()
|
private(set) var datasource = TimelineDatasource()
|
||||||
|
private let statusFetcher: TimelineStatusFetching
|
||||||
private let cache = TimelineCache()
|
private let cache = TimelineCache()
|
||||||
private var isCacheEnabled: Bool {
|
private var isCacheEnabled: Bool {
|
||||||
canFilterTimeline && timeline.supportNewestPagination && client?.isAuth == true
|
canFilterTimeline && timeline.supportNewestPagination && client?.isAuth == true
|
||||||
|
@ -93,7 +94,8 @@ import SwiftUI
|
||||||
var scrollToIndexAnimated: Bool = false
|
var scrollToIndexAnimated: Bool = false
|
||||||
var marker: Marker.Content?
|
var marker: Marker.Content?
|
||||||
|
|
||||||
init() {
|
init(statusFetcher: TimelineStatusFetching = TimelineStatusFetcher()) {
|
||||||
|
self.statusFetcher = statusFetcher
|
||||||
pendingStatusesObserver.scrollToIndex = { [weak self] index in
|
pendingStatusesObserver.scrollToIndex = { [weak self] index in
|
||||||
self?.scrollToIndexAnimated = true
|
self?.scrollToIndexAnimated = true
|
||||||
self?.scrollToIndex = index
|
self?.scrollToIndex = index
|
||||||
|
@ -123,41 +125,6 @@ import SwiftUI
|
||||||
timeline = oldValue
|
timeline = oldValue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleEvent(event: any StreamEvent) async {
|
|
||||||
if let event = event as? StreamEventUpdate,
|
|
||||||
let client,
|
|
||||||
timeline == .home,
|
|
||||||
canStreamEvents,
|
|
||||||
isTimelineVisible,
|
|
||||||
await !datasource.contains(statusId: event.status.id)
|
|
||||||
{
|
|
||||||
pendingStatusesObserver.pendingStatuses.insert(event.status.id, at: 0)
|
|
||||||
let newStatus = event.status
|
|
||||||
await datasource.insert(newStatus, at: 0)
|
|
||||||
await cache()
|
|
||||||
StatusDataControllerProvider.shared.updateDataControllers(for: [event.status], client: client)
|
|
||||||
let statuses = await datasource.getFiltered()
|
|
||||||
withAnimation {
|
|
||||||
statusesState = .display(statuses: statuses, nextPageState: .hasNextPage)
|
|
||||||
}
|
|
||||||
} else if let event = event as? StreamEventDelete {
|
|
||||||
await datasource.remove(event.status)
|
|
||||||
await cache()
|
|
||||||
let statuses = await datasource.getFiltered()
|
|
||||||
withAnimation {
|
|
||||||
statusesState = .display(statuses: statuses, nextPageState: .hasNextPage)
|
|
||||||
}
|
|
||||||
} else if let event = event as? StreamEventStatusUpdate, let client {
|
|
||||||
if let originalIndex = await datasource.indexOf(statusId: event.status.id) {
|
|
||||||
StatusDataControllerProvider.shared.updateDataControllers(for: [event.status], client: client)
|
|
||||||
await datasource.replace(event.status, at: originalIndex)
|
|
||||||
await cache()
|
|
||||||
let statuses = await datasource.getFiltered()
|
|
||||||
statusesState = .display(statuses: statuses, nextPageState: .hasNextPage)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - Cache
|
// MARK: - Cache
|
||||||
|
@ -216,10 +183,8 @@ extension TimelineViewModel: StatusesFetcher {
|
||||||
func fetchStatuses(from: Marker.Content) async throws {
|
func fetchStatuses(from: Marker.Content) async throws {
|
||||||
guard let client else { return }
|
guard let client else { return }
|
||||||
statusesState = .loading
|
statusesState = .loading
|
||||||
var statuses: [Status] = try await client.get(endpoint: timeline.endpoint(sinceId: nil,
|
var statuses: [Status] = try await statusFetcher.fetchFirstPage(client: client,
|
||||||
maxId: from.lastReadId,
|
timeline: timeline)
|
||||||
minId: nil,
|
|
||||||
offset: 0))
|
|
||||||
|
|
||||||
StatusDataControllerProvider.shared.updateDataControllers(for: statuses, client: client)
|
StatusDataControllerProvider.shared.updateDataControllers(for: statuses, client: client)
|
||||||
|
|
||||||
|
@ -286,10 +251,8 @@ extension TimelineViewModel: StatusesFetcher {
|
||||||
// And then we fetch statuses again toget newest statuses from there.
|
// And then we fetch statuses again toget newest statuses from there.
|
||||||
await fetchNewestStatuses(pullToRefresh: false)
|
await fetchNewestStatuses(pullToRefresh: false)
|
||||||
} else {
|
} else {
|
||||||
var statuses: [Status] = try await client.get(endpoint: timeline.endpoint(sinceId: nil,
|
var statuses: [Status] = try await statusFetcher.fetchFirstPage(client: client,
|
||||||
maxId: nil,
|
timeline: timeline)
|
||||||
minId: nil,
|
|
||||||
offset: 0))
|
|
||||||
|
|
||||||
StatusDataControllerProvider.shared.updateDataControllers(for: statuses, client: client)
|
StatusDataControllerProvider.shared.updateDataControllers(for: statuses, client: client)
|
||||||
|
|
||||||
|
@ -308,7 +271,8 @@ extension TimelineViewModel: StatusesFetcher {
|
||||||
canStreamEvents = false
|
canStreamEvents = false
|
||||||
let initialTimeline = timeline
|
let initialTimeline = timeline
|
||||||
|
|
||||||
let newStatuses = await fetchAndDedupNewStatuses(latestStatus: latestStatus, client: client)
|
let newStatuses = try await fetchAndDedupNewStatuses(latestStatus: latestStatus,
|
||||||
|
client: client)
|
||||||
|
|
||||||
guard !newStatuses.isEmpty,
|
guard !newStatuses.isEmpty,
|
||||||
isTimelineVisible,
|
isTimelineVisible,
|
||||||
|
@ -327,8 +291,11 @@ extension TimelineViewModel: StatusesFetcher {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private func fetchAndDedupNewStatuses(latestStatus: String, client: Client) async -> [Status] {
|
private func fetchAndDedupNewStatuses(latestStatus: String, client: Client) async throws -> [Status] {
|
||||||
var newStatuses = await fetchNewPages(minId: latestStatus, maxPages: 5)
|
var newStatuses = try await statusFetcher.fetchNewPages(client: client,
|
||||||
|
timeline: timeline,
|
||||||
|
minId: latestStatus,
|
||||||
|
maxPages: 5)
|
||||||
let ids = await datasource.get().map(\.id)
|
let ids = await datasource.get().map(\.id)
|
||||||
newStatuses = newStatuses.filter { status in
|
newStatuses = newStatuses.filter { status in
|
||||||
!ids.contains(where: { $0 == status.id })
|
!ids.contains(where: { $0 == status.id })
|
||||||
|
@ -378,34 +345,6 @@ extension TimelineViewModel: StatusesFetcher {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private func fetchNewPages(minId: String, maxPages: Int) async -> [Status] {
|
|
||||||
guard let client else { return [] }
|
|
||||||
var allStatuses: [Status] = []
|
|
||||||
var latestMinId = minId
|
|
||||||
do {
|
|
||||||
for _ in 1 ... maxPages {
|
|
||||||
if Task.isCancelled { break }
|
|
||||||
|
|
||||||
let newStatuses: [Status] = try await client.get(endpoint: timeline.endpoint(
|
|
||||||
sinceId: nil,
|
|
||||||
maxId: nil,
|
|
||||||
minId: latestMinId,
|
|
||||||
offset: nil
|
|
||||||
))
|
|
||||||
|
|
||||||
if newStatuses.isEmpty { break }
|
|
||||||
|
|
||||||
StatusDataControllerProvider.shared.updateDataControllers(for: newStatuses, client: client)
|
|
||||||
allStatuses.insert(contentsOf: newStatuses, at: 0)
|
|
||||||
latestMinId = newStatuses.first?.id ?? latestMinId
|
|
||||||
}
|
|
||||||
} catch {
|
|
||||||
return allStatuses
|
|
||||||
}
|
|
||||||
|
|
||||||
return allStatuses
|
|
||||||
}
|
|
||||||
|
|
||||||
enum NextPageError: Error {
|
enum NextPageError: Error {
|
||||||
case internalError
|
case internalError
|
||||||
}
|
}
|
||||||
|
@ -413,10 +352,10 @@ extension TimelineViewModel: StatusesFetcher {
|
||||||
func fetchNextPage() async throws {
|
func fetchNextPage() async throws {
|
||||||
let statuses = await datasource.get()
|
let statuses = await datasource.get()
|
||||||
guard let client, let lastId = statuses.last?.id else { throw NextPageError.internalError }
|
guard let client, let lastId = statuses.last?.id else { throw NextPageError.internalError }
|
||||||
let newStatuses: [Status] = try await client.get(endpoint: timeline.endpoint(sinceId: nil,
|
let newStatuses: [Status] = try await statusFetcher.fetchNextPage(client: client,
|
||||||
maxId: lastId,
|
timeline: timeline,
|
||||||
minId: nil,
|
lastId: lastId,
|
||||||
offset: statuses.count))
|
offset: statuses.count)
|
||||||
|
|
||||||
await datasource.append(contentOf: newStatuses)
|
await datasource.append(contentOf: newStatuses)
|
||||||
StatusDataControllerProvider.shared.updateDataControllers(for: newStatuses, client: client)
|
StatusDataControllerProvider.shared.updateDataControllers(for: newStatuses, client: client)
|
||||||
|
@ -441,7 +380,7 @@ extension TimelineViewModel: StatusesFetcher {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - MARKER
|
// MARK: - Marker handling
|
||||||
|
|
||||||
extension TimelineViewModel {
|
extension TimelineViewModel {
|
||||||
func fetchMarker() async -> Marker.Content? {
|
func fetchMarker() async -> Marker.Content? {
|
||||||
|
@ -466,3 +405,55 @@ extension TimelineViewModel {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MARK: - Event handling
|
||||||
|
|
||||||
|
extension TimelineViewModel {
|
||||||
|
func handleEvent(event: any StreamEvent) async {
|
||||||
|
guard let client = client, canStreamEvents, isTimelineVisible else { return }
|
||||||
|
|
||||||
|
switch event {
|
||||||
|
case let updateEvent as StreamEventUpdate:
|
||||||
|
await handleUpdateEvent(updateEvent, client: client)
|
||||||
|
case let deleteEvent as StreamEventDelete:
|
||||||
|
await handleDeleteEvent(deleteEvent)
|
||||||
|
case let statusUpdateEvent as StreamEventStatusUpdate:
|
||||||
|
await handleStatusUpdateEvent(statusUpdateEvent, client: client)
|
||||||
|
default:
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func handleUpdateEvent(_ event: StreamEventUpdate, client: Client) async {
|
||||||
|
guard timeline == .home,
|
||||||
|
await !datasource.contains(statusId: event.status.id) else { return }
|
||||||
|
|
||||||
|
pendingStatusesObserver.pendingStatuses.insert(event.status.id, at: 0)
|
||||||
|
await datasource.insert(event.status, at: 0)
|
||||||
|
await cache()
|
||||||
|
StatusDataControllerProvider.shared.updateDataControllers(for: [event.status], client: client)
|
||||||
|
await updateStatusesState()
|
||||||
|
}
|
||||||
|
|
||||||
|
private func handleDeleteEvent(_ event: StreamEventDelete) async {
|
||||||
|
await datasource.remove(event.status)
|
||||||
|
await cache()
|
||||||
|
await updateStatusesState()
|
||||||
|
}
|
||||||
|
|
||||||
|
private func handleStatusUpdateEvent(_ event: StreamEventStatusUpdate, client: Client) async {
|
||||||
|
guard let originalIndex = await datasource.indexOf(statusId: event.status.id) else { return }
|
||||||
|
|
||||||
|
StatusDataControllerProvider.shared.updateDataControllers(for: [event.status], client: client)
|
||||||
|
await datasource.replace(event.status, at: originalIndex)
|
||||||
|
await cache()
|
||||||
|
await updateStatusesState()
|
||||||
|
}
|
||||||
|
|
||||||
|
private func updateStatusesState() async {
|
||||||
|
let statuses = await datasource.getFiltered()
|
||||||
|
withAnimation {
|
||||||
|
statusesState = .display(statuses: statuses, nextPageState: .hasNextPage)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,60 @@
|
||||||
|
import Foundation
|
||||||
|
import Models
|
||||||
|
import Network
|
||||||
|
|
||||||
|
protocol TimelineStatusFetching: Sendable {
|
||||||
|
func fetchFirstPage(client: Client?,
|
||||||
|
timeline: TimelineFilter) async throws -> [Status]
|
||||||
|
func fetchNewPages(client: Client?,
|
||||||
|
timeline: TimelineFilter,
|
||||||
|
minId: String,
|
||||||
|
maxPages: Int) async throws -> [Status]
|
||||||
|
func fetchNextPage(client: Client?,
|
||||||
|
timeline: TimelineFilter,
|
||||||
|
lastId: String,
|
||||||
|
offset: Int) async throws -> [Status]
|
||||||
|
}
|
||||||
|
|
||||||
|
enum StatusFetcherError: Error {
|
||||||
|
case noClientAvailable
|
||||||
|
}
|
||||||
|
|
||||||
|
struct TimelineStatusFetcher: TimelineStatusFetching {
|
||||||
|
func fetchFirstPage(client: Client?, timeline: TimelineFilter) async throws -> [Status] {
|
||||||
|
guard let client = client else { throw StatusFetcherError.noClientAvailable }
|
||||||
|
return try await client.get(endpoint: timeline.endpoint(sinceId: nil,
|
||||||
|
maxId: nil,
|
||||||
|
minId: nil,
|
||||||
|
offset: 0))
|
||||||
|
}
|
||||||
|
|
||||||
|
func fetchNewPages(client: Client?, timeline: TimelineFilter, minId: String, maxPages: Int) async throws -> [Status] {
|
||||||
|
guard let client = client else { throw StatusFetcherError.noClientAvailable }
|
||||||
|
var allStatuses: [Status] = []
|
||||||
|
var latestMinId = minId
|
||||||
|
for _ in 1...maxPages {
|
||||||
|
if Task.isCancelled { break }
|
||||||
|
|
||||||
|
let newStatuses: [Status] = try await client.get(endpoint: timeline.endpoint(
|
||||||
|
sinceId: nil,
|
||||||
|
maxId: nil,
|
||||||
|
minId: latestMinId,
|
||||||
|
offset: nil
|
||||||
|
))
|
||||||
|
|
||||||
|
if newStatuses.isEmpty { break }
|
||||||
|
|
||||||
|
allStatuses.insert(contentsOf: newStatuses, at: 0)
|
||||||
|
latestMinId = newStatuses.first?.id ?? latestMinId
|
||||||
|
}
|
||||||
|
return allStatuses
|
||||||
|
}
|
||||||
|
|
||||||
|
func fetchNextPage(client: Client?, timeline: TimelineFilter, lastId: String, offset: Int) async throws -> [Status] {
|
||||||
|
guard let client = client else { throw StatusFetcherError.noClientAvailable }
|
||||||
|
return try await client.get(endpoint: timeline.endpoint(sinceId: nil,
|
||||||
|
maxId: lastId,
|
||||||
|
minId: nil,
|
||||||
|
offset: offset))
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue