Just add this to your settings.gradle:
dependencyResolutionManagement {
repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)
repositories {
google()
mavenCentral()
maven { url 'https://jitpack.io' }
}
}
Then, in your build.gradle:
implementation 'com.github.ygorluizfrazao.composed-permissions:core:VERSION_NAME'
implementation 'com.github.ygorluizfrazao.composed-permissions:ext:VERSION_NAME'
A Convenience library aimed to help dealing with Android Jetpack Compose runtime permission system in a functional intuitive way to avoid extra unecessary complexity and coupling.
It is always a pain to deal with permissions in android, thats specially true in compose, with states, recomposition and all. Suddenly, your viewmodel logic will contain a bunch of extra code, flow controls, conditionals, etc. Just to deal with a platform specific logic. I believe ViewModels exist to proper translate you business logic to the platform UI and vice versa, not to deal with very very speific stuff.
Whenever you want:
- Worry about your requirements, not Android's.
- Less coupling.
- Less boilerplate.
- Dealing with
shouldShowRequestPermissionRationale
. - Atomize your permissions requests.
Just do what you would do in your code, compose aware in a lambda block.
The main entry point of this library is the Composable Lambda WithPermission
.
@Composable
fun WithPermission(
userDrivenAskingStrategy: UserDrivenAskingStrategy<Map<String, Boolean>>,
initialStateContent: @Composable () -> Unit,
rationalePrompt: @Composable (state: PermissionFlowStateEnum, permissionsStatus: Map<String, Boolean>, callMeWhen: PermissionsAskingStrategyInteraction) -> Unit,
terminalStateContent: @Composable (state: PermissionFlowStateEnum, permissionsStatus: Map<String, Boolean>) -> Unit
)
userDrivenAskingStrategy
: Contract that defines how the process of permission handling will occur.initialStateContent
: How is your content beforeuserDrivenAskingStrategy
reaches a terminal state?rationalePrompt
: How is your content when Android denies the permission?state: PermissionFlowStateEnum
: Tell at which state the underlyinguserDrivenAskingStrategy
is. it can be:NOT_STARTED
: The strategy is in its initial state.STARTED
: Started processing.DENIED_BY_SYSTEM
: The permission request was denied by Android (Who would have guessed, huh).APP_PROMPT
: Waiting for app handled prompt (rationalePrompt
).TERMINAL_GRANTED
: All permissions were granted.TERMINAL_DENIED
: Not all permissions were granted.
permissionsStatus: Map<String, Boolean>
: Current status of the required permissions.callMeWhen: PermissionsAskingStrategyInteraction
: Since you will handle therationalePrompt
you have to inform the state machine about the user interaction, the, you call its methods in the apropriate case.
terminalStateContent
: How is your content after the process finishes? (Can succed or fail, you will know about this in thepermissionsStatus
parameter)- It's params have the same logic as stated in the previous item.
Wait, i don't want to deal with this rationalePrompt
stuff, it is still boilerplate and i feel really lazy now...
Then, my dear friend, import the ext
dependency too. It has a function that abstracts this bit of code inside a Materialv3 AlertDialog
, you will just have to pass some params to create it.
This.
@Composable
fun WithPermission(
userDrivenAskingStrategy: UserDrivenAskingStrategy<Map<String, Boolean>>,
permissionDialogProperties: PermissionDialogProperties = createPermissionDialogProperties(),
initialStateContent: @Composable () -> Unit,
terminalStateContent: @Composable (state: PermissionFlowStateEnum, permissionsStatus: Map<String, Boolean>) -> Unit
)
And that.
data class PermissionDialogProperties(
val shape: Shape,
val dialogPaddingValues: PaddingValues,
val verticalArrangement: Arrangement.HorizontalOrVertical,
val title: (state: PermissionFlowStateEnum, permissionsStatus: Map<String, Boolean>) -> String,
val titleStyle: TextStyle,
val message: (state: PermissionFlowStateEnum, permissionsStatus: Map<String, Boolean>) -> String,
val messageStyle: TextStyle,
val grantButtonText: (state: PermissionFlowStateEnum, permissionsStatus: Map<String, Boolean>) -> String,
val denyButtonText: (state: PermissionFlowStateEnum, permissionsStatus: Map<String, Boolean>) -> String,
val onGrantButtonClicked: (state: PermissionFlowStateEnum, permissionsStatus: Map<String, Boolean>) -> Unit,
val onDenyButtonClicked: (state: PermissionFlowStateEnum, permissionsStatus: Map<String, Boolean>) -> Unit
)
So you can localize and theme your dialog apropriately, but, if you don't want the headache, you can use:
createPermissionDialogProperties()
and change only what you want to change.
IMPORTANT NOTE
You should pass at least onGrantButtonClicked
to do something in your app when the user wants to grante the permission manually, an example would be to open the app settings, like this:
fun Context.goToAppSettings() {
Intent(
Settings.ACTION_APPLICATION_DETAILS_SETTINGS,
Uri.fromParts("package", packageName, null)
).also { startActivity(it) }
}
permissionDialogProperties = createPermissionDialogProperties(
onGrantButtonClicked = { _, _ ->
context.goToAppSettings()
clicked = false
},
)
Here is an app of notes which needs permission to record audio and location in very specific place.
- Create and remember your
UserDrivenAskingStrategy
:
var canStart by rememberSaveable {
mutableStateOf(false)
}
val userDrivenAskingStrategy =
rememberUserDrivenAskingStrategy(
type = AskingStrategy.STOP_ASKING_ON_USER_DENIAL,
permissions = listOf(
Manifest.permission.RECORD_AUDIO,
Manifest.permission.ACCESS_COARSE_LOCATION,
Manifest.permission.ACCESS_FINE_LOCATION
),
canStart = { canStart }
)
IMPORTANT NOTE
The canStart: () -> Boolean
param tells the UserDrivenAskingStrategy
when it should start asking. It is {true}
by default, which means, it will start the process as soon as the composition happens. In the above example, we want to control the start of the flow, so, we pass our own implementation.
- Call
WithPermission
"with" the controls that depends on specific permissions.
NotesList(...) {
Row(...) {
WithPermission(
userDrivenAskingStrategy = userDrivenAskingStrategy,
permissionDialogProperties = createPermissionDialogProperties(
onGrantButtonClicked = { _, _ ->
context.goToAppSettings()
canStart = false
},
),
initialStateContent = {
IconButton(onClick = { canStart = true }) {
IconResource.fromImageVector(
Icons.Default.Warning,
""
).ComposeIcon()
}
}) { _, permissionMap ->
val permissionsMapState = remember {
mutableStateOf(permissionMap)
}
val grantedPermissions = remember {
derivedStateOf {
permissionsMapState.value.filter { (_, granted) -> granted }
}
}
val deniedPermissions = remember {
derivedStateOf {
permissionsMapState.value.filter { (_, granted) -> !granted }
}
}
val coroutineScope = rememberCoroutineScope()
var showPopup by remember {
mutableStateOf(false)
}
if (deniedPermissions.value.isEmpty()) {
IconButton(onClick = {
showPopup = true
coroutineScope.launch {
Toast.makeText(
context,
"Call a fancy ViewModel Action!",
Toast.LENGTH_SHORT
)
.show()
}
}) {
IconResource.fromImageVector(
Icons.Default.CheckCircle,
""
).ComposeIcon()
}
} else {
IconButton(onClick = {
showPopup = true
coroutineScope.launch {
Toast.makeText(
context,
"Granted: ${grantedPermissions.value}\nDenied: ${deniedPermissions.value}",
Toast.LENGTH_SHORT
)
.show()
}
}) {
IconResource.fromImageVector(
Icons.Default.Cancel,
""
).ComposeIcon()
}
}
if (showPopup) {
Popup(
properties = PopupProperties(
dismissOnBackPress = true,
dismissOnClickOutside = true,
focusable = true
),
onDismissRequest = { showPopup = false },
popupPositionProvider = object : PopupPositionProvider {
override fun calculatePosition(
anchorBounds: IntRect,
windowSize: IntSize,
layoutDirection: LayoutDirection,
popupContentSize: IntSize
): IntOffset {
return IntOffset(
(windowSize.width - popupContentSize.width) / 2,
(windowSize.height - popupContentSize.height) / 2
)
}
}) {
Surface(
border = BorderStroke(1.dp, LocalContentColor.current),
shape = MaterialTheme.shapes.medium
) {
Column(
modifier = Modifier.padding(MaterialTheme.spacing.medium),
verticalArrangement = Arrangement.spacedBy(MaterialTheme.spacing.medium)
) {
Text(
text = "Granted Permissions: \n${
grantedPermissions.value.map { it.key.toHumanLanguage() + "\n" }
.joinToString(separator = "")
}"
)
Divider()
Text(
text = "Denied Permissions: \n${
deniedPermissions.value.map { it.key.toHumanLanguage() + "\n" }
.joinToString(separator = "")
}"
)
}
}
}
}
}
...
}
}
And that's it, everything will be automated for you. If this is not enough for you, check the other classes and functions in the package, almost all of them have their KDOC's documented.
It uses a finite state machine to track at which point of a process the user is. Inside these state machines we have a StateFlow variable. When this stateflow variable updates, the UI recompose. Also, it resolves the state every time lifecycle is resumed.
This is very initial and little test was done, please, help me improve it. Also, you can whatever you want with it, it's free to use, modify, copy, paste and distribute, just remember to give the credits.
If you want, can, and feels like it, you can