请原谅我缺乏知识,我正在尝试制作我的第一个iOS应用,我的目标是将我所有的联系人导入到swiftui视图中:
//ContentView.swift
import SwiftUI
struct ContentView: View {
var contact = contactData
@ObservedObject var contacts = ContactList()
var body: some View {
NavigationView {
List {
ForEach(contact) { item in
VStack(alignment: .leading) {
HStack {
Text(item.contactName)
}
}
}
.onDelete { index in
self.contacts.contact.remove(at: index.first!)
}
}
.navigationBarTitle(Text("Contacts"))
.navigationBarItems(
trailing: EditButton()
)
}
}
}
#if DEBUG
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
ContentView()
}
}
#endif
struct Contacts: Identifiable {
var id = UUID()
var contactName: String
}
let contactData = [
Contacts(contactName: "Foo Bar"),
Contacts(contactName: "Johnny Appleseed")
]
和
//ContactList.swift
import Combine
class ContactList: ObservableObject {
@Published var contact: [Contacts] = contactData
}
使用Combine API和.onDelete
函数,我想删除多个联系人(当前不是iOS中的功能),然后将其返回到联系人应用中。
尽管在获取联系人列表方面我还是很棘手,但是我尝试了多种不同的方式来使用Swift进行操作,例如:Fetching all contacts in iOS Swift?
var contacts = [CNContact]()
let keys = [CNContactFormatter.descriptorForRequiredKeysForStyle(.FullName)]
let request = CNContactFetchRequest(keysToFetch: keys)
do {
try self.contactStore.enumerateContactsWithFetchRequest(request) {
(contact, stop) in
// Array containing all unified contacts from everywhere
contacts.append(contact)
}
}
catch {
print("unable to fetch contacts")
}
但是,如果不抛出大量错误,这似乎就无法正常工作。目前,我并不担心删除,而是主要集中在仅导入联系人上。
有人知道使用SwiftUI和Contacts API是否有很好的方法?或至少可以指出我正确的方向?我了解我使用的是Xcode 11 Beta 5,这可能会导致弃用不同的API产生问题,但似乎Contact API在Xcode 11中相对没有变化。
答案 0 :(得分:2)
您通常使用ObservableObject
进行模型/控制器交互。将它们发布会使得我可以将结果分配给变量。
当前它会在onAppear
异步内部开始获取操作,因为否则会阻止UI加载,并且无法请求联系人。
缺少的是更好的授权处理(请参阅this),deletion/creation,刷新(在其他地方调用获取)和更全面的功能。
有条件的允许我向联系人显示或错误,从而可以快速发现错误。
注意:您必须在 info.plist 中添加“隐私联系人”描述,以免崩溃。否则它应该可以正常工作,而无需进行其他任何更改。
注2:由于 SwiftUI 仅按接线方式和 UI 如@dfd所述进行了更改。如何检索联系人仍然完全相同。
import Contacts
import SwiftUI
import os
class ContactStore: ObservableObject {
@Published var contacts: [CNContact] = []
@Published var error: Error? = nil
func fetch() {
os_log("Fetching contacts")
do {
let store = CNContactStore()
let keysToFetch = [CNContactGivenNameKey as CNKeyDescriptor,
CNContactMiddleNameKey as CNKeyDescriptor,
CNContactFamilyNameKey as CNKeyDescriptor,
CNContactImageDataAvailableKey as CNKeyDescriptor,
CNContactImageDataKey as CNKeyDescriptor]
os_log("Fetching contacts: now")
let containerId = store.defaultContainerIdentifier()
let predicate = CNContact.predicateForContactsInContainer(withIdentifier: containerId)
let contacts = try store.unifiedContacts(matching: predicate, keysToFetch: keysToFetch)
os_log("Fetching contacts: succesfull with count = %d", contacts.count)
self.contacts = contacts
} catch {
os_log("Fetching contacts: failed with %@", error.localizedDescription)
self.error = error
}
}
}
extension CNContact: Identifiable {
var name: String {
return [givenName, middleName, familyName].filter{ $0.count > 0}.joined(separator: " ")
}
}
struct ContactsView: View {
@EnvironmentObject var store: ContactStore
var body: some View {
VStack{
Text("Contacts")
if store.error == nil {
List(store.contacts) { (contact: CNContact) in
return Text(contact.name)
}.onAppear{
DispatchQueue.main.async {
self.store.fetch()
}
}
} else {
Text("error: \(store.error!.localizedDescription)")
}
}
}
}
struct ContactsViewOrError: View {
var body: some View {
ContactsView().environmentObject(ContactStore())
}
}