mirror of
https://github.com/yattee/yattee.git
synced 2025-01-06 01:20:31 +05:30
188 lines
7.4 KiB
Swift
188 lines
7.4 KiB
Swift
|
import SwiftUI
|
||
|
|
||
|
struct ImportSettingsAccountRow: View {
|
||
|
var account: Account
|
||
|
var fileModel: ImportSettingsFileModel
|
||
|
|
||
|
@State private var password = ""
|
||
|
|
||
|
@State private var isValid = false
|
||
|
@State private var isValidated = false
|
||
|
@State private var isValidating = false
|
||
|
@State private var validationError: String?
|
||
|
@State private var validationDebounce = Debounce()
|
||
|
|
||
|
@ObservedObject private var model = ImportSettingsSheetViewModel.shared
|
||
|
|
||
|
func afterValidation() {
|
||
|
if isValid {
|
||
|
model.importableAccounts.insert(account.id)
|
||
|
model.selectedAccounts.insert(account.id)
|
||
|
model.importableAccountsPasswords[account.id] = password
|
||
|
} else {
|
||
|
model.selectedAccounts.remove(account.id)
|
||
|
model.importableAccounts.remove(account.id)
|
||
|
model.importableAccountsPasswords.removeValue(forKey: account.id)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var body: some View {
|
||
|
Button(action: { model.toggleAccount(account, accounts: accounts) }) {
|
||
|
let accountExists = AccountsModel.shared.find(account.id) != nil
|
||
|
|
||
|
VStack(alignment: .leading) {
|
||
|
HStack {
|
||
|
Text(account.username)
|
||
|
Spacer()
|
||
|
Image(systemName: "checkmark")
|
||
|
.foregroundColor(.accentColor)
|
||
|
.opacity(isChecked ? 1 : 0)
|
||
|
}
|
||
|
Text(account.instance?.description ?? "")
|
||
|
.font(.caption)
|
||
|
.foregroundColor(.secondary)
|
||
|
|
||
|
Group {
|
||
|
if let instanceID = account.instanceID {
|
||
|
if accountExists {
|
||
|
HStack {
|
||
|
Image(systemName: "xmark.circle.fill")
|
||
|
.foregroundColor(Color("AppRedColor"))
|
||
|
Text("Account already exists")
|
||
|
}
|
||
|
} else {
|
||
|
Group {
|
||
|
if InstancesModel.shared.find(instanceID) != nil {
|
||
|
HStack {
|
||
|
Image(systemName: "checkmark.circle.fill")
|
||
|
.foregroundColor(.green)
|
||
|
Text("Custom Location already exists")
|
||
|
}
|
||
|
} else if model.selectedInstances.contains(instanceID) {
|
||
|
HStack {
|
||
|
Image(systemName: "checkmark.circle.fill")
|
||
|
.foregroundColor(.green)
|
||
|
Text("Custom Location selected for import")
|
||
|
}
|
||
|
} else {
|
||
|
HStack {
|
||
|
Image(systemName: "xmark.circle.fill")
|
||
|
.foregroundColor(.red)
|
||
|
Text("Custom Location not selected for import")
|
||
|
}
|
||
|
.foregroundColor(Color("AppRedColor"))
|
||
|
}
|
||
|
}
|
||
|
.frame(minHeight: 20)
|
||
|
|
||
|
if account.password.isNil || account.password!.isEmpty {
|
||
|
Group {
|
||
|
if password.isEmpty {
|
||
|
HStack {
|
||
|
Image(systemName: "key")
|
||
|
Text("Password required to import")
|
||
|
}
|
||
|
.foregroundColor(Color("AppRedColor"))
|
||
|
} else {
|
||
|
AccountValidationStatus(
|
||
|
app: .constant(instance.app),
|
||
|
isValid: $isValid,
|
||
|
isValidated: $isValidated,
|
||
|
isValidating: $isValidating,
|
||
|
error: $validationError
|
||
|
)
|
||
|
}
|
||
|
}
|
||
|
.frame(minHeight: 20)
|
||
|
} else {
|
||
|
HStack {
|
||
|
Image(systemName: "checkmark.circle.fill")
|
||
|
.foregroundColor(.green)
|
||
|
|
||
|
Text("Password saved in import file")
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
.foregroundColor(.primary)
|
||
|
.font(.caption)
|
||
|
.padding(.vertical, 2)
|
||
|
|
||
|
if !accountExists && (account.password.isNil || account.password!.isEmpty) {
|
||
|
SecureField("Password", text: $password)
|
||
|
.onChange(of: password) { _ in validate() }
|
||
|
#if !os(tvOS)
|
||
|
.textFieldStyle(RoundedBorderTextFieldStyle())
|
||
|
#endif
|
||
|
}
|
||
|
}
|
||
|
.frame(maxWidth: .infinity, alignment: .leading)
|
||
|
.contentShape(Rectangle())
|
||
|
.onChange(of: isValid) { _ in afterValidation() }
|
||
|
.animation(nil, value: isChecked)
|
||
|
}
|
||
|
.buttonStyle(.plain)
|
||
|
}
|
||
|
|
||
|
var isChecked: Bool {
|
||
|
model.isSelectedForImport(account)
|
||
|
}
|
||
|
|
||
|
var locationsSettingsGroupImporter: LocationsSettingsGroupImporter? {
|
||
|
fileModel.locationsSettingsGroupImporter
|
||
|
}
|
||
|
|
||
|
var accounts: [Account] {
|
||
|
fileModel.locationsSettingsGroupImporter?.accounts ?? []
|
||
|
}
|
||
|
|
||
|
private var instance: Instance! {
|
||
|
(fileModel.locationsSettingsGroupImporter?.instances ?? []).first { $0.id == account.instanceID }
|
||
|
}
|
||
|
|
||
|
private var validator: AccountValidator {
|
||
|
AccountValidator(
|
||
|
app: .constant(instance.app),
|
||
|
url: instance.apiURLString,
|
||
|
account: Account(instanceID: instance.id, urlString: instance.apiURLString, username: account.username, password: password),
|
||
|
id: .constant(account.username),
|
||
|
isValid: $isValid,
|
||
|
isValidated: $isValidated,
|
||
|
isValidating: $isValidating,
|
||
|
error: $validationError
|
||
|
)
|
||
|
}
|
||
|
|
||
|
private func validate() {
|
||
|
isValid = false
|
||
|
validationDebounce.invalidate()
|
||
|
|
||
|
guard !account.username.isEmpty, !password.isEmpty else {
|
||
|
validator.reset()
|
||
|
return
|
||
|
}
|
||
|
|
||
|
isValidating = true
|
||
|
|
||
|
validationDebounce.debouncing(1) {
|
||
|
validator.validateAccount()
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#Preview {
|
||
|
let fileModel = ImportSettingsFileModel(url: URL(string: "https://gist.githubusercontent.com/arekf/578668969c9fdef1b3828bea864c3956/raw/f794a95a20261bcb1145e656c8dda00bea339e2a/yattee-recents.yatteesettings")!)
|
||
|
|
||
|
return List {
|
||
|
ImportSettingsAccountRow(
|
||
|
account: .init(name: "arekf", urlString: "https://instance.com", username: "arekf"),
|
||
|
fileModel: fileModel
|
||
|
)
|
||
|
ImportSettingsAccountRow(
|
||
|
account: .init(name: "arekf", urlString: "https://instance.com", username: "arekf", password: "a"),
|
||
|
fileModel: fileModel
|
||
|
)
|
||
|
}
|
||
|
}
|