152 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Swift
		
	
	
	
	
	
			
		
		
	
	
			152 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Swift
		
	
	
	
	
	
| //
 | |
| // 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 (c) 2022 Jellyfin & Jellyfin Contributors
 | |
| //
 | |
| 
 | |
| import JellyfinAPI
 | |
| import SwiftUI
 | |
| 
 | |
| struct LiveTVChannelItemWideElement: View {
 | |
| 	@FocusState
 | |
| 	private var focused: Bool
 | |
| 	@State
 | |
| 	private var loading: Bool = false
 | |
| 	@State
 | |
| 	private var isFocused: Bool = false
 | |
| 
 | |
| 	var channel: BaseItemDto
 | |
| 	var currentProgram: BaseItemDto?
 | |
| 	var currentProgramText: LiveTVChannelViewProgram
 | |
| 	var nextProgramsText: [LiveTVChannelViewProgram]
 | |
| 	var onSelect: (@escaping (Bool) -> Void) -> Void
 | |
| 
 | |
| 	var progressPercent: Double {
 | |
| 		if let currentProgram = currentProgram {
 | |
| 			let progressPercent = currentProgram.getLiveProgressPercentage()
 | |
| 			if progressPercent > 1.0 {
 | |
| 				return 1.0
 | |
| 			} else {
 | |
| 				return progressPercent
 | |
| 			}
 | |
| 		}
 | |
| 		return 0
 | |
| 	}
 | |
| 
 | |
| 	private var detailText: String {
 | |
| 		guard let program = currentProgram else {
 | |
| 			return ""
 | |
| 		}
 | |
| 		var text = ""
 | |
| 		if let season = program.parentIndexNumber,
 | |
| 		   let episode = program.indexNumber
 | |
| 		{
 | |
| 			text.append("\(season)x\(episode) ")
 | |
| 		} else if let episode = program.indexNumber {
 | |
| 			text.append("\(episode) ")
 | |
| 		}
 | |
| 		if let title = program.episodeTitle {
 | |
| 			text.append("\(title) ")
 | |
| 		}
 | |
| 		if let year = program.productionYear {
 | |
| 			text.append("\(year) ")
 | |
| 		}
 | |
| 		if let rating = program.officialRating {
 | |
| 			text.append("\(rating)")
 | |
| 		}
 | |
| 		return text
 | |
| 	}
 | |
| 
 | |
| 	var body: some View {
 | |
| 		ZStack {
 | |
| 			ZStack {
 | |
| 				HStack {
 | |
| 					ZStack(alignment: .center) {
 | |
| 						ImageView(channel.getPrimaryImage(maxWidth: 128))
 | |
| 							.aspectRatio(contentMode: .fit)
 | |
| 							.padding(.init(top: 0, leading: 0, bottom: 8, trailing: 0))
 | |
| 						VStack(alignment: .center) {
 | |
| 							Spacer()
 | |
| 								.frame(maxHeight: .infinity)
 | |
| 							GeometryReader { gp in
 | |
| 								ZStack(alignment: .leading) {
 | |
| 									RoundedRectangle(cornerRadius: 3)
 | |
| 										.fill(Color.gray)
 | |
| 										.opacity(0.4)
 | |
| 										.frame(minWidth: 0, maxWidth: .infinity, minHeight: 6, maxHeight: 6)
 | |
| 									RoundedRectangle(cornerRadius: 6)
 | |
| 										.fill(Color.jellyfinPurple)
 | |
| 										.frame(width: CGFloat(progressPercent * gp.size.width), height: 6)
 | |
| 								}
 | |
| 							}
 | |
| 							.frame(height: 6, alignment: .center)
 | |
| 							.padding(.init(top: 0, leading: 4, bottom: 0, trailing: 4))
 | |
| 						}
 | |
| 						if loading {
 | |
| 
 | |
| 							ProgressView()
 | |
| 						}
 | |
| 					}
 | |
| 					.aspectRatio(1.0, contentMode: .fit)
 | |
| 					VStack(alignment: .leading) {
 | |
| 						let channelNumber = channel.number != nil ? "\(channel.number ?? "") " : ""
 | |
| 						let channelName = "\(channelNumber)\(channel.name ?? "?")"
 | |
| 						Text(channelName)
 | |
| 							.font(.body)
 | |
| 							.lineLimit(1)
 | |
| 							.foregroundColor(Color.jellyfinPurple)
 | |
| 							.frame(alignment: .leading)
 | |
| 							.padding(.init(top: 0, leading: 0, bottom: 4, trailing: 0))
 | |
| 						programLabel(timeText: currentProgramText.timeDisplay, titleText: currentProgramText.title,
 | |
| 						             color: Color("TextHighlightColor"))
 | |
| 						if !nextProgramsText.isEmpty,
 | |
| 						   let nextItem = nextProgramsText[0]
 | |
| 						{
 | |
| 							programLabel(timeText: nextItem.timeDisplay, titleText: nextItem.title, color: Color.gray)
 | |
| 						}
 | |
| 						if nextProgramsText.count > 1,
 | |
| 						   let nextItem2 = nextProgramsText[1]
 | |
| 						{
 | |
| 							programLabel(timeText: nextItem2.timeDisplay, titleText: nextItem2.title, color: Color.gray)
 | |
| 						}
 | |
| 						Spacer()
 | |
| 					}
 | |
| 					Spacer()
 | |
| 				}
 | |
| 				.frame(alignment: .leading)
 | |
| 				.padding()
 | |
| 				.opacity(loading ? 0.5 : 1.0)
 | |
| 			}
 | |
| 			.background(RoundedRectangle(cornerRadius: 10, style: .continuous).fill(Color("BackgroundSecondaryColor")))
 | |
| 			.frame(height: 128)
 | |
| 			.onTapGesture {
 | |
| 				onSelect { loadingState in
 | |
| 					loading = loadingState
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		.background {
 | |
| 			RoundedRectangle(cornerRadius: 10, style: .continuous)
 | |
| 				.fill(Color("BackgroundColor"))
 | |
| 				.shadow(color: Color("ShadowColor"), radius: 4, x: 0, y: 0)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	@ViewBuilder
 | |
| 	func programLabel(timeText: String, titleText: String, color: Color) -> some View {
 | |
| 		HStack(alignment: .top) {
 | |
| 			Text(timeText)
 | |
| 				.font(.footnote)
 | |
| 				.lineLimit(2)
 | |
| 				.foregroundColor(color)
 | |
| 				.frame(width: 38, alignment: .leading)
 | |
| 			Text(titleText)
 | |
| 				.font(.footnote)
 | |
| 				.lineLimit(2)
 | |
| 				.foregroundColor(color)
 | |
| 		}
 | |
| 	}
 | |
| }
 |