網站首頁 編程語言 正文
手機權限檢查和申請
簡介
使用flutter進行app開發(fā),一定會用到手機的部分權限,包括通知推送、定位、相冊、存儲、相機、麥克風等。
而權限的檢查和獲取,最受歡迎的就是通過permission_handler這個插件來實現。
插件安裝
flutter pub add permission_handler
關于插件的具體內容,可以查看pub.dev https://pub.dev/packages/permission_handler
基本使用
獲取權限狀態(tài)
var status = await Permission.notification.status;
print(status);
status有以下幾種值:
isDenied:
- 在 iOS 上代表還未申請權限;
- 在android上代表還未申請權限或之前拒絕了權限。
isGranted:擁有全部權限。
isLimited:擁有部分權限。
isRestricted:擁有部分權限(僅限iOS)。
isPermanentlyDenied:權限已被永久拒絕。
申請權限
通過調用 request() 來獲取權限。在iOS上,首次申請權限才會自動彈出權限申請的對話框
// 申請通知權限:
await Permission.notification.request();
// 同時申請多個權限:
await [
Permission.notification,
Permission.photos,
].request();
用戶拒絕權限請求后的處理
Android:申請權限時
如果用戶選擇"拒絕",status將保持為isDenied,以后可以再次調用request()來取得權限;
如果用戶選擇"拒絕并不再詢問",status將變?yōu)閕sPermanentlyDenied,即以后無法再調用request()來取得權限,必須去應用設置界面手動開啟權限。
iOS:首次申請權限時,如果用戶選擇了"拒絕",status將變?yōu)閕sPermanentlyDenied,即以后無法再調用request()來取得權限,必須去應用設置界面手動開啟權限。
跳轉到應用設置界面
openAppSettings();
封裝
為了方便權限的檢查和請求,可以有兩種方式進行封裝:
封裝成方法
在該方法內進行權限判斷并彈窗來進行權限說明和操作。這種方式比較簡單,但是有個弊端就是當用戶去到應用設置界面操作權限后,返回應用時無法自動判斷用戶操作情況。所以推薦使用第二種方法。
封裝成一個頁面
這個頁面只有一個透明的Container,當需要進行權限判斷時,導航至該頁面,在這個頁面創(chuàng)建時進行權限判斷和操作。當用戶去到應用設置界面操作權限后,返回應用時使用 WidgetsBindingObserver 的 didChangeAppLifecycleState(AppLifecycleState state) 方法來再次進行權限判斷。以下為封裝之后的文件 permission_request.dart ,可直接復制粘貼后使用。
import 'dart:io';
import 'package:flutter/cupertino.dart';
import 'package:flutter/services.dart';
import 'package:permission_handler/permission_handler.dart';
/// 權限檢查及請求
///
/// 外部可通過此方法來進行權限的檢查和請求,將自動跳轉到`PermissionRequestPage`頁面。
///
/// 傳入 `Permission` 以及對應的權限名稱 `permissionTypeStr`,如果有權限則返回 `Future true`
///
/// `isRequiredPermission` 如果為 `true`,則 "取消" 按鈕將執(zhí)行 "退出app" 的操作
Future<bool> permissionCheckAndRequest(
BuildContext context,
Permission permission,
String permissionTypeStr,{
bool isRequiredPermission = false
}) async {
if (!await permission.status.isGranted) {
await Navigator.of(context).push(
PageRouteBuilder(
opaque: false,
pageBuilder: ((context, animation, secondaryAnimation) {
return PermissionRequestPage(permission, permissionTypeStr, isRequiredPermission: isRequiredPermission);
}))
);
} else {
return true;
}
return false;
}
class PermissionRequestPage extends StatefulWidget {
const PermissionRequestPage(this.permission, this.permissionTypeStr,{super.key, this.isRequiredPermission = false});
final Permission permission;
final String permissionTypeStr;
final bool isRequiredPermission;
@override
State<PermissionRequestPage> createState() => _PermissionRequestPageState();
}
class _PermissionRequestPageState extends State<PermissionRequestPage> with WidgetsBindingObserver {
bool _isGoSetting = false;
late final List<String> msgList;
@override
void initState() {
super.initState();
WidgetsBinding.instance.addObserver(this);
msgList = [
"${widget.permissionTypeStr}功能需要獲取您設備的${widget.permissionTypeStr}權限,否則可能無法正常工作。\n是否申請${widget.permissionTypeStr}權限?",
"${widget.permissionTypeStr}權限不全,是否重新申請權限?",
"沒有${widget.permissionTypeStr}權限,您可以手動開啟權限",
widget.isRequiredPermission ? "退出應用" : "取消"
];
checkPermission(widget.permission);
}
@override
void didChangeAppLifecycleState(AppLifecycleState state) {
super.didChangeAppLifecycleState(state);
// 監(jiān)聽 app 從后臺切回前臺
if (state == AppLifecycleState.resumed && _isGoSetting) {
checkPermission(widget.permission);
}
}
/// 校驗權限
void checkPermission(Permission permission) async {
final status = await permission.status;
if (status.isGranted) {
_popPage();
return;
}
// 還未申請權限或之前拒絕了權限(在 iOS 上為首次申請權限,拒絕后將變?yōu)?`永久拒絕權限`)
if (status.isDenied) {
showAlert(permission, msgList[0], msgList[3], _isGoSetting ? "前往應用中心" : "確定");
}
// 權限已被永久拒絕
if (status.isPermanentlyDenied) {
_isGoSetting = true;
showAlert(permission, msgList[2], msgList[3], _isGoSetting ? "前往應用中心" : "確定");
}
// 擁有部分權限
if (status.isLimited) {
if (Platform.isIOS || Platform.isMacOS) _isGoSetting = true;
showAlert(permission, msgList[1], msgList[3], _isGoSetting ? "前往應用中心" : "確定");
}
// 擁有部分權限(僅限 iOS)
if (status.isRestricted) {
if (Platform.isIOS || Platform.isMacOS) _isGoSetting = true;
showAlert(permission, msgList[1], msgList[3], _isGoSetting ? "前往應用中心" : "確定");
}
}
void showAlert(Permission permission, String message, String cancelMsg, String confirmMsg) {
showCupertinoDialog(
context: context,
builder: (context) {
return CupertinoAlertDialog(
title: const Text("溫馨提示"),
content: Text(message),
actions: [
CupertinoDialogAction(
child: Text(cancelMsg),
onPressed: () {
widget.isRequiredPermission ? _quitApp() : _popDialogAndPage(context);
}),
CupertinoDialogAction(
child: Text(confirmMsg),
onPressed: () {
if (_isGoSetting) {
openAppSettings();
_isGoSetting = true;
} else {
requestPermisson(permission);
}
_popDialog(context);
})
],
);
}
);
}
/// 申請權限
void requestPermisson(Permission permission) async {
// 申請權限
await permission.request();
// 再次校驗
checkPermission(permission);
}
@override
void dispose() {
WidgetsBinding.instance.removeObserver(this);
super.dispose();
}
@override
Widget build(BuildContext context) {
return Container();
}
/// 退出應用程序
void _quitApp() {
SystemChannels.platform.invokeMethod("SystemNavigator.pop");
}
/// 關閉整個權限申請頁面
void _popDialogAndPage(BuildContext dialogContext) {
_popDialog(dialogContext);
_popPage();
}
/// 關閉彈窗
void _popDialog(BuildContext dialogContext) {
Navigator.of(dialogContext).pop();
}
/// 關閉透明頁面
void _popPage() {
Navigator.of(context).pop();
}
}
AndroidManifest.xml
mediaflutter/example/android/app/src/main/AndroidManifest.xml
<uses-permission android:name="android.permission.CAMERA" />
在外部想要檢查及獲取權限的地方,調用permissionCheckAndRequest方法即可:
// 導入封裝好的文件
import 'package:xxxx/permission_request.dart';
// ...其它代碼
ElevatedButton(
onPressed: () async {
// 調用封裝好的權限檢查和請求方法
bool result = await permissionCheckAndRequest(
context,
Permission.notification,
"通知"
);
if (result) print("已擁有該權限");
},
child: const Text("權限檢查")
)
原文鏈接:https://blog.csdn.net/weixin_43874301/article/details/127782932
相關推薦
- 2022-12-25 pycharm導入第三方庫的兩種方法(永不報錯)_python
- 2022-09-18 iOS開發(fā)探索多線程GCD隊列示例詳解_IOS
- 2022-02-19 數據結構C語言鏈表的實現介紹_C 語言
- 2022-07-26 Nginx本地配置SSL訪問的實例教程_nginx
- 2022-07-19 Python數據分析之NumPy常用函數使用詳解_python
- 2022-09-24 pytorch使用nn.Moudle實現邏輯回歸_python
- 2022-11-26 .Net中Task?Parallel?Library的進階用法_實用技巧
- 2022-06-02 利用Python繪畫雙擺操作分享_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學習環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發(fā)現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支