110 lines
3.1 KiB
Swift
110 lines
3.1 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 CollectionView
|
|
import Factory
|
|
import JellyfinAPI
|
|
import SwiftUI
|
|
|
|
struct UserListView: View {
|
|
|
|
@EnvironmentObject
|
|
private var router: UserListCoordinator.Router
|
|
|
|
@ObservedObject
|
|
var viewModel: UserListViewModel
|
|
|
|
@State
|
|
private var longPressedUser: SwiftfinStore.State.User?
|
|
|
|
@ViewBuilder
|
|
private var listView: some View {
|
|
CollectionView(items: viewModel.users) { _, user, _ in
|
|
UserProfileButton(user: user)
|
|
.onSelect {
|
|
viewModel.signIn(user: user)
|
|
}
|
|
.onLongPressGesture {
|
|
longPressedUser = user
|
|
}
|
|
}
|
|
.layout { _, layoutEnvironment in
|
|
.grid(
|
|
layoutEnvironment: layoutEnvironment,
|
|
layoutMode: .adaptive(withMinItemSize: 250),
|
|
itemSpacing: 20,
|
|
lineSpacing: 20,
|
|
sectionInsets: .init(top: 20, leading: 20, bottom: 20, trailing: 20)
|
|
)
|
|
}
|
|
.padding(50)
|
|
}
|
|
|
|
@ViewBuilder
|
|
private var noUserView: some View {
|
|
VStack(spacing: 50) {
|
|
L10n.signInGetStarted.text
|
|
.frame(maxWidth: 500)
|
|
.multilineTextAlignment(.center)
|
|
.font(.body)
|
|
|
|
Button {
|
|
router.route(to: \.userSignIn, viewModel.server)
|
|
} label: {
|
|
L10n.signIn.text
|
|
.bold()
|
|
.font(.callout)
|
|
.frame(width: 400, height: 75)
|
|
.background(Color.jellyfinPurple)
|
|
}
|
|
.buttonStyle(.card)
|
|
}
|
|
}
|
|
|
|
var body: some View {
|
|
ZStack {
|
|
|
|
ImageView(viewModel.userSession.client.fullURL(with: Paths.getSplashscreen()))
|
|
.ignoresSafeArea()
|
|
|
|
Color.black
|
|
.opacity(0.9)
|
|
.ignoresSafeArea()
|
|
|
|
if viewModel.users.isEmpty {
|
|
noUserView
|
|
.offset(y: -50)
|
|
} else {
|
|
listView
|
|
}
|
|
}
|
|
.navigationTitle(viewModel.server.name)
|
|
.if(!viewModel.users.isEmpty) { view in
|
|
view.toolbar {
|
|
ToolbarItem(placement: .navigationBarTrailing) {
|
|
Button {
|
|
router.route(to: \.userSignIn, viewModel.server)
|
|
} label: {
|
|
Image(systemName: "person.crop.circle.fill.badge.plus")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
.alert(item: $longPressedUser) { user in
|
|
Alert(
|
|
title: Text(user.username),
|
|
primaryButton: .destructive(L10n.remove.text, action: { viewModel.remove(user: user) }),
|
|
secondaryButton: .cancel()
|
|
)
|
|
}
|
|
.onAppear {
|
|
viewModel.fetchUsers()
|
|
}
|
|
}
|
|
}
|