195 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Swift
		
	
	
	
			
		
		
	
	
			195 lines
		
	
	
		
			5.8 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 SwiftUI
 | |
| 
 | |
| struct PosterButton<Item: Poster, Content: View, ImageOverlay: View, ContextMenu: View>: View {
 | |
| 
 | |
|     @FocusState
 | |
|     private var isFocused: Bool
 | |
| 
 | |
|     private var item: Item
 | |
|     private var type: PosterType
 | |
|     private var itemScale: CGFloat
 | |
|     private var horizontalAlignment: HorizontalAlignment
 | |
|     private var content: () -> Content
 | |
|     private var imageOverlay: () -> ImageOverlay
 | |
|     private var contextMenu: () -> ContextMenu
 | |
|     private var onSelect: () -> Void
 | |
|     private var onFocus: () -> Void
 | |
|     private var singleImage: Bool
 | |
| 
 | |
|     private var itemWidth: CGFloat {
 | |
|         type.width * itemScale
 | |
|     }
 | |
| 
 | |
|     var body: some View {
 | |
|         VStack(alignment: horizontalAlignment) {
 | |
|             Button {
 | |
|                 onSelect()
 | |
|             } label: {
 | |
|                 Group {
 | |
|                     switch type {
 | |
|                     case .portrait:
 | |
|                         ImageView(item.portraitPosterImageSource(maxWidth: itemWidth))
 | |
|                             .failure {
 | |
|                                 InitialFailureView(item.displayName.initials)
 | |
|                             }
 | |
|                             .posterStyle(type: type, width: itemWidth)
 | |
|                     case .landscape:
 | |
|                         ImageView(item.landscapePosterImageSources(maxWidth: itemWidth, single: singleImage))
 | |
|                             .failure {
 | |
|                                 InitialFailureView(item.displayName.initials)
 | |
|                             }
 | |
|                             .posterStyle(type: type, width: itemWidth)
 | |
|                     }
 | |
|                 }
 | |
|                 .overlay {
 | |
|                     imageOverlay()
 | |
|                         .posterStyle(type: type, width: itemWidth)
 | |
|                 }
 | |
|             }
 | |
|             .buttonStyle(.card)
 | |
|             .contextMenu(menuItems: {
 | |
|                 contextMenu()
 | |
|             })
 | |
|             .posterShadow()
 | |
|             .focused($isFocused)
 | |
| 
 | |
|             content()
 | |
|                 .zIndex(-1)
 | |
|         }
 | |
|         .frame(width: itemWidth)
 | |
|         .onChange(of: isFocused) { newValue in
 | |
|             guard newValue else { return }
 | |
|             onFocus()
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| extension PosterButton where Content == PosterButtonDefaultContentView<Item>,
 | |
|     ImageOverlay == EmptyView,
 | |
|     ContextMenu == EmptyView
 | |
| {
 | |
|     init(item: Item, type: PosterType, singleImage: Bool = false) {
 | |
|         self.init(
 | |
|             item: item,
 | |
|             type: type,
 | |
|             itemScale: 1,
 | |
|             horizontalAlignment: .leading,
 | |
|             content: { PosterButtonDefaultContentView(item: item) },
 | |
|             imageOverlay: { EmptyView() },
 | |
|             contextMenu: { EmptyView() },
 | |
|             onSelect: {},
 | |
|             onFocus: {},
 | |
|             singleImage: singleImage
 | |
|         )
 | |
|     }
 | |
| }
 | |
| 
 | |
| extension PosterButton {
 | |
|     func horizontalAlignment(_ alignment: HorizontalAlignment) -> Self {
 | |
|         var copy = self
 | |
|         copy.horizontalAlignment = alignment
 | |
|         return copy
 | |
|     }
 | |
| 
 | |
|     func scaleItem(_ scale: CGFloat) -> Self {
 | |
|         var copy = self
 | |
|         copy.itemScale = scale
 | |
|         return copy
 | |
|     }
 | |
| 
 | |
|     @ViewBuilder
 | |
|     func content<C: View>(@ViewBuilder _ content: @escaping () -> C) -> PosterButton<Item, C, ImageOverlay, ContextMenu> {
 | |
|         PosterButton<Item, C, ImageOverlay, ContextMenu>(
 | |
|             item: item,
 | |
|             type: type,
 | |
|             itemScale: itemScale,
 | |
|             horizontalAlignment: horizontalAlignment,
 | |
|             content: content,
 | |
|             imageOverlay: imageOverlay,
 | |
|             contextMenu: contextMenu,
 | |
|             onSelect: onSelect,
 | |
|             onFocus: onFocus,
 | |
|             singleImage: singleImage
 | |
|         )
 | |
|     }
 | |
| 
 | |
|     @ViewBuilder
 | |
|     func imageOverlay<O: View>(@ViewBuilder _ imageOverlay: @escaping () -> O) -> PosterButton<Item, Content, O, ContextMenu> {
 | |
|         PosterButton<Item, Content, O, ContextMenu>(
 | |
|             item: item,
 | |
|             type: type,
 | |
|             itemScale: itemScale,
 | |
|             horizontalAlignment: horizontalAlignment,
 | |
|             content: content,
 | |
|             imageOverlay: imageOverlay,
 | |
|             contextMenu: contextMenu,
 | |
|             onSelect: onSelect,
 | |
|             onFocus: onFocus,
 | |
|             singleImage: singleImage
 | |
|         )
 | |
|     }
 | |
| 
 | |
|     @ViewBuilder
 | |
|     func contextMenu<M: View>(@ViewBuilder _ contextMenu: @escaping () -> M) -> PosterButton<Item, Content, ImageOverlay, M> {
 | |
|         PosterButton<Item, Content, ImageOverlay, M>(
 | |
|             item: item,
 | |
|             type: type,
 | |
|             itemScale: itemScale,
 | |
|             horizontalAlignment: horizontalAlignment,
 | |
|             content: content,
 | |
|             imageOverlay: imageOverlay,
 | |
|             contextMenu: contextMenu,
 | |
|             onSelect: onSelect,
 | |
|             onFocus: onFocus,
 | |
|             singleImage: singleImage
 | |
|         )
 | |
|     }
 | |
| 
 | |
|     func onSelect(_ action: @escaping () -> Void) -> Self {
 | |
|         var copy = self
 | |
|         copy.onSelect = action
 | |
|         return copy
 | |
|     }
 | |
| 
 | |
|     func onFocus(_ action: @escaping () -> Void) -> Self {
 | |
|         var copy = self
 | |
|         copy.onFocus = action
 | |
|         return copy
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: default content view
 | |
| 
 | |
| struct PosterButtonDefaultContentView<Item: Poster>: View {
 | |
| 
 | |
|     let item: Item
 | |
| 
 | |
|     var body: some View {
 | |
|         VStack(alignment: .leading) {
 | |
|             if item.showTitle {
 | |
|                 Text(item.displayName)
 | |
|                     .font(.footnote)
 | |
|                     .fontWeight(.regular)
 | |
|                     .foregroundColor(.primary)
 | |
|                     .lineLimit(2)
 | |
|             }
 | |
| 
 | |
|             if let description = item.subtitle {
 | |
|                 Text(description)
 | |
|                     .font(.caption)
 | |
|                     .fontWeight(.medium)
 | |
|                     .foregroundColor(.secondary)
 | |
|                     .lineLimit(2)
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |