IceCubesApp/Packages/Account/Sources/Account/Follow/FollowButton.swift

130 lines
3.9 KiB
Swift
Raw Normal View History

import Combine
2022-12-23 15:21:31 +00:00
import Foundation
import Models
import Network
2023-01-17 10:36:01 +00:00
import SwiftUI
2022-12-23 15:21:31 +00:00
@MainActor
public class FollowButtonViewModel: ObservableObject {
var client: Client?
2023-01-17 10:36:01 +00:00
2022-12-23 15:21:31 +00:00
public let accountId: String
public let shouldDisplayNotify: Bool
2023-01-22 05:38:30 +00:00
public let relationshipUpdated: (Relationship) -> Void
@Published public private(set) var relationship: Relationship
2023-01-17 10:36:01 +00:00
@Published public private(set) var isUpdating: Bool = false
public init(accountId: String,
relationship: Relationship,
shouldDisplayNotify: Bool,
2023-01-22 05:38:30 +00:00
relationshipUpdated: @escaping ((Relationship) -> Void))
{
2022-12-23 15:21:31 +00:00
self.accountId = accountId
self.relationship = relationship
self.shouldDisplayNotify = shouldDisplayNotify
self.relationshipUpdated = relationshipUpdated
2022-12-23 15:21:31 +00:00
}
2023-01-17 10:36:01 +00:00
2022-12-23 15:21:31 +00:00
func follow() async {
guard let client else { return }
isUpdating = true
do {
relationship = try await client.post(endpoint: Accounts.follow(id: accountId, notify: false, reblogs: true))
relationshipUpdated(relationship)
2022-12-23 15:21:31 +00:00
} catch {
print("Error while following: \(error.localizedDescription)")
}
isUpdating = false
}
2023-01-17 10:36:01 +00:00
2022-12-23 15:21:31 +00:00
func unfollow() async {
guard let client else { return }
isUpdating = true
do {
relationship = try await client.post(endpoint: Accounts.unfollow(id: accountId))
relationshipUpdated(relationship)
2022-12-23 15:21:31 +00:00
} catch {
print("Error while unfollowing: \(error.localizedDescription)")
}
isUpdating = false
}
2023-01-17 10:36:01 +00:00
func toggleNotify() async {
guard let client else { return }
do {
relationship = try await client.post(endpoint: Accounts.follow(id: accountId,
notify: !relationship.notifying,
reblogs: relationship.showingReblogs))
relationshipUpdated(relationship)
} catch {
print("Error while following: \(error.localizedDescription)")
}
}
func toggleReboosts() async {
guard let client else { return }
do {
relationship = try await client.post(endpoint: Accounts.follow(id: accountId,
notify: relationship.notifying,
reblogs: !relationship.showingReblogs))
relationshipUpdated(relationship)
} catch {
print("Error while switching reboosts: \(error.localizedDescription)")
}
}
2022-12-23 15:21:31 +00:00
}
public struct FollowButton: View {
@EnvironmentObject private var client: Client
@StateObject private var viewModel: FollowButtonViewModel
2023-01-17 10:36:01 +00:00
2022-12-23 15:21:31 +00:00
public init(viewModel: FollowButtonViewModel) {
_viewModel = StateObject(wrappedValue: viewModel)
}
2023-01-17 10:36:01 +00:00
2022-12-23 15:21:31 +00:00
public var body: some View {
VStack(alignment: .trailing) {
Button {
Task {
if viewModel.relationship.following {
await viewModel.unfollow()
} else {
await viewModel.follow()
}
}
} label: {
if viewModel.relationship.requested == true {
Text("account.follow.requested")
2022-12-23 15:21:31 +00:00
} else {
Text(viewModel.relationship.following ? "account.follow.following" : "account.follow.follow")
2022-12-23 15:21:31 +00:00
}
}
if viewModel.relationship.following,
viewModel.shouldDisplayNotify
{
HStack {
Button {
Task {
await viewModel.toggleNotify()
}
} label: {
Image(systemName: viewModel.relationship.notifying ? "bell.fill" : "bell")
}
Button {
Task {
await viewModel.toggleReboosts()
}
} label: {
Image(systemName: viewModel.relationship.showingReblogs ? "arrow.left.arrow.right.circle.fill" : "arrow.left.arrow.right.circle")
}
}
2022-12-23 15:21:31 +00:00
}
}
.buttonStyle(.bordered)
.disabled(viewModel.isUpdating)
2022-12-23 15:21:31 +00:00
.onAppear {
viewModel.client = client
}
}
}