-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathProductPage.swift
189 lines (166 loc) · 5.16 KB
/
ProductPage.swift
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
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
import SwiftUI
import Northwind
/**
* A SwiftUI view managing a form that shows a product and allows editing
* of the same.
*
* This also demos fetching of related records (the product supplier and the
* product category).
*/
@MainActor
struct ProductPage: View {
/// The database is passed down by the Application struct in the environment.
@Environment(\.database) private var database
/// The snapshot is the current value that got fetched from the database
let snapshot : Product
/// This is used to tell the list about a change we did. I.e. a save.
/// In a real app there should be more app structure around this, but that
/// won't be "Lighter", but "Heavier" (stay tuned™️).
let onSave : ( Product ) -> Void
/// If we allow editing, we can keep a copy of the snapshot for that purpose.
@State private var product : Product = Product(id: 0, productName: "")
@State private var supplier : Supplier?
@State private var category : NorthwindCategory?
/// Since SQLite records are by definition equatable, we can just compare the
/// full record for changes.
private var hasChanges : Bool {
snapshot != product
}
func setup() async {
do {
// We also fetch the supplier and category info using the relationship
// features of Lighter/Enlighter.
// Those can be run concurrently (even though it doesn't give much here,
// for demo purposes only). We can use `async let` for this.
let product = product // for strict concurrency
let database = database
async let supplier = database.suppliers .find(for: product)
async let category = database.categories.find(for: product)
( self.supplier, self.category ) = try await ( supplier, category )
}
catch { // Invest the time for proper error handling...
print("ERROR: Failed to fetch a relationship.")
}
}
/// Revert changes
private func revert() {
product = snapshot
}
private func save() async {
do {
// While not required, it is always a good idea to update in an
// transaction as this ensures order and consistency.
let product = product // strict concurrency
let supplier = supplier
try await database.transaction { tx in
try tx.update(product)
// we always save the supplier, demo only.
if let supplier = supplier {
try tx.update(supplier)
}
}
await MainActor.run {
// Tell the parent view that the record got modified, so that the
// List can be updated.
self.onSave(product)
}
}
catch {
print("ERROR: failed to save:", error)
}
}
var body: some View {
Group {
#if os(macOS)
macOS
#else
iOS
#endif
}
.task { await setup() }
.onChange(of: snapshot) { newValue in
// If the selection changes, make a copy of the fetched record for
// editing.
product = newValue
}
.onAppear {
// If the view loads, make a copy of the fetched record for editing.
product = snapshot
}
.toolbar {
ToolbarItem(placement: .destructiveAction) {
Button(action: revert) {
Label("revert", systemImage: "arrow.counterclockwise")
}
.keyboardShortcut("r", modifiers: [ .command ])
.help("Revert changes made to the record.")
.disabled(!hasChanges)
}
ToolbarItem(placement: .confirmationAction) {
Button(action: { Task { await self.save() } }) {
Label("save", systemImage: "checkmark.circle")
}
.keyboardShortcut("s", modifiers: [ .command ])
.help("Save changes made to the record.")
.disabled(!hasChanges)
}
}
}
private var iOS: some View {
Form {
ProductForm(product: $product)
.padding()
.navigationTitle(product.productName)
if let category = category {
Section("Category") {
CategoryInfo(category: category)
}
}
if let supplier = supplier {
SupplierForm(supplier: Binding( // deal with the nil value
get: { supplier }, set: { self.supplier = $0 }
))
.labelStyle(.titleAndIcon)
}
}
}
private var macOS: some View {
VStack(alignment: .center, spacing: 0) {
HStack {
Spacer()
Text(product.productName)
.font(.title)
Spacer()
}
.padding()
.background(alignment: .trailing) {
Text("Edited")
.font(.footnote)
.foregroundColor(.red)
.opacity(hasChanges ? 1.0 : 0.0)
.padding()
}
Divider()
ScrollView {
Form {
ProductForm(product: $product)
}
.padding()
if let category = category {
Divider()
CategoryInfo(category: category)
}
if let supplier = supplier {
Divider()
Form {
SupplierForm(supplier: Binding( // deal with the nil value
get: { supplier }, set: { self.supplier = $0 }
))
}
.padding([ .horizontal, .bottom ])
}
Spacer()
}
}
}
}