import SwiftUI import Timeline import Network import KeychainSwift import Env import DesignSystem @main struct IceCubesApp: App { public static let defaultServer = "mastodon.social" @Environment(\.scenePhase) private var scenePhase @StateObject private var appAccountsManager = AppAccountsManager() @StateObject private var currentInstance = CurrentInstance() @StateObject private var currentAccount = CurrentAccount() @StateObject private var watcher = StreamWatcher() @StateObject private var quickLook = QuickLook() @StateObject private var theme = Theme() @State private var selectedTab: Tab = .timeline @State private var selectSidebarItem: Tab? = .timeline @State private var popToRootTab: Tab = .other private var availableTabs: [Tab] { appAccountsManager.currentClient.isAuth ? Tab.loggedInTabs() : Tab.loggedOutTab() } var body: some Scene { WindowGroup { appView .tint(theme.tintColor) .onAppear { setNewClientsInEnv(client: appAccountsManager.currentClient) setBarsColor(color: theme.primaryBackgroundColor) } .preferredColorScheme(theme.selectedScheme == ColorScheme.dark ? .dark : .light) .environmentObject(appAccountsManager) .environmentObject(appAccountsManager.currentClient) .environmentObject(quickLook) .environmentObject(currentAccount) .environmentObject(currentInstance) .environmentObject(theme) .environmentObject(watcher) } .onChange(of: scenePhase, perform: { scenePhase in handleScenePhase(scenePhase: scenePhase) }) .onChange(of: appAccountsManager.currentClient) { newClient in setNewClientsInEnv(client: newClient) if newClient.isAuth { watcher.watch(streams: [.user, .direct]) } } .onChange(of: theme.primaryBackgroundColor) { newValue in setBarsColor(color: newValue) } } @ViewBuilder private var appView: some View { if UIDevice.current.userInterfaceIdiom == .pad || UIDevice.current.userInterfaceIdiom == .mac { if ProcessInfo.processInfo.isiOSAppOnMac { splitView } else { splitView .quickLookPreview($quickLook.url, in: quickLook.urls) } } else { tabBarView .quickLookPreview($quickLook.url, in: quickLook.urls) } } private func badgeFor(tab: Tab) -> Int { if tab == .notifications && selectedTab != tab { return watcher.unreadNotificationsCount } else if tab == .messages && selectedTab != tab { return watcher.unreadMessagesCount } return 0 } private var tabBarView: some View { TabView(selection: .init(get: { selectedTab }, set: { newTab in if newTab == selectedTab { /// Stupid hack to trigger onChange binding in tab views. popToRootTab = .other DispatchQueue.main.asyncAfter(deadline: .now() + 0.01) { popToRootTab = selectedTab } } selectedTab = newTab })) { ForEach(availableTabs) { tab in tab.makeContentView(popToRootTab: $popToRootTab) .tabItem { tab.label } .tag(tab) .badge(badgeFor(tab: tab)) .toolbarBackground(theme.primaryBackgroundColor.opacity(0.50), for: .tabBar) } } } private var splitView: some View { NavigationSplitView { List(availableTabs, selection: $selectSidebarItem) { tab in NavigationLink(value: tab) { tab.label } } .scrollContentBackground(.hidden) .background(theme.secondaryBackgroundColor) .navigationSplitViewColumnWidth(200) } detail: { selectSidebarItem?.makeContentView(popToRootTab: $popToRootTab) } } private func setNewClientsInEnv(client: Client) { currentAccount.setClient(client: client) currentInstance.setClient(client: client) watcher.setClient(client: client) } private func handleScenePhase(scenePhase: ScenePhase) { switch scenePhase { case .background: watcher.stopWatching() case .active: watcher.watch(streams: [.user, .direct]) case .inactive: break default: break } } private func setBarsColor(color: Color) { UINavigationBar.appearance().isTranslucent = true UINavigationBar.appearance().barTintColor = UIColor(color) } }