-
Notifications
You must be signed in to change notification settings - Fork 1.8k
Expand file tree
/
Copy pathSettingsTextField.swift
More file actions
69 lines (63 loc) · 1.91 KB
/
SettingsTextField.swift
File metadata and controls
69 lines (63 loc) · 1.91 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
import SwiftUI
struct SettingsTextField: View {
let title: String
let prompt: String
@Binding var text: String
let isSecure: Bool
@State private var localText: String = ""
@State private var debounceTimer: Timer?
var onDebouncedChange: ((String) -> Void)?
init(title: String, prompt: String, text: Binding<String>, isSecure: Bool = false, onDebouncedChange: ((String) -> Void)? = nil) {
self.title = title
self.prompt = prompt
self._text = text
self.isSecure = isSecure
self.onDebouncedChange = onDebouncedChange
self._localText = State(initialValue: text.wrappedValue)
}
var body: some View {
Form {
Group {
if isSecure {
SecureField(text: $localText, prompt: Text(prompt)) {
Text(title)
}
} else {
TextField(text: $localText, prompt: Text(prompt)) {
Text(title)
}
}
}
.textFieldStyle(.plain)
.multilineTextAlignment(.trailing)
.onChange(of: localText) { newValue in
text = newValue
debounceTimer?.invalidate()
debounceTimer = Timer.scheduledTimer(withTimeInterval: 3.0, repeats: false) { _ in
onDebouncedChange?(newValue)
}
}
.onAppear {
localText = text
}
}
.padding(10)
}
}
#Preview {
VStack(spacing: 10) {
SettingsTextField(
title: "Username",
prompt: "user",
text: .constant("")
)
Divider()
SettingsTextField(
title: "Password",
prompt: "pass",
text: .constant(""),
isSecure: true
)
}
.padding(.vertical, 10)
}