欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Flutter手機權(quán)限檢查與申請實現(xiàn)方法詳解

 更新時間:2022年11月15日 09:07:15   作者:后端碼匠  
使用flutter進(jìn)行app開發(fā),一定會用到手機的部分權(quán)限,包括通知推送、定位、相冊、存儲、相機、麥克風(fēng)等。而權(quán)限的檢查和獲取,最受歡迎的就是通過permission_handler這個插件來實現(xiàn)

手機權(quán)限檢查和申請

簡介

使用flutter進(jìn)行app開發(fā),一定會用到手機的部分權(quán)限,包括通知推送、定位、相冊、存儲、相機、麥克風(fēng)等。

而權(quán)限的檢查和獲取,最受歡迎的就是通過permission_handler這個插件來實現(xiàn)。

插件安裝

flutter pub add permission_handler

關(guān)于插件的具體內(nèi)容,可以查看pub.dev https://pub.dev/packages/permission_handler

基本使用

獲取權(quán)限狀態(tài)

var status = await Permission.notification.status;
print(status);

status有以下幾種值:

isDenied:

  • 在 iOS 上代表還未申請權(quán)限;
  • 在android上代表還未申請權(quán)限或之前拒絕了權(quán)限。

isGranted:擁有全部權(quán)限。

isLimited:擁有部分權(quán)限。

isRestricted:擁有部分權(quán)限(僅限iOS)。

isPermanentlyDenied:權(quán)限已被永久拒絕。

申請權(quán)限

通過調(diào)用 request() 來獲取權(quán)限。在iOS上,首次申請權(quán)限才會自動彈出權(quán)限申請的對話框

// 申請通知權(quán)限:
await Permission.notification.request();
// 同時申請多個權(quán)限:
await [
  Permission.notification,
  Permission.photos,
].request();

用戶拒絕權(quán)限請求后的處理

Android:申請權(quán)限時

如果用戶選擇"拒絕",status將保持為isDenied,以后可以再次調(diào)用request()來取得權(quán)限;

如果用戶選擇"拒絕并不再詢問",status將變?yōu)閕sPermanentlyDenied,即以后無法再調(diào)用request()來取得權(quán)限,必須去應(yīng)用設(shè)置界面手動開啟權(quán)限。

iOS:首次申請權(quán)限時,如果用戶選擇了"拒絕",status將變?yōu)閕sPermanentlyDenied,即以后無法再調(diào)用request()來取得權(quán)限,必須去應(yīng)用設(shè)置界面手動開啟權(quán)限。

跳轉(zhuǎn)到應(yīng)用設(shè)置界面

openAppSettings();

封裝

為了方便權(quán)限的檢查和請求,可以有兩種方式進(jìn)行封裝:

封裝成方法

在該方法內(nèi)進(jìn)行權(quán)限判斷并彈窗來進(jìn)行權(quán)限說明和操作。這種方式比較簡單,但是有個弊端就是當(dāng)用戶去到應(yīng)用設(shè)置界面操作權(quán)限后,返回應(yīng)用時無法自動判斷用戶操作情況。所以推薦使用第二種方法。

封裝成一個頁面

這個頁面只有一個透明的Container,當(dāng)需要進(jìn)行權(quán)限判斷時,導(dǎo)航至該頁面,在這個頁面創(chuàng)建時進(jìn)行權(quán)限判斷和操作。當(dāng)用戶去到應(yīng)用設(shè)置界面操作權(quán)限后,返回應(yīng)用時使用 WidgetsBindingObserver 的 didChangeAppLifecycleState(AppLifecycleState state) 方法來再次進(jìn)行權(quán)限判斷。以下為封裝之后的文件 permission_request.dart ,可直接復(fù)制粘貼后使用。

import 'dart:io';
import 'package:flutter/cupertino.dart';
import 'package:flutter/services.dart';
import 'package:permission_handler/permission_handler.dart';
/// 權(quán)限檢查及請求
/// 
/// 外部可通過此方法來進(jìn)行權(quán)限的檢查和請求,將自動跳轉(zhuǎn)到`PermissionRequestPage`頁面。
/// 
/// 傳入 `Permission` 以及對應(yīng)的權(quán)限名稱 `permissionTypeStr`,如果有權(quán)限則返回 `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}功能需要獲取您設(shè)備的${widget.permissionTypeStr}權(quán)限,否則可能無法正常工作。\n是否申請${widget.permissionTypeStr}權(quán)限?",
      "${widget.permissionTypeStr}權(quán)限不全,是否重新申請權(quán)限?",
      "沒有${widget.permissionTypeStr}權(quán)限,您可以手動開啟權(quán)限",
      widget.isRequiredPermission ? "退出應(yīng)用" : "取消"
    ];
    checkPermission(widget.permission);
  }
  @override
  void didChangeAppLifecycleState(AppLifecycleState state) {
    super.didChangeAppLifecycleState(state);
    // 監(jiān)聽 app 從后臺切回前臺
    if (state == AppLifecycleState.resumed && _isGoSetting) {
      checkPermission(widget.permission);
    }
  }
  /// 校驗權(quán)限
  void checkPermission(Permission permission) async {
    final status = await permission.status;
    if (status.isGranted) {
      _popPage();
      return;
    }
    // 還未申請權(quán)限或之前拒絕了權(quán)限(在 iOS 上為首次申請權(quán)限,拒絕后將變?yōu)?`永久拒絕權(quán)限`)
    if (status.isDenied) {
      showAlert(permission, msgList[0], msgList[3], _isGoSetting ? "前往應(yīng)用中心" : "確定");
    }
    // 權(quán)限已被永久拒絕
    if (status.isPermanentlyDenied) {
      _isGoSetting = true;
      showAlert(permission, msgList[2], msgList[3], _isGoSetting ? "前往應(yīng)用中心" : "確定");
    }
    // 擁有部分權(quán)限
    if (status.isLimited) {
      if (Platform.isIOS || Platform.isMacOS) _isGoSetting = true;
      showAlert(permission, msgList[1], msgList[3], _isGoSetting ? "前往應(yīng)用中心" : "確定");
    }
    // 擁有部分權(quán)限(僅限 iOS)
    if (status.isRestricted) {
      if (Platform.isIOS || Platform.isMacOS) _isGoSetting = true;
      showAlert(permission, msgList[1], msgList[3], _isGoSetting ? "前往應(yīng)用中心" : "確定");
    }
  }
  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);
              })
          ],
        );
      }
    );
  }
  /// 申請權(quán)限
  void requestPermisson(Permission permission) async {
    // 申請權(quán)限
    await permission.request();
    // 再次校驗
    checkPermission(permission);
  }
  @override
  void dispose() {
    WidgetsBinding.instance.removeObserver(this);
    super.dispose();
  }
  @override
  Widget build(BuildContext context) {
    return Container();
  }
  /// 退出應(yīng)用程序
  void _quitApp() {
    SystemChannels.platform.invokeMethod("SystemNavigator.pop");
  }
  /// 關(guān)閉整個權(quán)限申請頁面
  void _popDialogAndPage(BuildContext dialogContext) {
    _popDialog(dialogContext);
    _popPage();
  }
  /// 關(guān)閉彈窗
  void _popDialog(BuildContext dialogContext) {
    Navigator.of(dialogContext).pop();
  }
  /// 關(guān)閉透明頁面
  void _popPage() {
    Navigator.of(context).pop();
  }
}

AndroidManifest.xml

mediaflutter/example/android/app/src/main/AndroidManifest.xml

<uses-permission android:name="android.permission.CAMERA" />

在外部想要檢查及獲取權(quán)限的地方,調(diào)用permissionCheckAndRequest方法即可:

// 導(dǎo)入封裝好的文件
import 'package:xxxx/permission_request.dart';
// ...其它代碼
ElevatedButton(
  onPressed: () async {
    // 調(diào)用封裝好的權(quán)限檢查和請求方法
    bool result = await permissionCheckAndRequest(
                          context, 
                          Permission.notification,
                           "通知"
                        );
    if (result) print("已擁有該權(quán)限");
  },
  child: const Text("權(quán)限檢查")
)

到此這篇關(guān)于Flutter手機權(quán)限檢查與申請實現(xiàn)方法詳解的文章就介紹到這了,更多相關(guān)Flutter手機權(quán)限檢查內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Android自定義View實現(xiàn)打鉤動畫功能

    Android自定義View實現(xiàn)打鉤動畫功能

    本篇文章通過實例給大家分享了Android自定義View實現(xiàn)打鉤動畫功能的過程和代碼分享,有興趣需要的學(xué)習(xí)下吧。
    2017-12-12
  • Android仿微信QQ設(shè)置圖形頭像裁剪功能

    Android仿微信QQ設(shè)置圖形頭像裁剪功能

    最近在做畢業(yè)設(shè)計,想有一個功能和QQ一樣可以裁剪頭像并設(shè)置圓形頭像.圖片裁剪實現(xiàn)方式有兩種,一種是利用系統(tǒng)自帶的裁剪工具,一種是使用開源工具Cropper。本節(jié)就為大家?guī)砣绾问褂孟到y(tǒng)自帶的裁剪工具進(jìn)行圖片裁剪
    2016-10-10
  • android實現(xiàn)直播點贊飄心動畫效果

    android實現(xiàn)直播點贊飄心動畫效果

    這篇文章主要為大家詳細(xì)介紹了android實現(xiàn)直播點贊飄心動畫效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-03-03
  • Android實現(xiàn)自定義輪播圖片控件詳解

    Android實現(xiàn)自定義輪播圖片控件詳解

    這篇文章給大家主要介紹了Android實現(xiàn)自定義輪播圖片控件的詳細(xì)過程,文中通過實例代碼介紹的很詳細(xì),相信會對大家的理解和學(xué)習(xí)很有幫助,感興趣的朋友們下面來一起看看吧。
    2016-10-10
  • 務(wù)必掌握的Android十六進(jìn)制狀態(tài)管理最佳實踐

    務(wù)必掌握的Android十六進(jìn)制狀態(tài)管理最佳實踐

    這篇文章主要為大家介紹了務(wù)必掌握的Android十六進(jìn)制狀態(tài)管理最佳實踐,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • Android實現(xiàn)顯示電量的控件代碼

    Android實現(xiàn)顯示電量的控件代碼

    本文介紹了Android實現(xiàn)顯示電量的控件代碼,主要功能就是可以顯示電量,有需要的朋友可以來了解一下。
    2016-10-10
  • Android 自定義imageview實現(xiàn)圖片縮放實例詳解

    Android 自定義imageview實現(xiàn)圖片縮放實例詳解

    這篇文章主要介紹了Android 自定義imageview實現(xiàn)圖片縮放實例詳解的相關(guān)資料,需要的朋友可以參考下
    2017-04-04
  • android 判斷橫豎屏問題的詳解

    android 判斷橫豎屏問題的詳解

    本篇文章是對android中如何判斷橫豎屏的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-06-06
  • 如何讓安卓(Android)子控件超出父控件的范圍顯示

    如何讓安卓(Android)子控件超出父控件的范圍顯示

    下面我們來看一下關(guān)于安卓(Android)子控件超出父控件的范圍顯示出來的方法,希望這篇文章能夠幫助到各位安卓(Android)的開發(fā)者們,有碰到此問題的朋友可以進(jìn)來看看哦。
    2016-08-08
  • Android自定義簡單的頂部標(biāo)題欄

    Android自定義簡單的頂部標(biāo)題欄

    這篇文章主要為大家詳細(xì)介紹了Android自定義簡單的頂部標(biāo)題欄,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-11-11

最新評論