148 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			Swift
		
	
	
	
			
		
		
	
	
			148 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			Swift
		
	
	
	
| /*
 | |
|  * JellyfinPlayer/Swiftfin is subject to the terms of the Mozilla Public
 | |
|  * License, v2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, you can obtain one at https://mozilla.org/MPL/2.0/.
 | |
|  *
 | |
|  * Copyright 2021 Aiden Vigue & Jellyfin Contributors
 | |
|  */
 | |
| 
 | |
| import SwiftUI
 | |
| 
 | |
| struct LibraryView: View {
 | |
|     @StateObject var viewModel: LibraryViewModel
 | |
|     var title: String
 | |
| 
 | |
|     // MARK: tracks for grid
 | |
|     var defaultFilters = LibraryFilters(filters: [], sortOrder: [.ascending], withGenres: [], tags: [], sortBy: [.name])
 | |
| 
 | |
|     @State var isShowingSearchView = false
 | |
|     @State var isShowingFilterView = false
 | |
| 
 | |
|     @State private var tracks: [GridItem] = Array(repeating: .init(.flexible()), count: Int(UIScreen.main.bounds.size.width) / 125)
 | |
| 
 | |
|     func recalcTracks() {
 | |
|         tracks = Array(repeating: .init(.flexible()), count: Int(UIScreen.main.bounds.size.width) / 125)
 | |
|     }
 | |
| 
 | |
|     var body: some View {
 | |
|         Group {
 | |
|             if viewModel.isLoading == true {
 | |
|                 ProgressView()
 | |
|             } else if !viewModel.items.isEmpty {
 | |
|                 VStack {
 | |
|                     ScrollView(.vertical) {
 | |
|                         Spacer().frame(height: 16)
 | |
|                         LazyVGrid(columns: tracks) {
 | |
|                             ForEach(viewModel.items, id: \.id) { item in
 | |
|                                 NavigationLink(destination: ItemView(item: item)) {
 | |
|                                     VStack(alignment: .leading) {
 | |
|                                         ImageView(src: item.getPrimaryImage(maxWidth: 100), bh: item.getPrimaryImageBlurHash())
 | |
|                                             .frame(width: 100, height: 150)
 | |
|                                             .cornerRadius(10)
 | |
|                                             .overlay(
 | |
|                                                 ZStack {
 | |
|                                                     if(item.userData!.played ?? false) {
 | |
|                                                         Image(systemName: "circle.fill")
 | |
|                                                             .foregroundColor(.white)
 | |
|                                                         Image(systemName: "checkmark.circle.fill")
 | |
|                                                             .foregroundColor(Color(.systemBlue))
 | |
|                                                     }
 | |
|                                                 }.padding(2)
 | |
|                                                 .opacity(1)
 | |
|                                                 , alignment: .topTrailing).opacity(1)
 | |
|                                         Text(item.name ?? "")
 | |
|                                             .font(.caption)
 | |
|                                             .fontWeight(.semibold)
 | |
|                                             .foregroundColor(.primary)
 | |
|                                             .lineLimit(1)
 | |
|                                         if item.productionYear != nil {
 | |
|                                             Text(String(item.productionYear!))
 | |
|                                                 .foregroundColor(.secondary)
 | |
|                                                 .font(.caption)
 | |
|                                                 .fontWeight(.medium)
 | |
|                                         } else {
 | |
|                                             Text(item.type ?? "")
 | |
|                                                 .foregroundColor(.secondary)
 | |
|                                                 .font(.caption)
 | |
|                                                 .fontWeight(.medium)
 | |
|                                         }
 | |
|                                     }.frame(width: 100)
 | |
|                                 }
 | |
|                             }
 | |
|                         }.onRotate { _ in
 | |
|                             recalcTracks()
 | |
|                         }
 | |
|                         if viewModel.hasNextPage || viewModel.hasPreviousPage {
 | |
|                             HStack {
 | |
|                                 Spacer()
 | |
|                                 HStack {
 | |
|                                     Button {
 | |
|                                         viewModel.requestPreviousPage()
 | |
|                                     } label: {
 | |
|                                         Image(systemName: "chevron.left")
 | |
|                                             .font(.system(size: 25))
 | |
|                                     }.disabled(!viewModel.hasPreviousPage)
 | |
|                                     Text("Page \(String(viewModel.currentPage + 1)) of \(String(viewModel.totalPages))")
 | |
|                                         .font(.headline)
 | |
|                                         .fontWeight(.semibold)
 | |
|                                     Button {
 | |
|                                         viewModel.requestNextPage()
 | |
|                                     } label: {
 | |
|                                         Image(systemName: "chevron.right")
 | |
|                                             .font(.system(size: 25))
 | |
|                                     }.disabled(!viewModel.hasNextPage)
 | |
|                                 }
 | |
|                                 Spacer()
 | |
|                             }
 | |
|                         }
 | |
|                         Spacer().frame(height: 16)
 | |
|                     }
 | |
|                 }
 | |
|             } else {
 | |
|                 Text("No results.")
 | |
|             }
 | |
|         }
 | |
|         .navigationBarTitle(title, displayMode: .inline)
 | |
|         .toolbar {
 | |
|             ToolbarItemGroup(placement: .navigationBarTrailing) {
 | |
|                 if viewModel.hasPreviousPage {
 | |
|                     Button {
 | |
|                         viewModel.requestPreviousPage()
 | |
|                     } label: {
 | |
|                         Image(systemName: "chevron.left")
 | |
|                     }
 | |
|                 }
 | |
|                 if viewModel.hasNextPage {
 | |
|                     Button {
 | |
|                         viewModel.requestNextPage()
 | |
|                     } label: {
 | |
|                         Image(systemName: "chevron.right")
 | |
|                     }
 | |
|                 }
 | |
|                 Label("Icon One", systemImage: "line.horizontal.3.decrease.circle")
 | |
|                 .foregroundColor(viewModel.filters == defaultFilters ? .accentColor : Color(UIColor.systemOrange))
 | |
|                 .onTapGesture {
 | |
|                     isShowingFilterView = true
 | |
|                 }
 | |
|                 Button {
 | |
|                     isShowingSearchView = true
 | |
|                 } label: {
 | |
|                     Image(systemName: "magnifyingglass")
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         .sheet(isPresented: $isShowingFilterView) {
 | |
|             LibraryFilterView(filters: $viewModel.filters, enabledFilterType: viewModel.enabledFilterType)
 | |
|         }
 | |
|         .background(
 | |
|             NavigationLink(destination: LibrarySearchView(viewModel: .init(parentID: viewModel.parentID)),
 | |
|                            isActive: $isShowingSearchView) {
 | |
|                 EmptyView()
 | |
|             }
 | |
|         )
 | |
|     }
 | |
| }
 | |
| 
 | |
| // stream BM^S by nicki!
 | |
| //
 |