主要总结下以下常用权限的获取及请求授权用法等(均不考虑iOS8以下系统):
网络权限 推送权限 定位权限 通讯录权限 相机权限 相册权限
网络权限
使用系统CoreTelephony库实现网络权限的获取及监听
- 获取网络权限
func getTelephonyAuthorization() {
// 获取网络权限状态
let cellularData = CTCellularData.init()
switch cellularData.restrictedState {
case .restricted:
print("restricted");
case .notRestricted:
print("notRestricted");
case .restrictedStateUnknown:
print("restrictedStateUnknown");
}
}
- 请求(申请)网络权限
系统没有提供接口供开发者手动请求网络权限,iOS10及以上系统,应用首次请求网络时,系统会自动弹出权限选择框;而且一个应用只会弹出一次提示,提示过后就算卸载重装也不再提示;
ps:首次请求时,如
Alamofire.request("https://www.baidu.com")
弹出权限提示框
- iOS10及以上系统网络权限相关处理
iOS 10 之后首次安装的应用,请求网络时会弹出上面的提示框;在用户未选择前,所有网络请求都是失败的,这就会造成首页加载不出数据一片空白的情况;而且就算用户点击了允许,但在点击前的网络请求不会自动再次调用;对于这种情况,有几种解决办法:
- 封装网络框架,请求失败时,定时重新请求
- 提供数据空白页,空白页提供类似“重新加载”的按钮,允许及引导用户手动重新请求;
- 监听网络权限的变化,当监听到网络权限更改为允许后,重新请求;
let cellularData = CTCellularData.init()
func requestTelephonyAuthorization() {
// 监听网络权限变化
cellularData.cellularDataRestrictionDidUpdateNotifier = {state in
switch state {
case .restricted:
print("restricted");
case .notRestricted:
print("notRestricted");
case .restrictedStateUnknown:
print("restrictedStateUnknown");
}
}
}
详细处理流程,可参考iOS 10 之 网络权限带来的坑
推送权限
推送相关API iOS10更新了,对应的权限API有所不同;
- 获取推送权限
系统并没有提供获取推送权限状态的API,但可以通过NotificationSettings判断:
func getNotificationAuthorization() {
if #available(iOS 10.0, *) {
UNUserNotificationCenter.current().getNotificationSettings { (settings) in
switch settings.authorizationStatus.rawValue {
case 0:
print("not authorized")
default:
print("authorized")
}
}
}else {
guard let settings = UIApplication.shared.currentUserNotificationSettings else {
print("not settings");
return;
}
switch settings.types.rawValue {
case 0:
print("not authorized")
default:
print("authorized")
}
}
}
- 请求推送权限
iOS10之前系统,对于请求(申请)推送权限也没有具体的API;只有当应用设置NotificationSettings时,会自动请求推送权限并弹出提示框(提示框应用只会提示一次,不同于网络权限,应用卸载重装后,首次设置NotificationSettings时也会弹出提示框);
iOS10及之后系统,同上面一样;但系统增加了请求推送权限的接口,可以手动调用接口来请求申请权限,对于手动请求的还可以监听到用户的选项(允许还是拒绝);手动请求权限,应用也只会首次时弹提示框;
func requestNotificationAuthorization() {
if #available(iOS 10.0, *) {
let center = UNUserNotificationCenter.current()
center.requestAuthorization(options: [.alert,.sound,.badge]) { (granted, error) in
if granted {
print("authorized");
}else {
print("not authorized");
}
}
} else {
let setting = UIUserNotificationSettings.init(types: [.alert,.sound,.badge], categories: nil)
UIApplication.shared.registerUserNotificationSettings(setting)
}
}
定位权限
定位权限有两种情况,一种是针对手机的(全部应用),一种是针对当前应用;两种情况都有对应接口获取权限状态;
- 获取定位权限
func getLocationAuthorization() {
// 手机的定位权限
if !CLLocationManager.locationServicesEnabled() {
print("disenable")
return;
}
// 应用的定位权限
let status = CLLocationManager.authorizationStatus()
switch status {
case .authorizedAlways:
print("always")
case .authorizedWhenInUse:
print("authorizedWhenInUse")
case .denied:
print("denied")
case .notDetermined:
print("notDetermined")
case .restricted:
print("restricted")
}
}
- 请求定位权限
使用相应接口定位时,会自动请求权限;也可以调用接口手动请求,应用只有首次使用时才会弹出权限提示框;与其他权限不同的是,手动请求定位权限是通过delegate回调的,在回调中可以监听并获取到用户的选项;
let manager = CLLocationManager.init()
func requestLocationAuthorization() {
manager.delegate = self
manager.requestAlwaysAuthorization()
manager.requestWhenInUseAuthorization()
}
func locationManager(_ manager: CLLocationManager, didChangeAuthorization status: CLAuthorizationStatus) {
switch status {
case .authorizedAlways:
print("always")
case .authorizedWhenInUse:
print("authorizedWhenInUse")
case .denied:
print("denied")
case .notDetermined:
print("notDetermined")
case .restricted:
print("restricted")
}
}
- 权限描述
为了让用户清楚了解申请权限的具体用途,iOS已强制开发者为某些权限在info.plist中添加对应的描述;该描述将显示在权限提示框上;这个描述一定要阐明权限的具体用途,比如该权限将用于什么功能之类的,否则app审核会被拒(亲身经历,血的教训);
定位权限描述的key有:
NSLocationAlwaysAndWhenInUseUsageDescription
,
NSLocationWhenInUseUsageDescription
通讯录权限
通讯录相关API iOS9更新了,对应的权限API有所不同;
- 获取通讯录权限
func getContactAuthorization() {
if #available(iOS 9.0, *) {
let status = CNContactStore.authorizationStatus(for: .contacts)
// ...
}else {
let status = ABAddressBookGetAuthorizationStatus()
// ...
}
}
- 请求通讯录权限
和上面权限类似;可以自动,也可以手动请求并监听权限
func requestContactAuthorization() {
if #available(iOS 9.0, *) {
let contact = CNContactStore.init()
contact.requestAccess(for: .contacts) { (granted, error) in
// ...
}
}else {
let addressBook = ABAddressBookCreateWithOptions(nil, nil)
ABAddressBookRequestAccessWithCompletion(addressBook as ABAddressBook) { (granted, error) in
// ...
}
}
}
- 权限描述
key:NSContactsUsageDescription
相机权限
- 获取相机权限
func getVideoAuthorization() {
let videoAuthorStatus = AVCaptureDevice.authorizationStatus(for: .video)
switch videoAuthorStatus {
case .authorized:
print("authorized");
case .denied:
print("denied");
case .notDetermined:
print("notDetermined");
case .restricted:
print("restricted");
}
}
- 请求相机权限
func requestVideoAuthorization() {
AVCaptureDevice.requestAccess(for: .video, completionHandler: { (granted) in
if granted {
print("authorized");
}else {
print("not authorized");
}
})
}
- 权限描述
key:NSCameraUsageDescription
相册权限
访问系统相册,保存图片至手机相册均需要该权限;
- 获取相册权限
func getPhotoAuthorization() {
let photoAuthorStatus = PHPhotoLibrary.authorizationStatus()
switch photoAuthorStatus {
case .authorized:
print("authorized");
case .denied:
print("denied");
case .notDetermined:
print("notDetermined");
case .restricted:
print("restricted");
}
}
- 请求相册权限
func requestPhotoAuthorization() {
PHPhotoLibrary.requestAuthorization { (status) in
if status == PHAuthorizationStatus.authorized {
print("authorized");
}else {
print("not authorized");
}
}
}
- 权限描述
key:NSPhotoLibraryUsageDescription
开发中使用场景
- 在首次使用具体某些功能时(如拍照,读取照片),系统会自动弹出提示框,向用户申请对应权限;某些权限,开发者也可以手动调用接口来请求、向用户申请相应权限;具体是选择自动请求还是手动请求,可以视需求而定;对于不是频繁使用的功能,可以在使用该功能时再申请权限,也即让系统自动请求;而手动调用接口请求,主要作用是:可以提前申请权限,或者在应用启动后统一申请要用到的权限;
- 需要用户授权的功能,有可能被用户拒绝授权;所以使用这些功能前要先获取对应的权限状态,如果权限是被拒绝的就需要进一步处理;一般的处理办法是,弹出提示框引导用户开启该权限,最好是提供一个按钮能直接跳转至系统设置界面;跳转至系统设置界面的代码:
// iOS8及以上系统
UIApplication.shared.openURL(NSURL.init(string: UIApplication.openSettingsURLString) as! URL)
- 引导用户去设置界面开启权限时,我们希望能得到反馈(即用户是否真的去开启了权限),如果用户去设置界面开启了权限后,再重新实现具体功能;比如,使用拍照功能时,首先判断相机权限状态,获取的状态是拒绝的就弹出提示框引导用户去设置界面开启;用户开启权限后,应用再重新调用拍照功能,就不需要用户再多做一步操作了;
但用户是否在设置界面开启权限,我们是不得而知的;但可以通过其他方式实现:
- 开启定时器,每间隔一段时间重新获取权限状态;当权限状态为允许时,再做相关处理;
- 当用户切换至系统设置界面再回到当前应用时,当前应用状态肯定会经过
BecomeActive
这个状态,那么我们可以添加UIApplication.didBecomeActiveNotification
通知监听这个状态,做相应处理;
NotificationCenter.default.addObserver(self, selector: #selector(becomeActive), name: UIApplication.didBecomeActiveNotification, object: nil)
以上权限中,通讯录权限、相册权限、相机权限当用户切换至系统设置界面更改后,应用将被kill