我的小部件调用Activity
范围之外的安全权限。是否可以在之外请求Android MActivity
的权限?
6 回答
我找到了一个似乎工作正常的解决方法。诀窍是创建一个透明的活动,该活动仅用于请求权限并在之后立即完成。当然,您仍然需要一个上下文,但它不一定是一个活动。活动可以通过广播返回结果(授予或拒绝)(因为startActivtyForResult
在活动之外不可能)。
您可以使用此活动:
import android.content.Intent
import android.content.pm.PackageManager
import android.os.Bundle
import android.support.v4.app.ActivityCompat
import android.support.v7.app.AppCompatActivity
internal const val PERMISSIONS_KEY = "permissions"
internal const val ACTION_PERMISSIONS_GRANTED = "GetPermissionsActivity.permissions_granted"
internal const val ACTION_PERMISSIONS_DENIED = "GetPermissionsActivity.permissions_denied"
class GetPermissionsActivity: AppCompatActivity() {
private val permissionRequestCode = 0
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
ActivityCompat.requestPermissions(
this,
intent.getStringArrayExtra(PERMISSIONS_KEY),
permissionRequestCode
)
}
override fun onRequestPermissionsResult(
requestCode: Int,
permissions: Array<out String>,
grantResults: IntArray
) {
if (requestCode == permissionRequestCode) {
if ((grantResults.isNotEmpty() && grantResults[0] == PackageManager.PERMISSION_GRANTED)) {
sendBroadcast(Intent(ACTION_PERMISSIONS_GRANTED))
} else {
sendBroadcast(Intent(ACTION_PERMISSIONS_DENIED))
}
finish()
} else {
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
}
}
}
而这种风格的活动
<style name="Theme.Transparent" parent="Theme.AppCompat">
<item name="android:windowIsTranslucent">true</item>
<item name="android:windowBackground">@android:color/transparent</item>
<item name="android:windowContentOverlay">@null</item>
<item name="android:windowNoTitle">true</item>
<item name="android:windowIsFloating">true</item>
<item name="android:backgroundDimEnabled">false</item>
<item name="android:windowAnimationStyle">@null</item>
</style>
在清单中:
<activity android:name="GetPermissionsActivity" android:theme="@style/Theme.Transparent" />
然后像这样使用它(需要上下文)
class SomeClass : BroadcastReceiver() {
private fun someFunction(context: Context) {
val intentFilter = IntentFilter()
intentFilter.addAction(ACTION_PERMISSIONS_GRANTED)
intentFilter.addAction(ACTION_PERMISSIONS_DENIED)
context.registerReceiver(this, intentFilter)
val intent = Intent(context, GetPermissionsActivity::class.java)
intent.putExtra(PERMISSIONS_KEY, arrayOf(<your permissions>))
context.startActivity(intent)
}
override fun onReceive(context: Context, intent: Intent) {
when {
intent.action == ACTION_PERMISSIONS_GRANTED -> {
context.unregisterReceiver(this)
onPermissionsGranted()
}
intent.action == ACTION_PERMISSIONS_DENIED -> {
context.unregisterReceiver(this)
onPermissionsDenied()
}
else -> super.onReceive(context, intent)
}
}
private fun onPermissionsGranted() {
// ...
}
private fun onPermissionsDenied() {
// ...
}
}
不,这是不可能的。您可以做的是发送通知,用户可以在其中点击,然后使用活动来请求/管理权限(可能带有对话框主题)。
您可以使用Easy Permissions库。
Android 要求这些请求来自Activity
. 使用 Easy Permissions 这不再是问题,您可以从任何地方请求许可,只要您提供Context
. 此外,如果您请求的权限已被授予,则不会提示用户。
充分披露,我们公司管理和开发这个免费使用的图书馆。话虽如此,我们相信它是一个有用的工具,否则我们不会分享它。
您只能向 Activity 或 Fragment 请求权限。
在 Activity 或 Fragment 中找出您认为 App 需要权限的点,然后调用 requestPermission 方法。发送通知将不起作用,因为您希望您的代码处理直到您获得请求的权限,然后从 onRequestPermissionResult() 方法恢复您的功能。
我认为可以在 Activity 之外请求权限,只要您使用该方法
ActivityCompat.requestPermissions (Activity 活动, String[] 权限, int requestCode)
从支持库中,并将 Activity 作为方法的参数传递。
例如:
ActivityCompat.requestPermissions(targetActivity, new String[] {Manifest.permission.CAMERA}, PERMISSION_REQUEST_CODE);
其中 targetActivity 是应该实现该方法的 Activity:
onRequestPermissionsResult (int requestCode, String[] 权限, int[] grantResults)
这就是处理 de 权限请求结果的方法。
我正在创建一个需要在许多活动中检查权限的应用程序,因此我创建了一个可以在应用程序中全局使用的静态类。它奏效了。这对我有用。
我创建了一个方法来检查像这样的不同类中的权限请求。
public class CheckForPermissions implements ActivityCompat.OnRequestPermissionsResultCallback {
private static final int MY_PERMISSIONS_REQUEST_READ_LOCATION = 1;
public static void checkForLocationPermissions(Activity context){
if(ContextCompat.checkSelfPermission(context, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED
|| ContextCompat.checkSelfPermission(context, Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED){
ActivityCompat.requestPermissions(context,
new String[]{Manifest.permission.ACCESS_FINE_LOCATION},
MY_PERMISSIONS_REQUEST_READ_LOCATION);
}
}}
在我的活动中,我调用了这样的方法
CheckForPermissions.checkForLocationPermissions(this);