jellyflood/Swiftfin/Views/SelectUserView/Components/UserRow.swift

159 lines
4.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) 2024 Jellyfin & Jellyfin Contributors
//
import Defaults
import JellyfinAPI
import SwiftUI
extension SelectUserView {
struct UserRow: View {
@Default(.accentColor)
private var accentColor
@Environment(\.colorScheme)
private var colorScheme
@Environment(\.isEditing)
private var isEditing
@Environment(\.isSelected)
private var isSelected
private let user: UserState
private let server: ServerState
private let showServer: Bool
private let action: () -> Void
private let onDelete: () -> Void
init(
user: UserState,
server: ServerState,
showServer: Bool,
action: @escaping () -> Void,
onDelete: @escaping () -> Void
) {
self.user = user
self.server = server
self.showServer = showServer
self.action = action
self.onDelete = onDelete
}
private var labelForegroundStyle: some ShapeStyle {
guard isEditing else { return .primary }
return isSelected ? .primary : .secondary
}
@ViewBuilder
private var personView: some View {
ZStack {
Group {
if colorScheme == .light {
Color.secondarySystemFill
} else {
Color.tertiarySystemBackground
}
}
.posterShadow()
RelativeSystemImageView(systemName: "person.fill", ratio: 0.5)
.foregroundStyle(.secondary)
}
.clipShape(.circle)
.aspectRatio(1, contentMode: .fill)
}
@ViewBuilder
private var userImage: some View {
ZStack {
Color.clear
ImageView(user.profileImageSource(client: server.client, maxWidth: 120))
.pipeline(.Swiftfin.branding)
.image { image in
image
.posterBorder(ratio: 1 / 2, of: \.width)
}
.placeholder { _ in
personView
}
.failure {
personView
}
if isEditing {
Color.black
.opacity(isSelected ? 0 : 0.5)
}
}
.aspectRatio(contentMode: .fill)
.clipShape(.circle)
}
@ViewBuilder
private var rowContent: some View {
HStack {
VStack(alignment: .leading, spacing: 5) {
Text(user.username)
.font(.title3)
.fontWeight(.semibold)
.foregroundStyle(labelForegroundStyle)
.lineLimit(2)
.multilineTextAlignment(.leading)
if showServer {
Text(server.name)
.font(.footnote)
.foregroundColor(Color(UIColor.lightGray))
}
}
Spacer()
if isEditing, isSelected {
Image(systemName: "checkmark.circle.fill")
.resizable()
.backport
.fontWeight(.bold)
.aspectRatio(1, contentMode: .fit)
.frame(width: 24, height: 24)
.symbolRenderingMode(.palette)
.foregroundStyle(accentColor.overlayColor, accentColor)
} else if isEditing {
Image(systemName: "circle")
.resizable()
.backport
.fontWeight(.bold)
.aspectRatio(1, contentMode: .fit)
.frame(width: 24, height: 24)
.foregroundStyle(.secondary)
}
}
}
var body: some View {
ListRow(insets: .init(horizontal: EdgeInsets.edgePadding)) {
userImage
.frame(width: 80)
.padding(.vertical, 8)
} content: {
rowContent
}
.onSelect(perform: action)
.contextMenu {
Button("Delete", role: .destructive) {
onDelete()
}
}
}
}
}