如何在SwiftUI列表中显示领域结果?
我已经能够将数据保存在Realm数据库中,但是无法在SwiftUI List
中显示结果.
I have been able to save data in a Realm database, but have been unable to show the results in a SwiftUI List
.
我知道我有数据,并且在控制台中打印结果没有问题.
I know I have the data and have no problem printing the results in the console.
是否可以将Realm Result
转换为可以在SwiftUI List
上显示的格式?
Is there a way to convert Realm Result
into a format that can be displayed on a SwiftUI List
?
import SwiftUI
import RealmSwift
import Combine
class Dog: Object {
@objc dynamic var name = ""
@objc dynamic var age = 0
override static func primaryKey() -> String? {
return "name"
}
}
class SaveDog {
func saveDog(name: String, age: String) {
let dog = Dog()
dog.age = Int(age)!
dog.name = name
// Get the default Realm
let realm = try! Realm()
print(Realm.Configuration.defaultConfiguration.fileURL!)
// Persist your data easily
try! realm.write {
realm.add(dog)
}
print(dog)
}
}
class RealmResults: BindableObject {
let didChange = PassthroughSubject<Void, Never>()
func getRealmResults() -> String{
let realm = try! Realm()
var results = realm.objects(Dog.self) { didSet
{didChange.send(())}}
print(results)
return results.first!.name
}
}
struct dogRow: View {
var dog = Dog()
var body: some View {
HStack {
Text(dog.name)
Text("\(dog.age)")
}
}
}
struct ContentView : View {
@State var dogName: String = ""
@State var dogAge: String = ""
let saveDog = SaveDog()
@ObjectBinding var savedResults = RealmResults()
let realm = try! Realm()
let dogs = Dog()
var body: some View {
VStack {
Text("Hello World")
TextField($dogName)
TextField($dogAge)
Button(action: {
self.saveDog.saveDog(name: self.dogName,
age:self.dogAge)
// self.savedResults.getRealmResults()
}) {
Text("Save")
}
//insert list here to show realm data
List(0 ..< 5) {
item in
Text(self.savedResults.getRealmResults())
} //Displays the same thing 5 times
}
}
}
#if DEBUG
struct ContentView_Previews : PreviewProvider {
static var previews: some View {
ContentView()
}
}
#endif
某些代码可能没有意义,因为我正在尝试几种方法来查看是否有任何效果.
Some of the code probably may not make sense because I was attempting several approaches to see if anything would work.
例如,此行将在列表视图中显示结果.
This line, for example, will display the result in the List View.
return results.first!.name
如果我只是返回结果,则列表文本视图中什么也不会显示.
If I just return results, nothing displays in the List Text View.
正如我在下面评论的那样,我将在有空的时候尝试使用ForEach方法.看起来很有希望.
As I have commented below I will attempt the ForEach approach when I have time. That looks promising.
您在List
或ForEach
中传递的数据必须符合Identifiable
协议.
The data that you pass in List
or a ForEach
must conform to the Identifiable
protocol.
您可以在Realm模型中采用它,也可以使用.identified(by:)
方法.
Either you adopt it in your Realm models or you use .identified(by:)
method.
即使那样,如果数据更改,View
也不会重新加载.
Even with that, the View
won't reload if the data changes.
您可以包装Results
并将其设置为BindableObject
,以便视图可以检测到更改并重新加载自身:
You could wrap Results
and make it a BindableObject
, so the view can detect the changes and reload itself:
class BindableResults<Element>: ObservableObject where Element: RealmSwift.RealmCollectionValue {
var results: Results<Element>
private var token: NotificationToken!
init(results: Results<Element>) {
self.results = results
lateInit()
}
func lateInit() {
token = results.observe { [weak self] _ in
self?.objectWillChange.send()
}
}
deinit {
token.invalidate()
}
}
并像这样使用它:
struct ContentView : View {
@ObservedObject var dogs = BindableResults(results: try! Realm().objects(Dog.self))
var body: some View {
List(dogs.results.identified(by: \.name)) { dog in
DogRow(dog: dog)
}
}
}