metatext/ServiceLayer/Sources/ServiceLayer/Services/TimelineService.swift

89 lines
3.6 KiB
Swift
Raw Normal View History

2020-10-05 07:04:15 +00:00
// Copyright © 2020 Metabolist. All rights reserved.
import Combine
import DB
import Foundation
import Mastodon
import MastodonAPI
2020-10-05 19:11:41 +00:00
public struct TimelineService {
2021-01-23 06:15:52 +00:00
public let sections: AnyPublisher<[CollectionSection], Error>
2020-10-05 07:04:15 +00:00
public let navigationService: NavigationService
2020-10-05 22:50:05 +00:00
public let nextPageMaxId: AnyPublisher<String, Never>
2021-02-08 01:46:51 +00:00
public let accountIdsForRelationships: AnyPublisher<Set<Account.Id>, Never>
2020-10-05 20:21:06 +00:00
public let title: AnyPublisher<String, Never>
2021-01-27 23:49:13 +00:00
public let titleLocalizationComponents: AnyPublisher<[String], Never>
2021-02-21 23:00:56 +00:00
public let announcesNewItems = true
2020-10-05 07:04:15 +00:00
private let timeline: Timeline
private let mastodonAPIClient: MastodonAPIClient
private let contentDatabase: ContentDatabase
2020-12-01 03:07:38 +00:00
private let nextPageMaxIdSubject = PassthroughSubject<String, Never>()
2021-02-08 01:46:51 +00:00
private let accountIdsForRelationshipsSubject = PassthroughSubject<Set<Account.Id>, Never>()
2020-10-05 07:04:15 +00:00
init(timeline: Timeline,
environment: AppEnvironment,
mastodonAPIClient: MastodonAPIClient,
contentDatabase: ContentDatabase) {
2020-10-05 07:04:15 +00:00
self.timeline = timeline
self.mastodonAPIClient = mastodonAPIClient
self.contentDatabase = contentDatabase
2022-11-21 01:25:04 +00:00
if case .home = timeline {
sections = contentDatabase.cleanHomeTimelinePublisher()
.collect()
.flatMap { _ in contentDatabase.timelinePublisher(timeline) }
.eraseToAnyPublisher()
} else {
sections = contentDatabase.timelinePublisher(timeline)
}
navigationService = NavigationService(environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)
2020-12-01 03:07:38 +00:00
nextPageMaxId = nextPageMaxIdSubject.eraseToAnyPublisher()
2021-02-08 01:46:51 +00:00
accountIdsForRelationships = accountIdsForRelationshipsSubject.eraseToAnyPublisher()
2020-10-05 07:04:15 +00:00
2021-01-27 23:49:13 +00:00
switch timeline {
case let .list(list):
title = Just(list.title).eraseToAnyPublisher()
titleLocalizationComponents = Empty().eraseToAnyPublisher()
case let .tag(tag):
2020-10-05 20:21:06 +00:00
title = Just("#".appending(tag)).eraseToAnyPublisher()
2021-01-27 23:49:13 +00:00
titleLocalizationComponents = Empty().eraseToAnyPublisher()
case .favorites:
title = Empty().eraseToAnyPublisher()
titleLocalizationComponents = Just(["favorites"]).eraseToAnyPublisher()
case .bookmarks:
title = Empty().eraseToAnyPublisher()
titleLocalizationComponents = Just(["bookmarks"]).eraseToAnyPublisher()
default:
2020-10-05 20:21:06 +00:00
title = Empty().eraseToAnyPublisher()
2021-01-27 23:49:13 +00:00
titleLocalizationComponents = Empty().eraseToAnyPublisher()
2020-10-05 07:04:15 +00:00
}
}
2020-10-05 19:11:41 +00:00
}
2020-10-05 07:04:15 +00:00
2020-10-05 19:11:41 +00:00
extension TimelineService: CollectionService {
2021-02-08 20:33:51 +00:00
public var positionTimeline: Timeline? { timeline }
2021-01-31 15:42:44 +00:00
public var preferLastPresentIdOverNextPageMaxId: Bool {
!timeline.ordered
}
2021-01-23 03:48:33 +00:00
public func request(maxId: String?, minId: String?, search: Search?) -> AnyPublisher<Never, Error> {
2020-10-05 22:50:05 +00:00
mastodonAPIClient.pagedRequest(timeline.endpoint, maxId: maxId, minId: minId)
2020-10-05 20:21:06 +00:00
.handleEvents(receiveOutput: {
2020-12-01 03:07:38 +00:00
if let maxId = $0.info.maxId {
nextPageMaxIdSubject.send(maxId)
}
2021-02-08 01:46:51 +00:00
accountIdsForRelationshipsSubject.send(
Set($0.result.map(\.account.id))
.union(Set($0.result.compactMap(\.reblog?.account.id))))
2020-10-05 20:21:06 +00:00
})
2020-10-05 07:04:15 +00:00
.flatMap { contentDatabase.insert(statuses: $0.result, timeline: timeline) }
.eraseToAnyPublisher()
}
}