jellyflood/Swiftfin/Views/UserListView.swift

89 lines
2.5 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 CollectionVGrid
import SwiftUI
struct UserListView: View {
@EnvironmentObject
private var router: UserListCoordinator.Router
@StateObject
private var viewModel: UserListViewModel
init(server: ServerState) {
self._viewModel = StateObject(wrappedValue: UserListViewModel(server: server))
}
private var noUserView: some View {
VStack {
L10n.signInGetStarted.text
.frame(minWidth: 50, maxWidth: 240)
.multilineTextAlignment(.center)
PrimaryButton(title: L10n.signIn)
.onSelect {
router.route(to: \.userSignIn, viewModel.server)
}
.frame(maxWidth: 300)
.frame(height: 50)
}
}
@ViewBuilder
private var gridView: some View {
CollectionVGrid(
viewModel.users,
layout: .minWidth(120, itemSpacing: 30, lineSpacing: 30)
) { user in
UserProfileButton(user: user, client: viewModel.client)
.onSelect {
viewModel.signIn(user: user)
}
.contextMenu(menuItems: {
Button(L10n.remove, systemImage: "trash", role: .destructive) {
viewModel.remove(user: user)
}
})
}
}
var body: some View {
Group {
if viewModel.users.isEmpty {
noUserView
.offset(y: -50)
} else {
gridView
}
}
.navigationTitle(viewModel.server.name)
.toolbar {
ToolbarItemGroup(placement: .topBarTrailing) {
if viewModel.users.isNotEmpty {
Button {
router.route(to: \.userSignIn, viewModel.server)
} label: {
Image(systemName: "person.crop.circle.fill.badge.plus")
}
}
Button {
router.route(to: \.serverDetail, viewModel.server)
} label: {
Image(systemName: "info.circle.fill")
}
}
}
.onAppear {
viewModel.fetchUsers()
}
}
}