Every android app runs on its own sandbox. If in needs to access any recourse outside its sandbox, it should request permission from the system. There are few permissions that are automatically granted and there are few permissions that would require user to grant permission. Eg: location,contacts, sms etc.,
All permissions that your app requires should be declared in the manifest file. Devices running Android 5.1(API level 22) and below, would request permission when the user installs the app and devices running Android 6 (API level 23) and above will request permissions when that app tries to use the resource which needs the permission i.e runtime permissions
So lets try a runtime permission sample program.
Step1: Create an android project with an Activity (PermissionActivity.java) and a layout file (activity_permission.xml)
Stept2: Now declare the required permission in the manifest file. This sample program would need SEND_SMS permission and your manifest file should look similar to the one below
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?xml version="1.0" encoding="utf-8"?> | |
<manifest xmlns:android="http://schemas.android.com/apk/res/android" | |
package="androidbite.runtimepermission"> | |
<uses-permission android:name="android.permission.SEND_SMS" /> | |
<application | |
android:allowBackup="true" | |
android:icon="@mipmap/ic_launcher" | |
android:label="@string/app_name" | |
android:supportsRtl="true" | |
android:theme="@style/AppTheme"> | |
<activity android:name=".PermissionActivity"> | |
<intent-filter> | |
<action android:name="android.intent.action.MAIN" /> | |
<category android:name="android.intent.category.LAUNCHER" /> | |
</intent-filter> | |
</activity> | |
</application> | |
</manifest> |
Step3: Now lets design the screen with a TextView and a Button. The textview is used to display the current status of the permission and the button is used to request permission if not granted and if granted proceed with the functionality with the acquired permission
activity_permission.xml:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?xml version="1.0" encoding="utf-8"?> | |
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" | |
xmlns:tools="http://schemas.android.com/tools" | |
android:layout_width="match_parent" | |
android:layout_height="match_parent" | |
android:paddingBottom="@dimen/activity_vertical_margin" | |
android:paddingLeft="@dimen/activity_horizontal_margin" | |
android:paddingRight="@dimen/activity_horizontal_margin" | |
android:paddingTop="@dimen/activity_vertical_margin" | |
tools:context="androidbite.runtimepermission.PermissionActivity"> | |
<TextView | |
android:id="@+id/tvStatus" | |
android:layout_width="wrap_content" | |
android:layout_height="wrap_content" | |
android:layout_alignParentTop="true" | |
android:layout_centerHorizontal="true" | |
android:layout_marginTop="70dp" /> | |
<Button | |
android:id="@+id/bPermission" | |
android:layout_width="wrap_content" | |
android:layout_height="wrap_content" | |
android:text="Send SMS" | |
android:layout_below="@+id/tvStatus" | |
android:layout_centerHorizontal="true" | |
android:layout_marginTop="73dp" /> | |
</RelativeLayout> |
Step4: Now in the Activity, bind the UI components (TextView & Button). On click of the button request for SMS permission and proceed with the permission acquired.
PermissionActivity.java:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package androidbite.runtimepermission; | |
import android.Manifest; | |
import android.content.pm.PackageManager; | |
import android.os.Build; | |
import android.os.Bundle; | |
import android.support.annotation.NonNull; | |
import android.support.v4.app.ActivityCompat; | |
import android.support.v4.content.ContextCompat; | |
import android.support.v7.app.AppCompatActivity; | |
import android.view.View; | |
import android.widget.Button; | |
import android.widget.TextView; | |
import android.widget.Toast; | |
public class PermissionActivity extends AppCompatActivity { | |
Button bPermission; | |
TextView tvStatus; | |
private final int REQUEST_PERMISSION_SEND_SMS = 10000; | |
@Override | |
protected void onCreate(Bundle savedInstanceState) { | |
super.onCreate(savedInstanceState); | |
setContentView(R.layout.activity_permission); | |
tvStatus = (TextView) findViewById(R.id.tvStatus); | |
bPermission = (Button) findViewById(R.id.bPermission); | |
//Getting the permission status | |
int result = ContextCompat.checkSelfPermission(this, Manifest.permission.SEND_SMS); | |
tvStatus.setText((result == 0) ? "Granted" : "Denied"); | |
bPermission.setOnClickListener(new View.OnClickListener() { | |
@Override | |
public void onClick(View view) { | |
sendSMSFromApp(); | |
} | |
}); | |
} | |
public void sendSMSFromApp() { | |
if (Build.VERSION.SDK_INT >= 23) { | |
if (checkSelfPermission(android.Manifest.permission.SEND_SMS) == PackageManager.PERMISSION_GRANTED) { | |
// Do Someting with the granted permission; | |
Toast.makeText(PermissionActivity.this, "Permission Granted", Toast.LENGTH_SHORT).show(); | |
} else { | |
if (shouldShowRequestPermissionRationale(android.Manifest.permission.SEND_SMS)) { | |
ActivityCompat.requestPermissions(PermissionActivity.this, new String[]{android.Manifest.permission.SEND_SMS}, REQUEST_PERMISSION_SEND_SMS); | |
} else { | |
// Explain the user why you need the permission, May be show a dialog asking user to allow permission from settings | |
Toast.makeText(PermissionActivity.this, "Permission Denied with dont ask again", Toast.LENGTH_SHORT).show(); | |
} | |
} | |
} else { | |
// Permission is granted on app installation for pre Android M versions | |
Toast.makeText(PermissionActivity.this, "Permission Granted", Toast.LENGTH_SHORT).show(); | |
} | |
} | |
@Override | |
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { | |
super.onRequestPermissionsResult(requestCode, permissions, grantResults); | |
tvStatus.setText((grantResults[0] == 0) ? "Granted" : "Denied"); | |
Toast.makeText(PermissionActivity.this, permissions[0] + "=" + tvStatus.getText() , Toast.LENGTH_SHORT).show(); | |
} | |
} |
On click of the button, we would check if the API version is 23 & above then we should check if the app has the permission or not using checkSelfPermission method. If not request the user for the permission at runtime using ActivityCompat.requestPermissions.
During the permission request user may select Never ask again option and deny the permission. In such case, use the shouldShowRequestPremissionRationale which would be false, to determine whether to show an alternative explanation to user and direct him to setting menu or not.
For all actions performed by the user in the runtime permission request dialog, we would receive a callback in the onRequestPermissionResult method.
Step5: Run the app in an emulator or device with Android API level 23 or above. Click on the Send SMS button. System would prompt a dialog requesting a runtime permission to send sms from you app.
Output: