// // ContentView.swift // JellyfinPlayer // // Created by Aiden Vigue on 4/29/21. // import SwiftUI import KeychainSwift import SwiftyRequest import SwiftyJSON import Introspect import Sentry import SDWebImageSwiftUI class GlobalData: ObservableObject { @Published var user: SignedInUser? @Published var authToken: String = "" @Published var server: Server? @Published var authHeader: String = "" } extension View { func withHostingWindow(_ callback: @escaping (UIWindow?) -> Void) -> some View { self.background(HostingWindowFinder(callback: callback)) } } struct HostingWindowFinder: UIViewRepresentable { var callback: (UIWindow?) -> () func makeUIView(context: Context) -> UIView { let view = UIView() DispatchQueue.main.async { [weak view] in self.callback(view?.window) } return view } func updateUIView(_ uiView: UIView, context: Context) { } } struct PrefersHomeIndicatorAutoHiddenPreferenceKey: PreferenceKey { typealias Value = Bool static var defaultValue: Value = false static func reduce(value: inout Value, nextValue: () -> Value) { value = nextValue() || value } } struct ViewPreferenceKey: PreferenceKey { typealias Value = UIUserInterfaceStyle static var defaultValue: UIUserInterfaceStyle = .unspecified static func reduce(value: inout UIUserInterfaceStyle, nextValue: () -> UIUserInterfaceStyle) { value = nextValue() } } struct SupportedOrientationsPreferenceKey: PreferenceKey { typealias Value = UIInterfaceOrientationMask static var defaultValue: UIInterfaceOrientationMask = .allButUpsideDown static func reduce(value: inout UIInterfaceOrientationMask, nextValue: () -> UIInterfaceOrientationMask) { // use the most restrictive set from the stack value.formIntersection(nextValue()) } } extension View { /// Navigate to a new view. /// - Parameters: /// - view: View to navigate to. /// - binding: Only navigates when this condition is `true`. func navigate(to view: NewView, when binding: Binding) -> some View { NavigationView { ZStack { self .navigationBarTitle("") .navigationBarHidden(true) NavigationLink( destination: view .navigationBarTitle("") .navigationBarHidden(true), isActive: binding ) { EmptyView() } } } } } class PreferenceUIHostingController: UIHostingController { init(wrappedView: V) { let box = Box() super.init(rootView: AnyView(wrappedView .onPreferenceChange(PrefersHomeIndicatorAutoHiddenPreferenceKey.self) { box.value?._prefersHomeIndicatorAutoHidden = $0 }.onPreferenceChange(SupportedOrientationsPreferenceKey.self) { box.value?._orientations = $0 }.onPreferenceChange(ViewPreferenceKey.self) { box.value?._viewPreference = $0 } )) box.value = self } @objc required dynamic init?(coder aDecoder: NSCoder) { super.init(coder: aDecoder) } private class Box { weak var value: PreferenceUIHostingController? init() {} } // MARK: Prefers Home Indicator Auto Hidden public var _prefersHomeIndicatorAutoHidden = false { didSet { setNeedsUpdateOfHomeIndicatorAutoHidden() } } override var prefersHomeIndicatorAutoHidden: Bool { _prefersHomeIndicatorAutoHidden } // MARK: Lock orientation public var _orientations: UIInterfaceOrientationMask = .allButUpsideDown { didSet { UIViewController.attemptRotationToDeviceOrientation(); if(_orientations == .landscapeRight) { let value = UIInterfaceOrientation.landscapeRight.rawValue; UIDevice.current.setValue(value, forKey: "orientation") } } }; override var supportedInterfaceOrientations: UIInterfaceOrientationMask { _orientations } public var _viewPreference: UIUserInterfaceStyle = .unspecified { didSet { overrideUserInterfaceStyle = _viewPreference } }; } extension View { // Controls the application's preferred home indicator auto-hiding when this view is shown. func prefersHomeIndicatorAutoHidden(_ value: Bool) -> some View { preference(key: PrefersHomeIndicatorAutoHiddenPreferenceKey.self, value: value) } func supportedOrientations(_ supportedOrientations: UIInterfaceOrientationMask) -> some View { // When rendered, export the requested orientations upward to Root preference(key: SupportedOrientationsPreferenceKey.self, value: supportedOrientations) } func overrideViewPreference(_ viewPreference: UIUserInterfaceStyle) -> some View { // When rendered, export the requested orientations upward to Root preference(key: ViewPreferenceKey.self, value: viewPreference) } } struct ContentView: View { @Environment(\.managedObjectContext) private var viewContext @StateObject private var globalData = GlobalData() @EnvironmentObject var jsi: justSignedIn @FetchRequest(entity: Server.entity(), sortDescriptors: [NSSortDescriptor(keyPath: \Server.name, ascending: true)]) private var servers: FetchedResults @FetchRequest(entity: SignedInUser.entity(), sortDescriptors: [NSSortDescriptor(keyPath: \SignedInUser.username, ascending: true)]) private var savedUsers: FetchedResults @State private var needsToSelectServer = false; @State private var isSignInErrored = false; @State private var isNetworkErrored = false; @State private var isLoading = false; @State private var tabSelection: String = "Home"; @State private var libraries: [String] = []; @State private var library_names: [String: String] = [:]; @State private var librariesShowRecentlyAdded: [String] = []; @State private var libraryPrefillID: String = ""; @State private var showSettingsPopover: Bool = false; @Environment(\.verticalSizeClass) var verticalSizeClass: UserInterfaceSizeClass? @Environment(\.horizontalSizeClass) var horizontalSizeClass: UserInterfaceSizeClass? var isPortrait: Bool { let result = verticalSizeClass == .regular && horizontalSizeClass == .compact return result } func startup() { SentrySDK.start { options in options.dsn = "https://7ef695d745e942f8a52d69317c5ae241@o704459.ingest.sentry.io/5778161" options.debug = false // Enabled debug when first installing is always helpful options.releaseName = "ios-" + (Bundle.main.infoDictionary?["CFBundleVersion"] as! String); } let cache = SDImageCache(namespace: "tiny") cache.config.maxMemoryCost = 100 * 1024 * 1024 // 100MB memory cache.config.maxDiskSize = 1000 * 1024 * 1024 // 1000MB disk SDImageCachesManager.shared.addCache(cache) SDWebImageManager.defaultImageCache = SDImageCachesManager.shared _libraries.wrappedValue = [] _library_names.wrappedValue = [:] _librariesShowRecentlyAdded.wrappedValue = [] if(servers.isEmpty) { _isLoading.wrappedValue = false; _needsToSelectServer.wrappedValue = true; } else { _isLoading.wrappedValue = true; let savedUser = savedUsers[0]; let keychain = KeychainSwift(); if(keychain.get("AccessToken_\(savedUser.user_id ?? "")") != nil) { _globalData.wrappedValue.authToken = keychain.get("AccessToken_\(savedUser.user_id ?? "")") ?? "" _globalData.wrappedValue.server = servers[0] _globalData.wrappedValue.user = savedUser } let appVersion = Bundle.main.infoDictionary?["CFBundleShortVersionString"] as? String; globalData.authHeader = "MediaBrowser Client=\"SwiftFin\", Device=\"\(UIDevice.current.name)\", DeviceId=\"\(globalData.user?.device_uuid ?? "")\", Version=\"\(appVersion ?? "0.0.1")\", Token=\"\(globalData.authToken)\""; let request = RestRequest(method: .get, url: (globalData.server?.baseURI ?? "") + "/Users/Me") request.headerParameters["X-Emby-Authorization"] = globalData.authHeader request.contentType = "application/json" request.acceptType = "application/json" request.responseData() { (result: Result, RestError>) in switch result { case .success( let resp): do { let json = try JSON(data: resp.body) _libraries.wrappedValue = json["Configuration"]["OrderedViews"].arrayObject as? [String] ?? []; let array2 = json["Configuration"]["LatestItemsExcludes"].arrayObject as? [String] ?? [] _librariesShowRecentlyAdded.wrappedValue = _libraries.wrappedValue.filter { element in return !array2.contains(element) } let request2 = RestRequest(method: .get, url: (globalData.server?.baseURI ?? "") + "/Users/\(globalData.user?.user_id ?? "")/Views") request2.headerParameters["X-Emby-Authorization"] = globalData.authHeader request2.contentType = "application/json" request2.acceptType = "application/json" request2.responseData() { (result2: Result, RestError>) in switch result2 { case .success( let resp): do { let json2 = try JSON(data: resp.body) for (_,item2):(String, JSON) in json2["Items"] { _library_names.wrappedValue[item2["Id"].string ?? ""] = item2["Name"].string ?? "" } if(_libraries.wrappedValue.count == 0 && _librariesShowRecentlyAdded.wrappedValue.count == 0) { for (_,item2):(String, JSON) in json2["Items"] { _libraries.wrappedValue.append(item2["Id"].string ?? "") _librariesShowRecentlyAdded.wrappedValue.append(item2["Id"].string ?? "") } } } catch { } break case .failure( _): break } _isLoading.wrappedValue = false; } } catch { } break case .failure( let error): if(error.response?.status.code == 401) { _isLoading.wrappedValue = false; _isSignInErrored.wrappedValue = true; } else { _isLoading.wrappedValue = false; _isNetworkErrored.wrappedValue = true; } } } } } var body: some View { if(!jsi.did) { LoadingView(isShowing: $isLoading) { TabView(selection: $tabSelection) { NavigationView() { VStack { NavigationLink(destination: ConnectToServerView(isActive: $needsToSelectServer), isActive: $needsToSelectServer) { EmptyView() }.isDetailLink(false) NavigationLink(destination: ConnectToServerView(skip_server: true, skip_server_prefill: globalData.server, reauth_deviceId: globalData.user?.device_uuid ?? "", isActive: $isSignInErrored), isActive: $isSignInErrored) { EmptyView() }.isDetailLink(false) if(!needsToSelectServer && !isSignInErrored) { VStack(alignment: .leading) { ScrollView() { Spacer().frame(height: self.isPortrait ? 0 : 15) ContinueWatchingView() NextUpView().padding(EdgeInsets(top: 4, leading: 0, bottom: 0, trailing: 0)) ForEach(librariesShowRecentlyAdded, id: \.self) { library_id in VStack(alignment: .leading) { HStack() { Text("Latest \(library_names[library_id] ?? "")").font(.title2).fontWeight(.bold).padding(EdgeInsets(top: 0, leading: 0, bottom: 0, trailing: 16)) Spacer() NavigationLink(destination: LibraryView(prefill: library_id, names: library_names, libraries: libraries, filter: "&SortBy=DateCreated&SortOrder=Descending")) { Text("See All").font(.subheadline).fontWeight(.bold) } }.padding(EdgeInsets(top: 0, leading: 16, bottom: 0, trailing: 16)) LatestMediaView(library: library_id) }.padding(EdgeInsets(top: 4, leading: 0, bottom: 0, trailing: 0)) } Spacer().frame(height: 7) } } } } .navigationTitle("Home") .toolbar { ToolbarItemGroup(placement: .navigationBarTrailing) { Button { showSettingsPopover = true; } label: { Image(systemName: "gear") } } }.popover( isPresented: self.$showSettingsPopover, arrowEdge: .bottom) { SettingsView(close: $showSettingsPopover).environmentObject(self.globalData) } } .tabItem({ Text("Home") Image(systemName: "house") }) .tag("Home") NavigationView() { LibraryView(prefill: "", names: library_names, libraries: libraries) .navigationTitle("Library") } .tabItem({ Text("All Media") Image(systemName: "folder") }) .tag("All Media") }.edgesIgnoringSafeArea(isPortrait ? [] : [.leading,.trailing]) }.environmentObject(globalData) .edgesIgnoringSafeArea(isPortrait ? [] : [.leading,.trailing]) .onAppear(perform: startup) .navigationViewStyle(StackNavigationViewStyle()) .alert(isPresented: $isNetworkErrored) { Alert(title: Text("Network Error"), message: Text("Couldn't connect to Jellyfin"), dismissButton: .default(Text("Ok"))) }.introspectTabBarController { (UITabBarController) in UITabBarController.tabBar.isHidden = false } } else { Text("Signing in...") .onAppear(perform: { DispatchQueue.main.async { [self] in usleep(500000); self.jsi.did = false; } }) } } } struct ContentView_Previews: PreviewProvider { static var previews: some View { ContentView().environment(\.managedObjectContext, PersistenceController.preview.container.viewContext) } }