jellyflood/Swiftfin/Views/ResetUserPasswordView/ResetUserPasswordView.swift

184 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) 2024 Jellyfin & Jellyfin Contributors
//
import Combine
import Defaults
import JellyfinAPI
import SwiftUI
struct ResetUserPasswordView: View {
private enum Field: Hashable {
case currentPassword
case newPassword
case confirmNewPassword
}
@Default(.accentColor)
private var accentColor
@EnvironmentObject
private var router: BasicNavigationViewCoordinator.Router
@FocusState
private var focusedField: Field?
@StateObject
private var viewModel: ResetUserPasswordViewModel
// MARK: - Password Variables
@State
private var currentPassword: String = ""
@State
private var newPassword: String = ""
@State
private var confirmNewPassword: String = ""
// MARK: - State Variables
@State
private var error: Error? = nil
@State
private var isPresentingError: Bool = false
@State
private var isPresentingSuccess: Bool = false
private let requiresCurrentPassword: Bool
// MARK: - Initializer
init(userID: String, requiresCurrentPassword: Bool) {
self._viewModel = StateObject(wrappedValue: ResetUserPasswordViewModel(userID: userID))
self.requiresCurrentPassword = requiresCurrentPassword
}
// MARK: - Body
var body: some View {
List {
if requiresCurrentPassword {
Section(L10n.currentPassword) {
UnmaskSecureField(L10n.currentPassword, text: $currentPassword) {
focusedField = .newPassword
}
.autocorrectionDisabled()
.textInputAutocapitalization(.none)
.focused($focusedField, equals: .currentPassword)
.disabled(viewModel.state == .resetting)
}
}
Section(L10n.newPassword) {
UnmaskSecureField(L10n.newPassword, text: $newPassword) {
focusedField = .confirmNewPassword
}
.autocorrectionDisabled()
.textInputAutocapitalization(.none)
.focused($focusedField, equals: .newPassword)
.disabled(viewModel.state == .resetting)
}
Section {
UnmaskSecureField(L10n.confirmNewPassword, text: $confirmNewPassword) {
viewModel.send(.reset(current: currentPassword, new: confirmNewPassword))
}
.autocorrectionDisabled()
.textInputAutocapitalization(.none)
.focused($focusedField, equals: .confirmNewPassword)
.disabled(viewModel.state == .resetting)
} header: {
Text(L10n.confirmNewPassword)
} footer: {
if newPassword != confirmNewPassword {
Label(L10n.passwordsDoNotMatch, systemImage: "exclamationmark.circle.fill")
.labelStyle(.sectionFooterWithImage(imageStyle: .orange))
}
}
Section {
if viewModel.state == .resetting {
ListRowButton(L10n.cancel) {
viewModel.send(.cancel)
if requiresCurrentPassword {
focusedField = .currentPassword
} else {
focusedField = .newPassword
}
}
.foregroundStyle(.red, .red.opacity(0.2))
} else {
ListRowButton(L10n.save) {
focusedField = nil
viewModel.send(.reset(current: currentPassword, new: confirmNewPassword))
}
.disabled(newPassword != confirmNewPassword || viewModel.state == .resetting)
.foregroundStyle(accentColor.overlayColor, accentColor)
.opacity(newPassword != confirmNewPassword ? 0.5 : 1)
}
} footer: {
Text(L10n.passwordChangeWarning)
}
}
.interactiveDismissDisabled(viewModel.state == .resetting)
.navigationBarBackButtonHidden(viewModel.state == .resetting)
.navigationTitle(L10n.password)
.navigationBarTitleDisplayMode(.inline)
.navigationBarCloseButton {
router.dismissCoordinator()
}
.onFirstAppear {
if requiresCurrentPassword {
focusedField = .currentPassword
} else {
focusedField = .newPassword
}
}
.onReceive(viewModel.events) { event in
switch event {
case let .error(eventError):
UIDevice.feedback(.error)
error = eventError
isPresentingError = true
case .success:
UIDevice.feedback(.success)
isPresentingSuccess = true
}
}
.topBarTrailing {
if viewModel.state == .resetting {
ProgressView()
}
}
.alert(
L10n.error,
isPresented: $isPresentingError,
presenting: error
) { _ in
Button(L10n.dismiss, role: .cancel) {
focusedField = .newPassword
}
} message: { error in
Text(error.localizedDescription)
}
.alert(
L10n.success,
isPresented: $isPresentingSuccess
) {
Button(L10n.dismiss, role: .cancel) {
router.dismissCoordinator()
}
} message: {
Text(L10n.passwordChangedMessage)
}
}
}