我有一个使用 WebView 的应用程序,其中网站可能会请求使用设备的地理位置。我的主要活动有以下内容:
public void onGeolocationPermissionsShowPrompt(final String origin, final GeolocationPermissions.Callback callback) {
m_geolocationCallback = null;
m_geolocationOrigin = null;
// If we don't have location permissions, we must request them first
if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
// Show rationale if necessary
if (ActivityCompat.shouldShowRequestPermissionRationale(MainActivity.this, Manifest.permission.ACCESS_FINE_LOCATION)) {
new AlertDialog.Builder(MainActivity.this)
.setMessage(getString(R.string.permission_location_rationale, origin))
.setNeutralButton(android.R.string.ok, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialogInterface, int i) {
m_geolocationOrigin = origin;
m_geolocationCallback = callback;
ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, PERMISSIONS_REQUEST_ACCESS_FINE_LOCATION);
}
})
.show();
}
else {
m_geolocationOrigin = origin;
m_geolocationCallback = callback;
ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, PERMISSIONS_REQUEST_ACCESS_FINE_LOCATION);
}
}
// Otherwise just tell webview that permission has been granted
else {
callback.invoke(origin, true, false);
}
}
处理权限请求结果,
MainActivity
还有以下:
@Override
public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
switch (requestCode) {
case PERMISSIONS_REQUEST_ACCESS_FINE_LOCATION:
// Permission granted
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
if (m_geolocationCallback != null) {
m_geolocationCallback.invoke(m_geolocationOrigin, true, false);
}
}
// Permission denied
else {
// In this case, the user checked "Don't ask again" and denied permission
if (Build.VERSION.SDK_INT >= 23 && !shouldShowRequestPermissionRationale(permissions[0])) {
Toast.makeText(this, R.string.permission_location_denied, Toast.LENGTH_LONG).show();
}
if (m_geolocationCallback != null) {
m_geolocationCallback.invoke(m_geolocationOrigin, false, false);
}
}
return;
}
}
我遇到的问题是,每当一个网站请求使用地理定位并且我拒绝它(导致
invoke(origin, false false)
被调用)时,onGeolocationPermissionsShowPrompt
方法就会立即被调用,我就陷入了无休止拒绝许可的循环中到应用程序。
如果我在权限请求中选中“不再询问”,则效果很好,但我希望能够在稍后的时间再次请求用户权限。
我尝试通过将拒绝回调调用更改为来解决此问题
m_geolocationCallback.invoke(m_geolocationOrigin, false, true)
,但这并没有达到预期的效果。这将防止停止对 onGeolocationPermissionsShowPrompt
的连续调用,但这也意味着即使稍后启用权限,给定的源也无法再使用地理位置,而且我发现无法撤消此操作(即使通过卸载/重新安装应用程序)。
关于如何解决这个问题有什么想法吗?
这看起来像是 Android Webview 中的一个错误。如果在 grant=false 的情况下异步调用
callback.invoke
,则会发生无限循环,但如果在 onGeolocationPermissionsShowPrompt
内部同步调用,则不会发生无限循环。
我的解决方法是在用户拒绝地理定位权限时节省时间。如果在短时间内再次调用
onGeolocationPermissionsShowPrompt
并且 checkSelfPermission
未返回 PERMISSION_GRANTED,则运行 callback.invoke(origin, false, false)
并且不重新提示用户。
我们可以像下面这样修复它:
// set param 'retain' true, to prevent permission request continuously
callback?.invoke(origin, isGranted, true)
if (!isGranted) {
// clear permission state, so the user can request permission again
Handler().postDelayed({ GeolocationPermissions.getInstance().clear(origin) }, 500)
}