Firemonkey android read_phone_state运行时权限要求获取IMEI

Dej*_*jan 9 delphi permissions android firemonkey delphi-10.2-tokyo

如何在运行时获取read_phone_state权限以获取IMEI号?

  if not HasPermission('android.permission.READ_PHONE_STATE') then
     begin

      //ASK AND GET PERMISSION ?

     end;


function TForm1.HasPermission(const Permission: string): Boolean;
begin
  //Permissions listed at http://d.android.com/reference/android/Manifest.permission.html
{$IF RTLVersion >= 30}
  Result := TAndroidHelper.Context.checkCallingOrSelfPermission(
{$ELSE}
  Result := SharedActivityContext.checkCallingOrSelfPermission(
{$ENDIF}
    StringToJString(Permission)) =
    TJPackageManager.JavaClass.PERMISSION_GRANTED;


end;
Run Code Online (Sandbox Code Playgroud)

kri*_*son 3

编辑:抱歉我没有在 FireMonkey 上做更多的功课。这就是我把注意力集中在不属于它的话题上所得到的结果。我添加此内容是为了使我的答案更值得赏金。

如果您可以将应用清单上的 限制targetSdk为 22 (5.1 Lollipop),那么用户将必须授予安装权限,因此HasPermission永远不应返回 false。(不知道 FireMonkey 是如何工作的)。

如果您想使用 Marshmallow+ 中的动态权限功能,以下是我从此页面收集的一些信息:

您需要有权访问Activity回调方法onRequestPermissionsResult。以下是您必须跨越的所有障碍:

  • 使用开源工具 Dex2Jar 将 Androidclasses.dex文件从 Delphi 转换回 Java,以便您可以针对该类进行编译FMXNativeActivity
  • FMXNativeActivity在 Java 中编写一个定义native方法的子类(让我们调用它onRequestPermissionsResultNative并覆盖该onRequestPermissionsResult方法以调用本机方法。
  • 运行javac以获取包含子类的 .class 文件
  • 运行jar将.class文件放入.jar文件中
  • 运行dx.bat将您的 .jar 文件转换为 Android .dex 文件
  • 运行DexMerger将你的.dex文件合并到Delphi的classes.dex文件中
  • 现在剩下要做的就是编写一些棘手的 Delphi 代码来定义您的onRequestPermissionsResultNative方法并将其注册到 JNI 环境。哦,不要忘记在本机方法中切换到正确的线程。

我引用的链接显示了如何使用onActivityResult. 您必须针对其他方法调整这些步骤。

我什至没有讨论如何处理操作系统暂停您的应用程序以请求用户许可并在之后恢复它。

让这成为所有人的一个教训:不要相信跨平台工具;不要相信跨平台工具。你会失望的。


我使用 Java 而不是 Delphi,所以你必须在这里进行一些推断。

和你一样,我必须获取 IMEI 号码,系统对话框会询问用户类似以下内容:“允许应用拨打和管理电话吗?” 我需要向用户解释该应用程序只是获取设备 ID,不会拨打或管理电话。所以

  • 检查您是否有权限
  • 如果您没有权限,请检查是否应该显示说明
  • 如果不需要出示说明,则启动权限请求操作

我应该提到这一点shouldShowRequestPermissionRationale,并且requestPermissions是类中的方法Activity

    private static final int READ_PHONE_STATE_PERMISSIONS_REQUEST = 2;

    private boolean mHasReadRationale;

    void doPermissionsStuff() {
        // version checking code omitted, this block runs for marshmallow and later
        if (checkSelfPermission(Manifest.permission.READ_PHONE_STATE) == PackageManager.PERMISSION_GRANTED) {
            // do the operation that needs the permission here
        } else {
            // the flag indicates if the rationale dialog has already been displayed
            if (! mHasReadRationale && shouldShowRequestPermissionRationale(Manifest.permission.READ_PHONE_STATE)) {
                // pop a dialog that explains what's going on to the user
            } else {
                requestPermissions(new String[] {Manifest.permission.READ_PHONE_STATE}, READ_PHONE_STATE_PERMISSIONS_REQUEST);
            }
        }
    }
Run Code Online (Sandbox Code Playgroud)

在此对话框的肯定按钮中(即用户想要继续)将标志设置mHasReadRationale为 true 并调用doPermissionsStuff再次调用。(对于取消,我将用户带回到上一个屏幕。)

为了获得requestPermissions操作的结果,您需要重写 的Activity方法onRequestPermissionsResult

private boolean mPermissionDenied;

@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {

    switch (requestCode) {
    case READ_PHONE_STATE_PERMISSIONS_REQUEST:
        // I'm only checking for one permission, so I make assumptions here
        if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
            // you can do the operation that needs the permission now
        } else {
            mPermissionDenied = true;  // set a flag for checking later
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

显然,当系统请求用户许可时,它会停止您的应用程序,因此您此时无法显示 UI 来告诉用户您没有权限。因此,我设置了一个标志,当应用程序恢复时,告诉用户该应用程序没有执行该操作的权限。

@Override
protected void onResumeFragments() {
    super.onResumeFragments();
    if (mPermissionDenied) {
        // show dialog to the user that the app can't do the operation because it doesn't have permission
        mPermissionDenied = false;
    }
 }
Run Code Online (Sandbox Code Playgroud)

这是一个示例流程:

  • 用户想要免费试用,应用程序需要获取 IMEI,这样他们就不能一遍又一遍地获得免费试用,天哪。应用程序调用doPermissionsStuff().
  • 应用程序调用checkSelfPermission()并确定尚未授予权限
  • 应用程序调用shouldShowRequestPermissionRationale(). 在我的经验中,shouldShowRequestPermissionRationale()只有在用户拒绝一次权限后才返回 true。因此,您还没有向用户显示基本原理 UI。
  • 应用程序调用requestPermissions()
  • 系统将询问用户“允许应用程序拨打和管理电话吗?”
  • 用户认为这太可怕了,并按下“否”按钮。
  • onRequestPermissionsResult()被调用并返回拒绝结果并mPermissionDenied设置。
  • onResumeFragments()被调用并向用户显示一个对话框,表明他们无法获得免费试用,因为该应用程序没有权限。
  • 用户决定再试一次。doPermissionsStuff()叫做。
  • 应用程序调用checkSelfPermission()并(再次)确定尚未授予权限
  • 应用程序调用shouldShowRequestPermissionRationale(). 这次它返回true。
  • 应用程序会向用户显示一条平静且舒缓的消息,表明不,我们不会接管您的手机,我们只是想要该死的 IMEI 号码,仅此而已,如果您不允许该应用程序访问 IMEI,您无法获得免费试用。我必须在某个地方划清界限。
  • 用户按下继续,因此mHasReadRationale标志设置为 true 并且doPermissionsStuff()再次调用方法。
  • 应用程序调用checkSelfPermission()- 你猜怎么着?尚未授予该权限
  • 由于设置了该标志,用户不会获得基本原理 UI。
  • 应用程序调用requestPermissions()
  • 系统将询问用户“允许应用程序拨打和管理电话吗?”
  • 用户听天由命并按“是”。
  • onRequestPermissionsResult()被调用并获得已授予的结果,免费试用注册将继续进行。

您还应该查看 Google 的示例代码:https://developer.android.com/samples/RuntimePermissions/index.html