DEV Community

Cover image for Using SFSymbols in Jetpack Compose
Dinko Marinac
Dinko Marinac

Posted on • Edited on

Using SFSymbols in Jetpack Compose

Motivation

When working with startups that primarily target USA customers it's very common to first develop an iOS app and then later create an Android clone.

Sometimes, you are faced with a challenge:

Develop an Android app with iOS design and backend developed with that in mind.

Almost every iOS app on the market relies on SF Symbols, Apple's default icon pack similar to Google Material icons or Font Awesome icons.

I will show a very elegant way to handle SF Symbols using Jetpack Compose.

Let's begin!

1. Convert SF Symbols to drawables

Since Apple will not provide you with the icons you need to use SF Symbols on other platforms, you have to look elsewhere.

I've stumbled upon this public Figma file with almost every SF Symbol you will ever need.

Find the symbols you need, and export them as an SVG.

Use the SVG to create a vector asset using Android Studio.

2. Create a custom Painter

SF Symbols are usually resolved from strings in iOS, so we will do the same. Create SFSymbolUtil with a method that returns an appropriate drawable.

private object SFSymbolUtil {
    fun getDrawableFor(symbolName: String): Int? {
        return when (symbolName) {
            "party.popper.fill" -> R.drawable.ic_party_popper_fill_colored
            "checkmark.rectangle.fill" -> R.drawable.ic_checkmark_rectangle_fill
            "figure.run.square.stack.fill" -> R.drawable.ic_figure_run_square_stack_fill_colored
            else -> {
                Timber.e("Could resolve SFSymbol: $symbolName")
                null
            }
        }
    }
}
Enter fullscreen mode Exit fullscreen mode

Now we can use that util to create a custom painter for SF Symbols.

@Composable
fun painterSFSymbol(sfSymbolName: String): Painter {
    SFSymbolUtil.getDrawableFor(symbolName = sfSymbolName)?.let {
        return painterResource(id = it)
    }
    return painterResource(id = R.drawable.ic_empty)
}
Enter fullscreen mode Exit fullscreen mode

If the painter receives an SF Symbol for which we don't have a drawable, we can use an empty drawable to draw nothing on the screen.

<vector xmlns:android="http://schemas.android.com/apk/res/android"
    android:width="1dp"
    android:height="1dp"
    android:viewportWidth="1"
    android:viewportHeight="1">

    <path
        android:pathData="M0,0Z"
        android:fillColor="#ffffff"/>

</vector>
Enter fullscreen mode Exit fullscreen mode

That's it, you can now easily use this Painter with your Image composable.

    Image(
        painter = painterSFSymbol(sfSymbolName = "checkmark.rectangle.fill")),
        contentDescription = null,
        modifier = Modifier.size(100.dp),
        contentScale = ContentScale.Fit,
    )
Enter fullscreen mode Exit fullscreen mode

If you have enjoyed or found this helpful, make sure to follow me for more content like this!

Read this on my website.

Image of Wix Studio

2025: Your year to build apps that sell

Dive into hands-on resources and actionable strategies designed to help you build and sell apps on the Wix App Market.

Get started

Top comments (0)

👋 Kindness is contagious

Engage with a sea of insights in this enlightening article, highly esteemed within the encouraging DEV Community. Programmers of every skill level are invited to participate and enrich our shared knowledge.

A simple "thank you" can uplift someone's spirits. Express your appreciation in the comments section!

On DEV, sharing knowledge smooths our journey and strengthens our community bonds. Found this useful? A brief thank you to the author can mean a lot.

Okay