Impressia/Vernissage/Views/NotificationsView.swift
2023-01-23 18:01:27 +01:00

207 lines
8.0 KiB
Swift

//
// https://mczachurski.dev
// Copyright © 2023 Marcin Czachurski and the repository contributors.
// Licensed under the MIT License.
//
import SwiftUI
import MastodonKit
struct NotificationsView: View {
@EnvironmentObject var applicationState: ApplicationState
@State var accountId: String
@State private var notifications: [MastodonKit.Notification] = []
@State private var allItemsLoaded = false
@State private var firstLoadFinished = false
@State private var minId: String?
@State private var maxId: String?
private let defaultPageSize = 20
var body: some View {
List {
ForEach(notifications, id: \.id) { notification in
switch notification.type {
case .favourite, .reblog, .mention, .status, .poll, .update:
if let status = notification.status, let statusViewModel = StatusViewModel(status: status) {
NavigationLink(value: RouteurDestinations.status(
id: statusViewModel.id,
blurhash: statusViewModel.mediaAttachments.first?.blurhash,
metaImageWidth: statusViewModel.getImageWidth(),
metaImageHeight: statusViewModel.getImageHeight())
) {
NotificationRow(notification: notification)
}
.buttonStyle(EmptyButtonStyle())
}
case .follow, .followRequest, .adminSignUp:
NavigationLink(value: RouteurDestinations.userProfile(
accountId: notification.account.id,
accountDisplayName: notification.account.displayNameWithoutEmojis,
accountUserName: notification.account.acct)
) {
NotificationRow(notification: notification)
}
case .adminReport:
if let targetAccount = notification.report?.targetAccount {
NavigationLink(value: RouteurDestinations.userProfile(
accountId: targetAccount.id,
accountDisplayName: targetAccount.displayNameWithoutEmojis,
accountUserName: targetAccount.acct)
) {
NotificationRow(notification: notification)
}
}
}
}
if allItemsLoaded == false && firstLoadFinished == true {
HStack {
Spacer()
LoadingIndicator()
.task {
await self.loadMoreNotifications()
}
Spacer()
}
.listRowSeparator(.hidden)
}
}.overlay {
if firstLoadFinished == false {
LoadingIndicator()
} else {
if self.notifications.isEmpty {
VStack {
Image(systemName: "bell")
.font(.largeTitle)
.padding(.bottom, 4)
Text("Unfortunately, there is nothing here.")
.font(.title3)
}.foregroundColor(.lightGrayColor)
}
}
}
.navigationBarTitle("Notifications")
.listStyle(PlainListStyle())
.refreshable {
await self.loadNewNotifications()
}
.task {
if self.notifications.isEmpty == false {
return
}
await self.loadNotifications()
}
}
func loadNotifications() async {
do {
let linkable = try await NotificationService.shared.notifications(
forAccountId: self.accountId,
andContext: self.applicationState.accountData,
maxId: maxId,
minId: minId,
limit: 5)
await self.downloadAllImages(notifications: linkable.data)
self.minId = linkable.link?.minId
self.maxId = linkable.link?.maxId
self.notifications = linkable.data
if linkable.data.isEmpty || linkable.data.count < 5 {
self.allItemsLoaded = true
}
self.firstLoadFinished = true
} catch {
ErrorService.shared.handle(error, message: "Error during download notifications from server.", showToastr: !Task.isCancelled)
}
}
private func loadMoreNotifications() async {
do {
let linkable = try await NotificationService.shared.notifications(
forAccountId: self.accountId,
andContext: self.applicationState.accountData,
maxId: self.maxId,
limit: self.defaultPageSize)
await self.downloadAllImages(notifications: linkable.data)
self.maxId = linkable.link?.maxId
self.notifications.append(contentsOf: linkable.data)
if linkable.data.isEmpty || linkable.data.count < self.defaultPageSize {
self.allItemsLoaded = true
}
} catch {
ErrorService.shared.handle(error, message: "Error during download notifications from server.", showToastr: !Task.isCancelled)
}
}
private func loadNewNotifications() async {
do {
let linkable = try await NotificationService.shared.notifications(
forAccountId: self.accountId,
andContext: self.applicationState.accountData,
minId: self.minId,
limit: self.defaultPageSize)
if let first = linkable.data.first, self.notifications.contains(where: { notification in notification.id == first.id }) {
// We have all notifications, we don't have to do anything.
return
}
await self.downloadAllImages(notifications: linkable.data)
self.minId = linkable.link?.minId
self.notifications.insert(contentsOf: linkable.data, at: 0)
} catch {
ErrorService.shared.handle(error, message: "Error during download notifications from server.", showToastr: !Task.isCancelled)
}
}
private func downloadAllImages(notifications: [MastodonKit.Notification]) async {
// Download all avatars into cache.
let accounts = notifications.map({ notification in notification.account })
await self.downloadAvatars(accounts: accounts)
// Download all images into cache.
var images: [(id: String, url: URL)] = []
for notification in notifications {
if let mediaAttachment = notification.status?.mediaAttachments {
images.append(contentsOf:
mediaAttachment
.filter({ attachment in
attachment.type == MediaAttachment.MediaAttachmentType.image
})
.map({
attachment in (id: attachment.id, url: attachment.url)
}))
}
}
await self.downloadImages(images: images)
}
private func downloadAvatars(accounts: [Account]) async {
await withTaskGroup(of: Void.self) { group in
for account in accounts {
group.addTask { await CacheAvatarService.shared.downloadImage(for: account.id, avatarUrl: account.avatar) }
}
}
}
private func downloadImages(images: [(id: String, url: URL)]) async {
await withTaskGroup(of: Void.self) { group in
for image in images {
group.addTask { await CacheImageService.shared.downloadImage(for: image.id, url: image.url) }
}
}
}
}