import AVFAudio import Defaults import MediaPlayer import SDWebImage import SDWebImagePINPlugin import SDWebImageWebPCoder import Siesta import SwiftUI struct ContentView: View { @EnvironmentObject private var accounts @EnvironmentObject private var comments @EnvironmentObject private var instances @EnvironmentObject private var navigation @EnvironmentObject private var player @EnvironmentObject private var playerControls @EnvironmentObject private var playlists @EnvironmentObject private var recents @EnvironmentObject private var search @EnvironmentObject private var subscriptions @EnvironmentObject private var thumbnailsModel @EnvironmentObject private var menu #if os(iOS) @Environment(\.horizontalSizeClass) private var horizontalSizeClass #endif var body: some View { Group { #if os(iOS) if horizontalSizeClass == .compact { AppTabNavigation() } else { AppSidebarNavigation() } #elseif os(macOS) AppSidebarNavigation() #elseif os(tvOS) TVNavigationView() #endif } .onAppear(perform: configure) .onChange(of: accounts.signedIn) { _ in subscriptions.load(force: true) playlists.load(force: true) } .environmentObject(accounts) .environmentObject(comments) .environmentObject(instances) .environmentObject(navigation) .environmentObject(player) .environmentObject(playlists) .environmentObject(recents) .environmentObject(search) .environmentObject(subscriptions) .environmentObject(thumbnailsModel) #if os(iOS) .overlay(videoPlayer) #endif // iOS 14 has problem with multiple sheets in one view // but it's ok when it's in background .background( EmptyView().sheet(isPresented: $navigation.presentingWelcomeScreen) { WelcomeScreen() .environmentObject(accounts) .environmentObject(navigation) } ) #if !os(tvOS) .onOpenURL { OpenURLHandler(accounts: accounts, player: player).handle($0) } .background( EmptyView().sheet(isPresented: $navigation.presentingAddToPlaylist) { AddToPlaylistView(video: navigation.videoToAddToPlaylist) .environmentObject(playlists) } ) .background( EmptyView().sheet(isPresented: $navigation.presentingPlaylistForm) { PlaylistFormView(playlist: $navigation.editedPlaylist) .environmentObject(accounts) .environmentObject(playlists) } ) .background( EmptyView().sheet(isPresented: $navigation.presentingSettings, onDismiss: openWelcomeScreenIfAccountEmpty) { SettingsView() .environmentObject(accounts) .environmentObject(instances) .environmentObject(player) } ) #endif .alert(isPresented: $navigation.presentingUnsubscribeAlert) { Alert( title: Text( "Are you sure you want to unsubscribe from \(navigation.channelToUnsubscribe.name)?" ), primaryButton: .destructive(Text("Unsubscribe")) { subscriptions.unsubscribe(navigation.channelToUnsubscribe.id) }, secondaryButton: .cancel() ) } } func configure() { SiestaLog.Category.enabled = .common SDImageCodersManager.shared.addCoder(SDImageWebPCoder.shared) SDWebImageManager.defaultImageCache = PINCache(name: "stream.yattee.app") #if !os(macOS) setupNowPlayingInfoCenter() #endif #if os(iOS) if Defaults[.lockPortraitWhenBrowsing] { Orientation.lockOrientation(.portrait, andRotateTo: .portrait) } #endif if let account = accounts.lastUsed ?? instances.lastUsed?.anonymousAccount ?? InstancesModel.all.first?.anonymousAccount { accounts.setCurrent(account) } if accounts.current.isNil { navigation.presentingWelcomeScreen = true } playlists.accounts = accounts search.accounts = accounts subscriptions.accounts = accounts comments.player = player menu.accounts = accounts menu.navigation = navigation menu.player = player playerControls.player = player player.accounts = accounts player.comments = comments player.controls = playerControls if !accounts.current.isNil { player.restoreQueue() } if !Defaults[.saveRecents] { recents.clear() } var section = Defaults[.visibleSections].min()?.tabSelection #if os(macOS) if section == .playlists { section = .search } #endif navigation.tabSelection = section ?? .search subscriptions.load() playlists.load() } func setupNowPlayingInfoCenter() { #if !os(macOS) try? AVAudioSession.sharedInstance().setCategory(.playback, mode: .moviePlayback) UIApplication.shared.beginReceivingRemoteControlEvents() #endif MPRemoteCommandCenter.shared().playCommand.addTarget { _ in player.play() return .success } MPRemoteCommandCenter.shared().pauseCommand.addTarget { _ in player.pause() return .success } MPRemoteCommandCenter.shared().previousTrackCommand.isEnabled = false MPRemoteCommandCenter.shared().nextTrackCommand.isEnabled = false MPRemoteCommandCenter.shared().changePlaybackPositionCommand.addTarget { remoteEvent in guard let event = remoteEvent as? MPChangePlaybackPositionCommandEvent else { return .commandFailed } player.backend.seek(to: event.positionTime) return .success } let skipForwardCommand = MPRemoteCommandCenter.shared().skipForwardCommand skipForwardCommand.isEnabled = true skipForwardCommand.preferredIntervals = [10] skipForwardCommand.addTarget { _ in player.backend.seek(relative: .secondsInDefaultTimescale(10)) return .success } let skipBackwardCommand = MPRemoteCommandCenter.shared().skipBackwardCommand skipBackwardCommand.isEnabled = true skipBackwardCommand.preferredIntervals = [10] skipBackwardCommand.addTarget { _ in player.backend.seek(relative: .secondsInDefaultTimescale(-10)) return .success } } func openWelcomeScreenIfAccountEmpty() { guard Defaults[.instances].isEmpty else { return } navigation.presentingWelcomeScreen = true } var videoPlayer: some View { VideoPlayerView() .environmentObject(accounts) .environmentObject(comments) .environmentObject(instances) .environmentObject(navigation) .environmentObject(player) .environmentObject(playerControls) .environmentObject(playlists) .environmentObject(recents) .environmentObject(subscriptions) .environmentObject(thumbnailsModel) .environment(\.navigationStyle, .sidebar) } } struct ContentView_Previews: PreviewProvider { static var previews: some View { ContentView() .injectFixtureEnvironmentObjects() } }