Android 12 未显示正确的权限请求屏幕

Mos*_*atz 5 android geolocation android-permissions

我的应用程序需要精确的用户位置,因为它要求用户位于特定位置才能执行任务。我已按照文档ACCESS_FINE_LOCATION获取同时请求两者和一起请求的正确方法ACCESS_COARSE_LOCATION

class LocationChooserFragment : BottomSheetDialogFragment() {
    // only relevant parts shown

    /**
     * A utility class that holds all the code for requesting location updates,
     * so that we can re-use it in multiple fragments.
     */
    private var locationTracker: LocationTracker? = null

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        // all the other initialization for viewbinding

        tryToGetLocationUpdates()
    }

    private fun tryToGetLocationUpdates() {
        val request = registerForActivityResult(ActivityResultContracts.RequestMultiplePermissions()) { permissions ->
            if (permissions[Manifest.permission.ACCESS_FINE_LOCATION] == true && permissions[Manifest.permission.ACCESS_FINE_LOCATION] == true) {
                requestLocationUpdates()
            }
        }

        when {
            ContextCompat.checkSelfPermission(requireContext(), Manifest.permission.ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED &&
            ContextCompat.checkSelfPermission(requireContext(), Manifest.permission.ACCESS_COARSE_LOCATION) == PackageManager.PERMISSION_GRANTED -> {
                Timber.i("Already have required permissions")
                requestLocationUpdates()
            }
            shouldShowRequestPermissionRationale(Manifest.permission.ACCESS_FINE_LOCATION) ||
            shouldShowRequestPermissionRationale(Manifest.permission.ACCESS_COARSE_LOCATION) -> {
                Timber.i("Showing permission rationale dialog")
                // A wrapper to show a dialog that explains what we need permissions for
                alertDialog(
                    title = getString(R.string.permissions),
                    msg = getString(R.string.why_we_need_location),
                    onClick = { yes ->
                        if (yes) {
                            request.launch(arrayOf(
                                Manifest.permission.ACCESS_FINE_LOCATION,
                                Manifest.permission.ACCESS_COARSE_LOCATION
                            ))
                        }
                    }
                )
            }
            else -> {
                Timber.i("Prompting for location permissions")
                request.launch(arrayOf(
                    Manifest.permission.ACCESS_FINE_LOCATION,
                    Manifest.permission.ACCESS_COARSE_LOCATION
                ))
            }
        }
    }

    private fun requestLocationUpdates() {
        if (locationProvider == null) {
            locationProvider = LocationProvider(requireContext())
        }
        locationProvider?.requestLocationUpdates()
    }
}
Run Code Online (Sandbox Code Playgroud)

我遇到的问题是我的用户没有收到允许精确定位的提示。

我期望从文档中看到图 3 中的提示:

图3

图 3.当您的应用在单个运行时请求中同时请求ACCESS_FINE_LOCATION和 时,会出现系统权限对话框。ACCESS_COARSE_LOCATION

相反,我从文档中看到图 2 中的提示:

图2

图 2.仅当您的应用程序请求时出现的系统权限对话框ACCESS_COARSE_LOCATION

此外,我希望当我已经拥有COARSE权限并再次请求FINE权限时COARSE,我应该看到“升级到精确权限”对话框,但我根本看不到任何提示。

我已经在 Pixel 3a(物理设备和模拟器)上进行了测试,并收到了其他用户在运行 Android 12 的各种 Pixel 和 Samsung Galaxy 设备上的报告。

compileSdk 30当我们使用andtargetSdkVersion 30时,问题并不存在build.gradle,但我们需要能够支持新功能和新版本的 Android(并且能够升级到仅支持使用较新的 SDK 版本进行构建的依赖项。

为什么权限对话框无法正确显示?

Mos*_*atz 11

将我的文件与构建的应用程序中生成的文件进行比较AndroidManifest.xml(使用 Android Studio 中的“分析 APK”功能),我发现生成的 APK 文件具有以下条目:

<uses-permission
    android:name="android.permission.ACCESS_FINE_LOCATION"
    android:maxSdkVersion="30" />
Run Code Online (Sandbox Code Playgroud)

显然,这maxSdkVersion正在阻止操作系统向我提供所需的权限。

grep在目录上使用app\build,我发现app\build\intermediates\manifest_merge_blame_file\appnameDebug\manifest-merger-blame-appname-debug-report.txt,其中包括这些行:

17    <uses-permission
17-->C:\path\to\my\appname\app\src\main\AndroidManifest.xml:12:2-76
18        android:name="android.permission.ACCESS_FINE_LOCATION"
18-->C:\path\to\my\appname\app\src\main\AndroidManifest.xml:12:19-73
19        android:maxSdkVersion="30" />
19-->[com.github.50ButtonsEach:flic2lib-android:1.3.1] C:\Users\moshe\.gradle\caches\transforms-3\dced3886509296f1029e8245af379a07\transformed\jetified-flic2lib-android-1.3.1\AndroidManifest.xml:17:9-35
Run Code Online (Sandbox Code Playgroud)

事实证明,该库的开发人员最初用于ACCESS_FINE_LOCATION连接附近的蓝牙设备,由于 Android 12 对此具有单独的权限,因此他们不再需要它,因此他们添加了最大 SDK 版本。

我将该tools:remove="android:maxSdkVersion"属性添加到ACCESS_FINE_LOCATION <uses-permission />元素中,现在我的权限可以正常工作。