jellyflood/Swiftfin/Views/ItemView/Components/AttributeHStack.swift
Joe Kribs 890bf1fa31
[iOS & tvOS] Video Range Types (#1449)
* `VideoRangeType` "Extension" for now it's an enum until 10.10. Otherwise, done.

* Limit lines to 1 and variable width as needed.

* CodeFactor issue resolution

* AttributeViewModifier -> AttributeBadge

* change API

* `WrappingHStack`

---------

Co-authored-by: Ethan Pippin <ethanpippin2343@gmail.com>
2025-03-14 13:01:23 -04:00

163 lines
6.3 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) 2025 Jellyfin & Jellyfin Contributors
//
import SwiftUI
import WrappingHStack
extension ItemView {
struct AttributesHStack: View {
@ObservedObject
var viewModel: ItemViewModel
@StoredValue(.User.itemViewAttributes)
private var itemViewAttributes
// MARK: - Body
var body: some View {
let badges = computeBadges()
if !badges.isEmpty {
WrappingHStack(badges, id: \.self, alignment: .center, spacing: .constant(8), lineSpacing: 8) { badgeItem in
badgeItem
.fixedSize(horizontal: true, vertical: false)
}
.foregroundStyle(Color(UIColor.darkGray))
.lineLimit(1)
.frame(maxWidth: 300)
}
}
// MARK: - Compute Badges
private func computeBadges() -> [AnyView] {
var badges: [AnyView] = []
var processedGroups = Set<ItemViewAttribute>()
for attribute in itemViewAttributes {
if processedGroups.contains(attribute) { continue }
processedGroups.insert(attribute)
switch attribute {
case .ratingCritics:
if let criticRating = viewModel.item.criticRating {
let badge = AnyView(
AttributeBadge(
style: .outline,
title: Text("\(criticRating, specifier: "%.0f")")
) {
if criticRating >= 60 {
Image(.tomatoFresh)
.symbolRenderingMode(.hierarchical)
} else {
Image(.tomatoRotten)
}
}
)
badges.append(badge)
}
case .ratingCommunity:
if let communityRating = viewModel.item.communityRating {
let badge = AnyView(
AttributeBadge(
style: .outline,
title: Text("\(communityRating, specifier: "%.01f")"),
systemName: "star.fill"
)
)
badges.append(badge)
}
case .ratingOfficial:
if let officialRating = viewModel.item.officialRating {
let badge = AnyView(
AttributeBadge(
style: .outline,
title: officialRating
)
)
badges.append(badge)
}
case .videoQuality:
if let mediaStreams = viewModel.selectedMediaSource?.mediaStreams {
// Resolution badge (if available). Only one of 4K or HD is shown.
if mediaStreams.has4KVideo {
let badge = AnyView(
AttributeBadge(
style: .fill,
title: "4K"
)
)
badges.append(badge)
} else if mediaStreams.hasHDVideo {
let badge = AnyView(
AttributeBadge(
style: .fill,
title: "HD"
)
)
badges.append(badge)
}
if mediaStreams.hasDolbyVision {
let badge = AnyView(
AttributeBadge(
style: .fill,
title: "DV"
)
)
badges.append(badge)
}
if mediaStreams.hasHDRVideo {
let badge = AnyView(
AttributeBadge(
style: .fill,
title: "HDR"
)
)
badges.append(badge)
}
}
case .audioChannels:
if let mediaStreams = viewModel.selectedMediaSource?.mediaStreams {
if mediaStreams.has51AudioChannelLayout {
let badge = AnyView(
AttributeBadge(
style: .fill,
title: "5.1"
)
)
badges.append(badge)
}
if mediaStreams.has71AudioChannelLayout {
let badge = AnyView(
AttributeBadge(
style: .fill,
title: "7.1"
)
)
badges.append(badge)
}
}
case .subtitles:
if let mediaStreams = viewModel.selectedMediaSource?.mediaStreams,
mediaStreams.hasSubtitles
{
let badge = AnyView(
AttributeBadge(
style: .outline,
title: "CC"
)
)
badges.append(badge)
}
}
}
return badges
}
}
}