Bubble/Threaded/Data/Navigator.swift

261 lines
8.3 KiB
Swift
Raw Normal View History

2023-12-29 11:17:37 +01:00
//Made by Lumaa
import Foundation
import SwiftUI
@Observable
public class Navigator: ObservableObject {
2023-12-29 11:17:37 +01:00
public var path: [RouterDestination] = []
public var presentedSheet: SheetDestination?
public var presentedCover: SheetDestination?
2023-12-29 11:17:37 +01:00
public var selectedTab: TabDestination = .timeline
public var showTabbar: Bool = true
2023-12-29 11:17:37 +01:00
public func navigate(to: RouterDestination) {
path.append(to)
if path.contains(where: { $0 == .settings }) {
toggleTabbar(false)
} else {
toggleTabbar(true)
}
2023-12-29 11:17:37 +01:00
}
2024-01-28 17:51:50 +01:00
public func removeSettingsOfPath() {
self.path = self.path.filter({ !RouterDestination.allSettings.contains($0) })
}
/// Defines the visibility of the main tab bar in from `ContentView`
/// - Parameter bool: `true` shows the tab bar and `false` hides the tab bar
public func toggleTabbar(_ bool: Bool? = nil) {
print("\((bool ?? !self.showTabbar) ? "shown" : "hide") the tab bar")
withAnimation(.easeInOut(duration: 0.4)) {
self.showTabbar = bool ?? !self.showTabbar
}
}
2023-12-29 11:17:37 +01:00
}
public class UniversalNavigator: Navigator {}
2023-12-29 11:17:37 +01:00
public enum TabDestination: Identifiable {
case timeline
case search
case activity
case profile
public var id: String {
switch self {
case .timeline:
return "timeline"
case .search:
return "search"
case .activity:
return "activity"
case .profile:
return "profile"
}
}
}
public enum SheetDestination: Identifiable {
case welcome
2024-01-27 08:56:22 +01:00
case shop
2023-12-29 11:17:37 +01:00
case mastodonLogin(logged: Binding<Bool>)
case post(content: String = "", replyId: String? = nil, editId: String? = nil)
2024-01-21 13:08:06 +01:00
case safari(url: URL)
case shareImage(image: UIImage, status: Status)
2023-12-29 11:17:37 +01:00
public var id: String {
switch self {
case .welcome:
return "welcome"
2024-01-27 08:56:22 +01:00
case .shop:
return "shop"
2023-12-29 11:17:37 +01:00
case .mastodonLogin:
return "login"
case .post:
return "post"
2024-01-21 13:08:06 +01:00
case .safari:
return "safari"
2024-01-21 16:34:26 +01:00
case .shareImage:
return "shareImage"
2023-12-29 11:17:37 +01:00
}
}
public var isCover: Bool {
switch self {
case .welcome:
return true
2024-01-27 08:56:22 +01:00
case .shop:
return true
2023-12-29 11:17:37 +01:00
case .mastodonLogin:
return false
case .post:
return false
2024-01-21 13:08:06 +01:00
case .safari:
return false
2024-01-21 16:34:26 +01:00
case .shareImage:
return false
2023-12-29 11:17:37 +01:00
}
}
}
public enum RouterDestination: Hashable {
case settings
case privacy
case appearence
2023-12-29 11:17:37 +01:00
case account(acc: Account)
2024-01-10 17:45:41 +01:00
case post(status: Status)
case about
case contacts
case timeline(timeline: TimelineFilter?)
2023-12-29 11:17:37 +01:00
}
2024-01-28 17:51:50 +01:00
extension RouterDestination {
static let allSettings: [RouterDestination] = [.settings, .privacy, .about, .appearence]
}
2023-12-29 11:17:37 +01:00
extension View {
func withAppRouter(_ navigator: Navigator) -> some View {
2023-12-29 11:17:37 +01:00
navigationDestination(for: RouterDestination.self) { destination in
switch destination {
case .settings:
SettingsView(navigator: navigator)
2023-12-29 11:17:37 +01:00
case .privacy:
PrivacyView()
case .appearence:
AppearenceView()
2023-12-29 11:17:37 +01:00
case .account(let acc):
ProfileView(account: acc)
2024-01-10 17:45:41 +01:00
case .post(let status):
PostDetailsView(status: status)
case .about:
AboutView()
case .contacts:
ContactsView()
case .timeline(let timeline):
PostsView(filter: timeline ?? .home, showHero: false)
2023-12-29 11:17:37 +01:00
}
}
}
func withSheets(sheetDestination: Binding<SheetDestination?>) -> some View {
sheet(item: sheetDestination) { destination in
viewSheet(destination: destination)
2023-12-29 11:17:37 +01:00
}
}
func withCovers(sheetDestination: Binding<SheetDestination?>) -> some View {
fullScreenCover(item: sheetDestination) { destination in
viewCover(destination: destination)
2023-12-29 11:17:37 +01:00
}
}
@available(*, deprecated, renamed: "withSheets", message: "These two cannot support themselves")
func withOver(sheetDestination: Binding<SheetDestination?>) -> some View {
self
.withCovers(sheetDestination: sheetDestination)
.withSheets(sheetDestination: sheetDestination)
}
@available(*, deprecated, message: "Causes bugs with sheets to display as covers")
2023-12-29 11:17:37 +01:00
private func viewRepresentation(destination: SheetDestination, isCover: Bool) -> some View {
Group {
if destination.isCover {
switch destination {
case .welcome:
ConnectView()
2024-01-27 08:56:22 +01:00
case .shop:
ShopView()
2023-12-29 11:17:37 +01:00
default:
2024-01-27 08:56:22 +01:00
EmptySheetView(destId: destination.id)
2023-12-29 11:17:37 +01:00
}
} else {
switch destination {
case .post(let content, let replyId, let editId):
NavigationStack {
PostingView(initialString: content, replyId: replyId, editId: editId)
.tint(Color(uiColor: UIColor.label))
}
2023-12-29 11:17:37 +01:00
case let .mastodonLogin(logged):
AddInstanceView(logged: logged)
.tint(Color.accentColor)
2024-01-21 13:08:06 +01:00
case let .safari(url):
SfSafariView(url: url)
.ignoresSafeArea()
case let .shareImage(image, status):
ShareSheet(image: image, status: status)
2023-12-29 11:17:37 +01:00
default:
2024-01-27 08:56:22 +01:00
EmptySheetView(destId: destination.id)
2023-12-29 11:17:37 +01:00
}
}
}
}
private func viewCover(destination: SheetDestination) -> some View {
Group {
switch destination {
case .welcome:
ConnectView()
case .shop:
ShopView()
default:
EmptySheetView(destId: destination.id)
}
}
}
private func viewSheet(destination: SheetDestination) -> some View {
Group {
switch destination {
case .post(let content, let replyId, let editId):
NavigationStack {
PostingView(initialString: content, replyId: replyId, editId: editId)
.tint(Color(uiColor: UIColor.label))
}
case let .mastodonLogin(logged):
AddInstanceView(logged: logged)
.tint(Color.accentColor)
case let .safari(url):
SfSafariView(url: url)
.ignoresSafeArea()
case let .shareImage(image, status):
ShareSheet(image: image, status: status)
default:
EmptySheetView(destId: destination.id)
}
}
}
2023-12-29 11:17:37 +01:00
}
2024-01-27 08:56:22 +01:00
/// This view is visible when the `viewRepresentation(destination: SheetDestination)` doesn't support the given `SheetDestination`
2024-01-27 08:56:22 +01:00
private struct EmptySheetView: View {
@Environment(\.dismiss) private var dismiss
var destId: String = ""
2024-01-28 17:51:50 +01:00
let str: String = .init(localized: "about.version-\(AppInfo.appVersion)")
2024-01-27 08:56:22 +01:00
var body: some View {
ZStack {
Rectangle()
.fill(Color.red.gradient)
2024-01-27 08:56:22 +01:00
.ignoresSafeArea()
VStack {
ContentUnavailableView(String("Missing view for \"\(destId.isEmpty ? "[EMPTY_DEST_ID]" : destId)\""), systemImage: "exclamationmark.triangle.fill", description: Text(String("Please notify Lumaa as soon as possible!\n\n\(str)")))
.foregroundStyle(.white)
Button {
dismiss()
} label: {
Text(String("Dismiss"))
}
.buttonStyle(LargeButton(filled: true))
}
2024-01-27 08:56:22 +01:00
}
}
}