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.当您的应用在单个运行时请求中同时请求
ACCESS_FINE_LOCATION
和 时,会出现系统权限对话框。ACCESS_COARSE_LOCATION
相反,我从文档中看到图 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 />
元素中,现在我的权限可以正常工作。
归档时间: |
|
查看次数: |
5282 次 |
最近记录: |