Clean Empty Jetpack Compose App Template

vtsen

Vincent Tsen

Posted on May 28, 2022

Clean Empty Jetpack Compose App Template

Simple and clean empty Jetpack Compose app example which doesn't have themes.xml and colors.xml.

For pure Jetpack Compose app, technically we don't need the themes.xml and colors.xml anymore, which are meant for Android View system.

However, I have seen a lot of code out there still have these themes.xml and colors.xml defined. Personally, I would remove them to avoid duplication or confusion.

1. Remove themes.xml and colors.xml

Removing themes.xml and colors.xml can cause compilation error due to it is still being referenced from AndroidManifest.xml. You need to modify the AndroidManifest.xml

  • Remove android:theme="@style/Theme.EmptyComposeApp" in both <application> and<activity>
<application
    ...
    android:theme="@style/Theme.EmptyComposeApp">
    <activity
        ...
        android:theme="@style/Theme.EmptyComposeApp">
        ...
    </activity>
</application>
Enter fullscreen mode Exit fullscreen mode
  • Replace with android:theme="@android:style/Theme.Material.Light.NoActionBar" in <application> which is the parent style in your themes.xml.
<application
    ...
    android:theme="@android:style/Theme.Material.Light.NoActionBar">
    ...
</application>
Enter fullscreen mode Exit fullscreen mode

The creates an app without the top action bar. The top action bar can be created using Scaffold composable function. See this Add Top App Bar Example in this Simple LazyColumn App.

I did try to not adding this android:theme="@android:style/Theme.Material.Light.NoActionBar" but the ComponentActivity() creates the top action bar by default which I have no idea to get rid of it using Jetpack Compose.

2. Set Status Bar Color with Compose

Well, the app works, but the default purple color in the status bar is gone.

If you look at the original themes.xml, it set the status bar color there.

<resources>
    <style name="Theme.EmptyComposeApp" parent="android:Theme.Material.Light.NoActionBar">
        <item name="android:statusBarColor">@color/purple_700</item>
    </style>
</resources>
Enter fullscreen mode Exit fullscreen mode

Since this has been removed, we need to implement this in Jetpack Compose. To do that, we need to use the System UI Controller from Accompanist.

  • Add com.google.accompanist:accompanist-systemuicontroller library:
dependencies {
    ...
    implementation "com.google.accompanist:accompanist-systemuicontroller:0.24.1-alpha"
    ...
}
Enter fullscreen mode Exit fullscreen mode
  • In ui\Theme\Theme.kt, add this to set purple_700 color which is also the primaryVariant color
val systemUiController = rememberSystemUiController()
systemUiController.setStatusBarColor(
    color = colors.primaryVariant
)
Enter fullscreen mode Exit fullscreen mode
  • The complete code looks like this:
fun EmptyComposeAppTheme(
    darkTheme: Boolean = isSystemInDarkTheme(),
    content: @Composable () -> Unit
) {
    val colors = if (darkTheme) {
        DarkColorPalette
    } else {
        LightColorPalette
    }

    val systemUiController = rememberSystemUiController()
    systemUiController.setStatusBarColor(
        color = colors.primaryVariant
    )

    MaterialTheme(
        colors = colors,
        typography = Typography,
        shapes = Shapes,
        content = content
    )
}
Enter fullscreen mode Exit fullscreen mode

The template project from Android Studio doesn't set the navigation bar color. Thus, we're not setting them here in the example above. However, we usually want to set the same color for both the navigation and status bars. In that case, you can use systemUiController.setSystemBarsColor():

val systemUiController = rememberSystemUiController()
systemUiController.setSystemBarsColor(
    color = colors.primaryVariant
)
Enter fullscreen mode Exit fullscreen mode

Since I want to use this example as my own template project. I'm going to change it to use systemUiController.setSystemBarsColor() in this app example.

3. Add Advertising ID Permission

The latest requirement for your app targeting SDK 33 is, you must declare this advertising ID (com.google.android.gms.permission.AD_ID) user permission in your manifest file.

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
    <uses-permission android:name="com.google.android.gms.permission.AD_ID"/>
    ...
</manifest>
Enter fullscreen mode Exit fullscreen mode

Issues

Preview is Not Rendering

There is a bug the preview is not working when the System UI controller is used. See issue tracker here.

java.lang.IllegalArgumentException: The Compose View must be hosted in an Activity with a Window!
Enter fullscreen mode Exit fullscreen mode

The workaround is you need to disable the System UI controller in preview by passing in the useSystemUIController parameter.

In Theme.kt:

@Composable  
fun EmptyComposeAppTheme(  
  ...  
  useSystemUIController: Boolean = true,  
  ...
) {  
    ...
    if (useSystemUIController) {  
        val systemUiController = rememberSystemUiController()  
        systemUiController.setStatusBarColor(  
            color = colors.primaryVariant)  
    }  
    ...
}
Enter fullscreen mode Exit fullscreen mode

In MainActivity.kt:

@Preview(showBackground = true)
@Composable
fun DefaultPreview() {
    EmptyComposeAppTheme(useSystemUIController = false) {
        ...
    }
}
Enter fullscreen mode Exit fullscreen mode

Clean Task is No Longer Needed

[Updated - Jan 11, 2023]: I found out that we no longer need to register the clean task in your build.gradle file as Gradle has already been implemented by default.

The following task can be removed from build.gradle file.

tasks {
    register("clean", Delete::class) {
        delete(rootProject.buildDir)
    }
}
Enter fullscreen mode Exit fullscreen mode

Summary

Now I only have strings.xml in my resource folder. Cool, isn't it?

Clean_Empty_Jetpack_Compose_App_01.png

I will probably also use this as starting point of any Jetpack Compose project. This diff here shows the changes needed to change the package and project names.

[Updated - 18 July, 2022]: If you rename the package and etc. from the template app, you're maybe getting the compilation error (usually is related to string resources). In that case, you can invalidate caches. I only turn on the Clear VCS Log caches and indexes and it works for me.

Source Code

GitHub Repository: Demo_CleanEmptyCompose


Originally published at https://vtsen.hashnode.dev.

💖 💪 🙅 🚩
vtsen
Vincent Tsen

Posted on May 28, 2022

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related