- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在构建一个 UI 来请求访问 CNContactStore 并处理用户之前(并且可能错误地)拒绝访问的情况。当我检测到当前状态为 .denied 时,我会显示一个 UIAlertController,它会解释并提议将它们带到应用程序设置以允许访问。
在当前状态为 .denied 的情况下调用 store.requestAccess() 时,应用程序会崩溃并显示两个错误:1) “A promise was finished with a nil error”。和 2) “一个 promise 被错误地完成了。”。调用堆栈如下所示。
我不擅长解释调用堆栈,但我认为错误来自 CNContactStore 内部。我不清楚我能做些什么来防止这个错误。
编辑:我根本没有在我的应用程序中使用 promise 链。
EDIT2:上面明确说明了我的代码中发生错误的确切位置。
import UIKit
import Contacts
import ContactsUI
final class ContactsAppHelper {
static let shared = ContactsAppHelper()
var store = CNContactStore()
func checkAccessStatus(_ completionHandler: @escaping (_ accessGranted: Bool) -> Void) {
let authorizationStatus = CNContactStore.authorizationStatus(for: .contacts)
switch authorizationStatus {
case .authorized:
completionHandler(true)
case .denied, .notDetermined:
store.requestAccess(for: .contacts, completionHandler: { (access, accessError) -> Void in
if access {
completionHandler(access)
}
else {
print("access denied")
DispatchQueue.main.sync {
self.showSettingsAlert(completionHandler)
}
}
})
default:
completionHandler(false)
}
}
private func showSettingsAlert(_ completionHandler: @escaping (_ accessGranted: Bool) -> Void) {
let msg = "This app requires access to Contacts to proceed. Would you like to open settings and grant permission?"
let alert = UIAlertController(title: nil, message: msg, preferredStyle: .alert)
alert.addAction(UIAlertAction(title: "Open Settings", style: .default) { action in
completionHandler(false)
UIApplication.shared.open(URL(string: UIApplication.openSettingsURLString)!)
})
alert.addAction(UIAlertAction(title: "Cancel", style: .cancel) { action in
completionHandler(false)
})
let vc = UIApplication.getPresentedViewController()
vc!.present(alert, animated: true)
}
}
extension UIApplication{
class func getPresentedViewController() -> UIViewController? {
var presentViewController = UIApplication.shared.keyWindow?.rootViewController
while let pVC = presentViewController?.presentedViewController
{
presentViewController = pVC
}
return presentViewController
}
}
这些是产生的调用堆栈:
2019-06-14 15:59:12.220116-0700 Sales Networker[805:28798] [Rx] A promise was finished with a nil error.
2019-06-14 15:59:12.226500-0700 Sales Networker[805:28798] [Rx] Call stack: (
0 ContactsFoundation 0x000000012242312a -[CNFuture finishWithError:] + 52
1 Contacts 0x000000010e0c2a82 __55-[CNDataMapperContactStore requestAccessForEntityType:]_block_invoke_2 + 187
2 Contacts 0x000000010e0ffc71 +[CNAuthorization requestAccessForEntityType:completionHandler:] + 77
3 Contacts 0x000000010e072c44 -[CNXPCDataMapper requestAccessForEntityType:completionHandler:] + 123
4 Contacts 0x000000010e0c299a __55-[CNDataMapperContactStore requestAccessForEntityType:]_block_invoke + 174
5 libsystem_trace.dylib 0x000000011269cf00 os_activity_apply_f + 66
6 Contacts 0x000000010e116d52 -[_CNContactsLogger requestingAccessForContacts:] + 225
7 Contacts 0x000000010e0c28a4 -[CNDataMapperContactStore requestAccessForEntityType:] + 177
8 Contacts 0x000000010e09b77f -[CNContactStore requestAccessForEntityType:completionHandler:] + 54
9 Sales Networker 0x000000010ccaadc7 $s15Sales_Networker17ContactsAppHelperC17checkAccessStatusyyySbcF + 535
10 Sales Networker 0x000000010cce9673 $s15Sales_Networker24OnBoardingViewControllerC17rightButtonTappedyySo8UIButtonCF + 451
11 Sales Networker 0x000000010cce9ddc $s15Sales_Networker24OnBoardingViewControllerC17rightButtonTappedyySo8UIButtonCFTo + 60
12 UIKitCore 0x0000000119c7d204 -[UIApplication sendAction:to:from:forEvent:] + 83
13 UIKitCore 0x00000001196d2c19 -[UIControl sendAction:to:forEvent:] + 67
14 UIKitCore 0x00000001196d2f36 -[UIControl _sendActionsForEvents:withEvent:] + 450
15 UIKitCore 0x00000001196d1eec -[UIControl touchesEnded:withEvent:] + 583
16 UIKitCore 0x0000000119cb5eee -[UIWindow _sendTouchesForEvent:] + 2547
17 UIKitCore 0x0000000119cb75d2 -[UIWindow sendEvent:] + 4079
18 UIKitCore 0x0000000119c95d16 -[UIApplication sendEvent:] + 356
19 UIKitCore 0x0000000119d66293 __dispatchPreprocessedEventFromEventQueue + 3232
20 UIKitCore 0x0000000119d68bb9 __handleEventQueueInternal + 5911
21 CoreFoundation 0x000000010ebd6be1 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION__ + 17
22 CoreFoundation 0x000000010ebd6463 __CFRunLoopDoSources0 + 243
23 CoreFoundation 0x000000010ebd0b1f __CFRunLoopRun + 1231
24 CoreFoundation 0x000000010ebd0302 CFRunLoopRunSpecific + 626
25 GraphicsServices 0x0000000114a702fe GSEventRunModal + 65
26 UIKitCore 0x0000000119c7bba2 UIApplicationMain + 140
27 Sales Networker 0x000000010cca7f6b main + 75
28 libdyld.dylib 0x0000000112416541 start + 1
)
2019-06-14 15:59:12.236494-0700 Sales Networker[805:28798] [Rx] A promise was finished incorrectly.
2019-06-14 15:59:12.236582-0700 Sales Networker[805:28798] [Rx] Result: (null)
2019-06-14 15:59:12.236669-0700 Sales Networker[805:28798] [Rx] Error : (null)
2019-06-14 15:59:12.238150-0700 Sales Networker[805:28798] [Rx] Call stack: (
0 ContactsFoundation 0x0000000122422ed8 -[CNFuture finishWithResult:error:] + 290
1 Contacts 0x000000010e0c2a82 __55-[CNDataMapperContactStore requestAccessForEntityType:]_block_invoke_2 + 187
2 Contacts 0x000000010e0ffc71 +[CNAuthorization requestAccessForEntityType:completionHandler:] + 77
3 Contacts 0x000000010e072c44 -[CNXPCDataMapper requestAccessForEntityType:completionHandler:] + 123
4 Contacts 0x000000010e0c299a __55-[CNDataMapperContactStore requestAccessForEntityType:]_block_invoke + 174
5 libsystem_trace.dylib 0x000000011269cf00 os_activity_apply_f + 66
6 Contacts 0x000000010e116d52 -[_CNContactsLogger requestingAccessForContacts:] + 225
7 Contacts 0x000000010e0c28a4 -[CNDataMapperContactStore requestAccessForEntityType:] + 177
8 Contacts 0x000000010e09b77f -[CNContactStore requestAccessForEntityType:completionHandler:] + 54
9 Sales Networker 0x000000010ccaadc7 $s15Sales_Networker17ContactsAppHelperC17checkAccessStatusyyySbcF + 535
10 Sales Networker 0x000000010cce9673 $s15Sales_Networker24OnBoardingViewControllerC17rightButtonTappedyySo8UIButtonCF + 451
11 Sales Networker 0x000000010cce9ddc $s15Sales_Networker24OnBoardingViewControllerC17rightButtonTappedyySo8UIButtonCFTo + 60
12 UIKitCore 0x0000000119c7d204 -[UIApplication sendAction:to:from:forEvent:] + 83
13 UIKitCore 0x00000001196d2c19 -[UIControl sendAction:to:forEvent:] + 67
14 UIKitCore 0x00000001196d2f36 -[UIControl _sendActionsForEvents:withEvent:] + 450
15 UIKitCore 0x00000001196d1eec -[UIControl touchesEnded:withEvent:] + 583
16 UIKitCore 0x0000000119cb5eee -[UIWindow _sendTouchesForEvent:] + 2547
17 UIKitCore 0x0000000119cb75d2 -[UIWindow sendEvent:] + 4079
18 UIKitCore 0x0000000119c95d16 -[UIApplication sendEvent:] + 356
19 UIKitCore 0x0000000119d66293 __dispatchPreprocessedEventFromEventQueue + 3232
20 UIKitCore 0x0000000119d68bb9 __handleEventQueueInternal + 5911
21 CoreFoundation 0x000000010ebd6be1 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION__ + 17
22 CoreFoundation 0x000000010ebd6463 __CFRunLoopDoSources0 + 243
23 CoreFoundation 0x000000010ebd0b1f __CFRunLoopRun + 1231
24 CoreFoundation 0x000000010ebd0302 CFRunLoopRunSpecific + 626
25 GraphicsServices 0x0000000114a702fe GSEventRunModal + 65
26 UIKitCore 0x0000000119c7bba2 UIApplicationMain + 140
27 Sales Networker 0x000000010cca7f6b main + 75
28 libdyld.dylib 0x0000000112416541 start + 1
)
最佳答案
我测试了你的代码,发现这里发生了两件事。
首先是当状态已经是.denied
时调用requestAccess(for:completion:)
时,您会在控制台上得到一个非致命的堆栈跟踪.您可以忽略此问题,也可以仅在 .notDetermined
状态下请求访问权限。
第二个问题与主队列上的同步调度有关。由于某种原因,这会导致访问冲突。解决方案是使用异步 分派(dispatch)。无论如何都没有充分的理由阻塞调用队列。
func checkAccessStatus(_ completionHandler: @escaping (_ accessGranted: Bool) -> Void) {
let authorizationStatus = CNContactStore.authorizationStatus(for: .contacts)
switch authorizationStatus {
case .authorized:
completionHandler(true)
case .denied:
self.showSettingsAlert(completionHandler)
case .notDetermined:
store.requestAccess(for: .contacts, completionHandler: { (access, accessError) -> Void in
if access {
completionHandler(access)
}
else {
print("access denied")
self.showSettingsAlert(completionHandler)
}
})
default:
completionHandler(false)
}
}
private func showSettingsAlert(_ completionHandler: @escaping (_ accessGranted: Bool) -> Void) {
let msg = "This app requires access to Contacts to proceed. Would you like to open settings and grant permission?"
let alert = UIAlertController(title: nil, message: msg, preferredStyle: .alert)
alert.addAction(UIAlertAction(title: "Open Settings", style: .default) { action in
completionHandler(false)
UIApplication.shared.open(URL(string: UIApplication.openSettingsURLString)!)
})
alert.addAction(UIAlertAction(title: "Cancel", style: .cancel) { action in
completionHandler(false)
})
DispatchQueue.main.async {
if let vc = UIApplication.getPresentedViewController() {
vc.present(alert, animated: true)
} else {
completionHandler(false)
}
}
}
关于ios - 来自 CNContactStore 请求访问的错误 "A promise was finished incorrectly",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56606093/
这两者有区别吗? startActivity(intent); finish(); startActivity(intent); ThisActivityName.this.finish(); 注意:
我有一个包含大量 url 链接的 arrayOfLinks。我需要从这些链接中获取图像。我正在使用以下代码来执行此操作。 - (void)getImages { NSArray *links
我想创建一条执行以下步骤的路线 读取文件 将文件分割成行 使用 BeanIO 解码行 调用(昂贵且耗时的)处理器 我想在处理完所有线路后完成这条路线。 所以我的路线的基本布局如下所示: from("d
我正在使用 Nodejs。我有一个异步的 forEach,因为我必须在 forEach 中等待结果。结果,我需要等待 forEach 完成,然后继续处理循环的结果。我找到了几种等待 forEach 的
我正在编写需要背靠背内核调用的 android renderscript 代码(有时一个内核的输出成为另一个内核的输入)。我还有一些全局指针,绑定(bind)到 Java 层的内存。每个内核更新那些全
伪代码: void draw() { Vertex* vertices = scene.GetVertexArray(); glEnableClientState(...);
当我创建一个 Android 应用程序时,我首先在其中创建了主 Activity ,然后添加了启动 Activity 以及一个正常 Activity 。 所以我的问题是每当我在应用程序中单击退出时,它
我正在尝试将一些 API 数据获取到我的应用程序,并且我有两个 Activity 。第一个是启动屏幕(就像加载应用程序时谷歌使用的那样),我想知道在哪里调用 finish() 来结束 Activity
我使用 创建了“新文件” Activity startActivityForResult(new Intent(MainActivity.this, NewFile.class),1); NewFil
我有一个以 SingleTask 模式 (Android 2.2) 运行的 Activity ,该 Activity 从其他 Activity 或服务接收 Intent 。我的 Activity 将每
我编写了以下代码来使用 TextView 和按钮显示消息框。单击按钮时, Activity 应该完成。但这有时会起作用,但并不一致。有时它会挂起。如何让它始终如一地工作?请帮忙 public clas
我正在搜索,当 onDestroy 在 android 应用程序上执行时,我发现它是在设备资源不足(RAM、CPU) 时执行的,并且当用户调用 finish() 时。 例如,当我按下后退按钮从一个 A
我正在编写一个 Android 应用程序,它显示一个带有一些元素和一些 CRUD 操作的 RecyclerView。我想做的一件事是从我的 RecyclerView 更新现有元素,为此我打开一个新 A
被测试的 Activity 是一个简单的棋盘游戏。当满足游戏结束的必要条件时,在执行几项操作以显示谁赢了等之后,将运行以下代码: // Return to main menu after 5 seco
我有一个 Activity ,它创建了我创建的名为classA的类的新实例。在classA中,我有一个方法将对象添加到位于classA中的STATICarrayList。 所以我的 Activity
我正在使用 Facebook Android SDK,并希望在用户登录并获取用户对象后关闭我的 Activity。实际上,我存储了它的一部分,但无论如何我都想关闭该 Activity 。
我以前从事c# dotnet开发,有一个Application.exit()类型的方法,可以立即关闭应用程序并释放内存。 我发现 android 的“finish()”方法应该这样做 但它没有这样做,
我有一个 Activity ,我调用了 finish() 方法,但 Activity 没有从内存中清除。 调用 finish() 后,我看到方法 onDestroy() 已成功执行(并且我清除了其中的
我想在我的MKMapView上设置一个区域,然后找到与 map 的东北角和西南角对应的坐标。 This code works just fine to do that: //Recenter and
我正在尝试使用从顶部滑动动画将一个 View 替换为另一个 View 。我有点遵循 http://gentlebytes.com/2011/09/auto-layout-in-lion/ 中发布的指南
我是一名优秀的程序员,十分优秀!