This repo presents some conventions and code style guidelines, with Android application development in mind. Following these will help you achieve readable and easy-to-maintain code.
π‘ Before we dive into the recommendations, it is important to say this:
If you are part of a team, take the time to define a set of rules and conventions with your colleagues before starting development. This will make your lives much easier in the future β both for you and for future developers.
Based on my experience, hereβs what I usually recommend:
- Acronyms and abbreviations can confuse developers,
- very descriptive names, however, do not.
- Generic names can also confuse developers.
- Do not be afraid to use long names.
- If you refactor, change, or update code, be sure to check that naming is still valid.
- Functions should use verbs and clearly describe the action they perform.
- Naming classes: keep it simple and descriptive.
- Avoid repetition in hierarchies and namespaces.
- Keep one tense. Do not mix tenses for functions serving the same purpose (e.g., renameAccountClicked,updateAccountClick).
- If possible, match the names with the business requirements.
some examples:
// avoid repetition in hierarchies and namespaces
sealed class AuthorizedLogin {
 β data object AuthorizedLoginSucceded
 β
 data object Succeded
 β data object AuthorizedLoginFailed
 β
 data object Failed
 β data object AuthorizedLoginInProgress
 β
 data object InProgress
}
// so later in the code you get this:
return AuthorizedLogin.Succeded 
// instead of this:
return AuthorizedLogin.AuthorizedLoginSucceded
// Acronyms and abbreviations can confuse developers,very descriptive names, however, do not.
β fun getData() { ... }
β
 fun getAccountData { ... }
β fun updAccount(account: Account) { ... }
β
 fun updateAccount(account: Account) { ... }
β fun updateAccount(newAccountName: String) { ... }
β
 fun updateAccountName(newAccountName: String) { ... }
- First, implement interfaces and abstract classes (if any). If thse functions are empty/not used place them at the bottom of the class.
- Second, place public functions at the top of the class.
- Finally, place private functions at the bottom of the class, following the same order in which they are used by the public functions. Example:
class AccountViewModel : SomeInterface {
 init {
 sendLogEvent()
 }
 override fun functionFromInterface() { ... }
 fun onAccountClick() {
 ...
 updateAccount()
 }
 private fun updateAccount() { ... }
 private fun sendLogEvent() { ... }
}
If there are many (usually more than two) chained calls, then wrap them and don't write them all on one line. It can help with debugging.
β val anchor = owner?.firstChild!!.siblings(forward = true).dropWhile { it is PsiComment || it is PsiWhiteSpace } 
β
 val anchor = owner
 ?.firstChild!!
 .siblings(forward = true)
 .dropWhile { request->
 request is PsiComment || request is PsiWhiteSpace 
 }
It's not always clear what arguments are passed into a function. Named arguments can solve this. In addition, when there are more than two arguments, use list wrapping.
// named arguments & parameter list wraping
drawSquare(
	x = 10,
	y = 10,
	width = 100,
	height = 100,
	fill = true
)
5. Avoid all usage of the implicit argument name it inside multi-line expressions or multi-line lambdas
 One of the most important conventions. Unless it is absolutely clear from context try to avoid implicit it.
- https://kotlinlang.org/docs/coding-conventions.html
- https://developer.android.com/topic/architecture/recommendations#naming-conventions
- https://developer.android.com/kotlin/style-guide
I usually use the suffix Screen for each new screen, Content for the main content of the screen, and View for smaller inner composables.
@Composable
fun AccountScreen() {
 val state by viewModel.state
 AccountContent() { ... }
}
@Composable
private fun AccountContent() {
 Column {
	Text(text = "AccountName")
 Image(src = "AccountPhoto")
 AccountDetailsView()
 }
 
}
@Composable
private fun AccountDetailsView() { ... }