SwiftUI support for Skip apps.
SkipUI vends the skip.ui
Kotlin package. It is a reimplementation of SwiftUI for Kotlin on Android using Jetpack Compose. Its goal is to mirror as much of SwiftUI as possible, allowing Skip developers to use SwiftUI with confidence.
SkipUI depends on the skip transpiler plugin. The transpiler must transpile SkipUI's own source code, and SkipUI relies on the transpiler's transformation of SwiftUI code. See Implementation Strategy for details. SkipUI also depends on the SkipFoundation and SkipModel packages.
SkipUI is part of the core SkipStack and is not intended to be imported directly.
The module is transparently adopted through the translation of import SwiftUI
into import skip.ui.*
by the Skip transpiler.
SkipUI - together with the Skip transpiler - has robust support for the building blocks of SwiftUI, including its state flow and declarative syntax. SkipUI also implements many of SwiftUI's basic layout and control views, as well as many core modifiers. It is possible to write an Android app entirely in SwiftUI utilizing SkipUI's current component set.
SkipUI is a young library, however, and much of SwiftUI's vast surface area is not yet implemented. You are likely to run into limitations while writing real-world apps. See Supported SwiftUI for a full list of supported components and constructs. Anything not listed there is likely not yet ported.
When you want to use a SwiftUI construct that has not been implemented, you have options. You can try to find a workaround using only supported components, embed Compose code directly, or add support to SkipUI. If you choose to enhance SkipUI itself, please consider contributing your code back for inclusion in the official release.
We welcome contributions to SkipUI. The Skip product documentation includes helpful instructions and tips on local Skip library development.
The most pressing need is to implement more core components and view modifiers. View modifiers in particular are a ripe source of low-hanging fruit. The Compose Modifier
type often has built-in functions that replicate SwiftUI modifiers, making these SwiftUI modifiers easy to implement.
To help fill in unimplemented API in SkipUI:
- Find unimplemented API. Unimplemented API will either be within
#if !SKIP
blocks, or will be marked with@available(*, unavailable)
. Note that most unimplementedView
modifiers are in theView.swift
source file. - Write an appropriate Compose implementation. See Implementation Strategy below.
- Write tests and/or playground code to exercise your component. See Tests.
- Submit a PR.
Other forms of contributions such as test cases, comments, and documentation are also welcome!
SkipUI does not work in isolation. It depends on transformations the skip transpiler plugin makes to SwiftUI code. And while Skip generally strives to write Kotlin that is similar to hand-crafted code, these SwiftUI transformations are not something you'd want to write yourself. Before discussing SkipUI's implementation, let's explore them.
Both SwiftUI and Compose are declarative UI frameworks. Both have mechanisms to track state and automatically re-render when state changes. SwiftUI models user interface elements with View
objects, however, while Compose models them with @Composable
functions. The Skip transpiler must therefore translate your code defining a View
graph into @Composable
function calls. This involves two primary transformations:
- The transpiler inserts code to sync
View
members that have special meanings in SwiftUI -@State
,@EnvironmentObject
, etc - with the corresponding Compose state mechanisms, which are not member-based. The syncing goes two ways, so that yourView
members are populated from Compose's state values, and changing yourView
members updates Compose's state values. - The transpiler turns
@ViewBuilders
- includingView.body
- into@Composable
function calls.
The second transformation in particular deserves some explanation, because it may help you to understand SkipUI's internal API. Consider the following simple example:
struct V: View {
let isHello: Bool
var body: some View {
if isHello {
Text("Hello!")
} else {
Text("Goodbye!")
}
}
}
The transpilation would look something like the following:
class V: View {
val isHello: Bool
constructor(isHello: Bool) {
this.isHello = isHello
}
override fun body(): View {
return ComposeView { composectx ->
if (isHello) {
Text("Hello!").Compose(context = composectx)
} else {
Text("Goodbye!").Compose(context = composectx)
}
ComposeResult.ok
}
}
...
}
Notice the changes to the body
content. Rather than returning an arbitrary view tree, the transpiled body
always returns a single ComposeView
, a special SkipUI view type that invokes a @Composable
block. The logic of the original body
is now within that block, and any View
that body
would have returned instead invokes its own Compose(context:)
function to render the corresponding Compose component. The Compose(context:)
function is part of SkipUI's View
API.
Thus the transpiler is able to turn any View.body
- actually any @ViewBuilder
- into a block of Compose code that it can invoke to render the desired content. A later section details how you can use ComposeView
yourself to move fluidly between SwiftUI and Compose when writing your Android UI.
SkipUI contains stubs for the entire SwiftUI framework. API generally goes through three phases:
- Code that no one has begun to port to Skip starts in
#if !SKIP
blocks. This hides it from the Skip transpiler. - The first implementation step is to move code out of
#if !SKIP
blocks so that it will be transpiled. This is helpful on its own, even if you just mark the API@available(*, unavailable)
because you are not ready to implement it for Compose. Anunavailable
attribute will provide Skip users with a clear error message, rather than relying on the Kotlin compiler to complain about unfound API.- When moving code out of a
#if !SKIP
block, please strip Apple's extensive API comments. There is no reason for Skip to duplicate the official SwiftUI documentation, and it obscures any Skip-specific implementation comments we may add. - SwiftUI uses complex generics extensively, and the generics systems of Swift and Kotlin have significant differences. You may have to replace some generics or generic constraints with looser typing in order to transpile successfully.
- Reducing the number of Swift extensions and instead folding API into the primary declaration of a type can make Skip's internal symbol storage more efficient. You may, should, however, leave
View
modifiers that are specific to a given component - e.g..navigationTitle
is specific toNavigationStack
- within the component's source file.
- When moving code out of a
- Finally, we add a Compose implementation and remove any
unavailable
attribute.
Note that SkipUI should remain buildable throughout this process. Being able to successfully compile SkipUI in Xcode helps us validate that our ported components still mesh with the rest of the framework.
Before implementing a component, familiarize yourself with SkipUI's View
protocol in Sources/View/View.swift
as well as the files in the Sources/Compose
directory. It is also helpful to browse the source code for components and modifiers that have already been ported. See the table of Supported SwiftUI.
The Text
view exemplifies a typical SwiftUI component implementation. Here is an abbreviated code sample:
public struct Text: View, Equatable, Sendable {
let text: String
public init(_ text: String) {
self.text = text
}
...
#if SKIP
@Composable public override func ComposeContent(context: ComposeContext) {
let modifier = context.modifier
let font = EnvironmentValues.shared.font ?? Font(fontImpl: { LocalTextStyle.current })
...
androidx.compose.material3.Text(text: text, modifier: modifier, style: font.fontImpl(), ...)
}
#else
public var body: some View {
stubView()
}
#endif
}
As you can see, the Text
type is defined just as it is in SwiftUI. We then use an #if SKIP
block to implement the composable View.ComposeContent
function for Android, while we stub the body
var to satisfy the Swift compiler. ComposeContent
makes the necessary Compose calls to render the component, applying the modifier from the given context
as well as any applicable environment values. If Text
had any child views, ComposeContent
would call child.Compose(context: context.content())
to compose its child content. (Note that View.Compose(context:)
delegates to View.ComposeContent(context:)
after performing other bookkeeping operations, which is why we override ComposeContent
rather than Compose
.)
Most modifiers, on the other hand, use the ComposeModifierView
to change the context
passed to the modified view. Here is the .opacity
modifier:
extension View {
public func opacity(_ opacity: Double) -> some View {
#if SKIP
return ComposeModifierView(contextView: self) { context in
context.modifier = context.modifier.alpha(Float(opacity))
}
#else
return self
#endif
}
}
Some modifiers have their own composition logic. These modifiers use a different ComposeModifierView
constructor whose block defines the composition. Here, for example, .task
uses Compose's LaunchedEffect
to run an asynchronous block the first time it is composed:
extension View {
public func task(id value: Any, priority: TaskPriority = .userInitiated, _ action: @escaping () async -> Void) -> some View {
#if SKIP
return ComposeModifierView(contentView: self) { view, context in
let handler = rememberUpdatedState(action)
LaunchedEffect(value) {
handler.value()
}
view.Compose(context: context)
}
#else
return self
#endif
}
}
Like other SwiftUI components, modifiers use #if SKIP ... #else ...
to stub the Swift implementation and keep SkipUI buildable in Xcode.
ComposeView
is an Android-only SwiftUI view that you can use to embed Compose code directly into your SwiftUI view tree. In the following example, we use a SwiftUI Text
to write "Hello from SwiftUI", followed by calling the androidx.compose.material3.Text()
Compose function to write "Hello from Compose" below it:
VStack {
Text("Hello from SwiftUI")
#if SKIP
ComposeView { _ in
androidx.compose.material3.Text("Hello from Compose")
return .ok
}
#endif
}
Skip also enhances all SwiftUI views with a Compose()
method, allowing you to use SwiftUI views from within Compose. The following example again uses a SwiftUI Text
to write "Hello from SwiftUI", but this time from within a ComposeView
:
#if SKIP
ComposeView { context in
androidx.compose.foundation.layout.Column(modifier: context.modifier) {
Text("Hello from SwiftUI").Compose(context: context.content())
androidx.compose.material3.Text("Hello from Compose")
}
return .ok
}
#endif
Or:
#if SKIP
ComposeView { context in
VStack {
Text("Hello from SwiftUI").Compose(context: context.content())
androidx.compose.material3.Text("Hello from Compose")
}.Compose(context: context) // Returns .ok
}
#endif
With ComposeView
and the Compose()
function, you can move fluidly between SwiftUI and Compose code. These techniques work not only with standard SwiftUI and Compose components, but with your own custom SwiftUI views and Compose functions as well.
Note that ComposeView
and the Compose()
function are only available in Android, so you must guard all uses with the #if SKIP
or #if os(Android)
compiler directives.
SkipUI supports loading images from URLs using SwiftUI's AsyncImage
. Our implementation uses the Coil library to download images on Android.
To display a standard SwiftUI Image
, SkipUI currently only supports the Image(systemName:)
constructor. The table below details the mapping between iOS and Android system images. Other system names are not supported. Loading images from resources is also not yet supported. These restrictions also apply to other components that load images, such as SwiftUI.Label
.
In addition to the system images below, you can display any emoji using Text
.
If these options do not meet your needs, consider embedding Compose code directly until resource loading is implemented.
iOS | Android |
---|---|
arrow.clockwise.circle | Icons.Outlined.Refresh |
arrow.forward | Icons.Outlined.ArrowForward |
arrow.forward.square | Icons.Outlined.ExitToApp |
arrow.left | Icons.Outlined.ArrowBack |
arrowtriangle.down.fill | Icons.Outlined.ArrowDropDown |
bell | Icons.Outlined.Notifications |
bell.fill | Icons.Filled.Notifications |
calendar | Icons.Outlined.DateRange |
cart | Icons.Outlined.ShoppingCart |
cart.fill | Icons.Filled.ShoppingCart |
checkmark | Icons.Outlined.Check |
checkmark.circle | Icons.Outlined.CheckCircle |
checkmark.circle.fill | Icons.Filled.CheckCircle |
chevron.down | Icons.Outlined.KeyboardArrowDown |
chevron.left | Icons.Outlined.KeyboardArrowLeft |
chevron.right | Icons.Outlined.KeyboardArrowRight |
chevron.up | Icons.Outlined.KeyboardArrowUp |
ellipsis | Icons.Outlined.MoreVert |
envelope | Icons.Outlined.Email |
envelope.fill | Icons.Filled.Email |
exclamationmark.triangle | Icons.Outlined.Warning |
exclamationmark.triangle.fill | Icons.Filled.Warning |
face.smiling | Icons.Outlined.Face |
gearshape | Icons.Outlined.Settings |
gearshape.fill | Icons.Filled.Settings |
hand.thumbsup | Icons.Outlined.ThumbUp |
hand.thumbsup.fill | Icons.Filled.ThumbUp |
heart | Icons.Outlined.FavoriteBorder |
heart.fill | Icons.Outlined.Favorite |
house | Icons.Outlined.Home |
house.fill | Icons.Filled.Home |
info.circle | Icons.Outlined.Info |
info.circle.fill | Icons.Filled.Info |
line.3.horizontal | Icons.Outlined.Menu |
list.bullet | Icons.Outlined.List |
location | Icons.Outlined.LocationOn |
location.fill | Icons.Filled.LocationOn |
lock | Icons.Outlined.Lock |
lock.fill | Icons.Filled.Lock |
magnifyingglass | Icons.Outlined.Search |
mappin.circle | Icons.Outlined.Place |
mappin.circle.fill | Icons.Filled.Place |
paperplane | Icons.Outlined.Send |
paperplane.fill | Icons.Filled.Send |
pencil | Icons.Outlined.Create |
person | Icons.Outlined.Person |
person.crop.circle | Icons.Outlined.AccountCircle |
person.crop.circle.fill | Icons.Filled.AccountCircle |
person.crop.square | Icons.Outlined.AccountBox |
person.crop.square.fill | Icons.Filled.AccountBox |
person.fill | Icons.Filled.Person |
phone | Icons.Outlined.Call |
phone.fill | Icons.Filled.Call |
play | Icons.Outlined.PlayArrow |
play.fill | Icons.Filled.PlayArrow |
plus | Icons.Outlined.Add |
plus.circle.fill | Icons.Outlined.AddCircle |
square.and.arrow.up | Icons.Outlined.Share |
square.and.arrow.up.fill | Icons.Filled.Share |
star | Icons.Outlined.Star |
star.fill | Icons.Filled.Star |
trash | Icons.Outlined.Delete |
trash.fill | Icons.Filled.Delete |
wrench | Icons.Outlined.Build |
wrench.fill | Icons.Filled.Build |
xmark | Icons.Outlined.Clear |
In Android-only code, you can also supply any androidx.compose.material.icons.Icons
image name as the systemName
. For example:
#if SKIP
Image(systemName: "Icons.Filled.Settings")
#endif
SwiftUI Lists
are powerful and flexible components. SkipUI currently supports the following patterns for specifying List
content.
Static content. Embed a child view for each row directly within the List
:
List {
Text("Row 1")
Text("Row 2")
Text("Row 3")
}
Indexed content. Specify an Int
range and a closure to create a row for each index:
List(1...100) { index in
Text("Row \(index)")
}
Collection content. Supply any RandomAccessCollection
- typically an Array
- and a closure to create a row for each element. If the elements do not implement the Identifiable
protocol, specify the key path to a property that can be used to uniquely identify each element:
List([person1, person2, person3], id: \.fullName) { person in
HStack {
Text(person.fullName)
Spacer()
Text(person.age)
}
}
ForEach
content. Use ForEach
to specify indexed or collection content. This allows you to mix content types.
List {
Text("People").bold()
ForEach([person1, person2, person3], id: \.fullName) { person in
HStack {
Text(person.fullName)
Spacer()
Text(person.age)
}
}
}
SkipUI does not support placing modifiers on Section
or ForEach
views within lists.
SwiftUI has three primary forms of navigation: TabView
, NavigationStack
, and modal presentations. SkipUI has implemented all three, albeit with the restrictions explained below.
SkipUI's TabView
does yet not support SwiftUI's overflow tab behavior. Adding too many tabs will just result in too many tabs rather than SwiftUI's automatic "More" tab.
Otherwise, TabView
acts as you would expect. NavigationStack
, however, has several restrictions you must be aware of.
In SwiftUI, you push vies onto a NavigationStack
with NavigationLink
. NavigationLink
has two ways to specify its destination view: embedding the view directly, or specifying a value that is mapped to a view through the .navigationDestination
modifier, as in the following code sample:
NavigationStack {
ListView()
.navigationTitle(Self.title)
}
struct ListView : View {
var body: some View {
List(City.allCases) { city in
NavigationLink(value: city) {
rowView(city: city)
}
}
.navigationDestination(for: City.self) { city in
CityView(city: city)
}
}
}
SkipUI does not support embedding a destination view directly in a NavigationLink
, and future support may prove difficult. Compose forces you to define fixed navigation routes, making any dynamic navigation a challenge.
SkipUI does support navigation with .navigationDestination
as in the example above, because we can map each modifier to a fixed Compose navigation route. Even this support, however, required some abuse of Compose's system in order to allow newly-pushed views to defined additional .navigationDestinations
. In fact, it is currently the case that if a pushed view defines a new .navigationDestination
for key type T
, it will overwrite any previous stack view's T
destination mapping. Take care not to unintentionally re-map the same key type in the same navigation stack.
Compose imposes an additional restriction as well: we must be able to stringify .navigationDestination
key types. SkipUI uses the following stringification algorithm:
- If the type is
Identifiable
, useString(describing: target.id)
- If the type is
RawRepresentable
, useString(describing: target.rawValue)
- Else use
String(describing: target)
Finally, SkipUI does not yet support binding to an array of destination values to specify the navigation stack.
For modal presentations, SkipUI supports the .sheet(isPresented:onDismiss:content:)
modifier only. We will add support for other forms of modal presentations in the future.
SkipUI utilizes a combination of unit tests, UI tests, and basic snapshot tests in which the snapshots are converted into ASCII art for easy processing.
Perhaps the most common way to test SkipUI's support for a SwiftUI component, however, is through the Skip playground app. Whenever you add or update support for a visible element of SwiftUI, make sure there is a playground that exercises the element. This not only gives us a mechanism to test appearance and behavior, but the playground app becomes a showcase of supported SwiftUI components on Android over time.
Component | Support Level | Notes |
---|---|---|
@AppStorage |
Medium | |
@Bindable |
Full | |
@Binding |
Full | |
@Environment |
Full | Custom keys supported, but most builtin keys not yet available |
@EnvironmentObject |
Full | |
@ObservedObject |
Full | |
@State |
Full | |
@StateObject |
Full | |
Custom Views | Full | |
AsyncImage |
High | Uses Coil |
Button |
High | |
Color |
High | |
Divider |
Full | |
EmptyView |
Full | |
Font |
Medium | |
ForEach |
Medium | Bindings as data not supported |
Form |
Full | |
Group |
Full | |
HStack |
Full | |
Image |
Low | See Images |
Label |
Low | See Images |
List |
Medium | See Lists |
NavigationLink |
Medium | See Navigation |
NavigationStack |
Medium | See Navigation |
ProgressView |
Medium | Labels not supported |
ScrollView |
Full | |
Section |
High | Only tested for Text content within List |
Slider |
Medium | Labels, onEditingChanged not supported |
Spacer |
Medium | minLength not supported |
TabView |
Medium | See Navigation |
Text |
High | Formatting not supported |
TextField |
High | Formatting not supported |
Toggle |
Medium | Styling, sources not supported |
VStack |
Full | |
ZStack |
Full | |
.aspectRatio |
Medium | Supported for images |
.background |
Low | Only color supported |
.bold |
Full | |
.border |
Full | |
.buttonStyle |
High | Custom styles not supported |
.clipped |
Low | Any resizable image clips automatically |
.disabled |
Full | |
.environment |
Full | |
.environmentObject |
Full | |
.font |
Full | |
.foregroundColor |
Full | |
.foregroundStyle |
Medium | Only color supported |
.frame |
Low | Only fixed dimensions supported |
.hidden |
Full | |
.italic |
Full | |
.labelsHidden |
Full | |
.lineLimit |
Medium | Only Int? value supported |
.listItemTint |
Full | |
.listStyle |
Full | |
.navigationDestination |
Medium | See Navigation |
.navigationTitle |
Full | |
.offset |
Full | |
.onLongPressGesture |
Medium | minimumDuration , maximumDistance , onPressingChanged not supported |
.onTapGesture |
High | Tap count > 2 not supported |
.opacity |
Full | |
.padding |
High | Compose does not support negative padding |
.progressViewStyle |
High | Custom styles not supported |
.resizable |
Low | capInsets and resizingMode not supported |
.rotationEffect |
Medium | |
.scaledToFill |
Medium | Supported for images |
.scaledToFit |
Medium | Supported for images |
.scaleEffect |
Medium | |
.sheet |
High | See Navigation |
.tabItem |
Full | |
.textFieldStyle |
Medium | .plain not supported |
.task |
Full | |
.tint |
Full |
androidx.compose.material package androidx.compose.ui.Modifier list
- Text (androidx.compose.material3.Text): Displays a text element on the screen.
- Button (androidx.compose.material3.Button): Creates a clickable button.
- Surface (androidx.compose.material3.Surface): Defines a surface with a background color and elevation.
- Image (androidx.compose.foundation.Image): Displays an image.
- Box (androidx.compose.foundation.Box): A composable that places its children in a box layout.
- Row (androidx.compose.foundation.layout.Row): Lays out its children in a horizontal row.
- Column (androidx.compose.foundation.layout.Column): Lays out its children in a vertical column.
- Spacer (androidx.compose.ui.layout.Spacer): Adds empty space between composables.
- Card (androidx.compose.material3.Card): Creates a Material Design card.
- TextField (androidx.compose.material3.TextField): Creates an editable text field.
- TopAppBar (androidx.compose.material3.TopAppBar): Creates a Material Design top app bar.
- BottomAppBar (androidx.compose.material3.BottomAppBar): Creates a Material Design bottom app bar.
- FloatingActionButton (androidx.compose.material3.FloatingActionButton): Creates a floating action button.
- AlertDialog (androidx.compose.material3.AlertDialog): Creates a Material Design alert dialog.
- ModalBottomSheetLayout (androidx.compose.material3.ModalBottomSheetLayout): Creates a modal bottom sheet.
- IconButton (androidx.compose.material3.IconButton): Creates an icon button.
- OutlinedTextField (androidx.compose.material3.OutlinedTextField): Creates an outlined text field.
- LazyColumn (androidx.compose.foundation.lazy.LazyColumn): Creates a lazily laid out column.
- LazyRow (androidx.compose.foundation.lazy.LazyRow): Creates a lazily laid out row.
- LazyVerticalGrid (androidx.compose.foundation.lazy.LazyVerticalGrid): Creates a lazily laid out vertical grid.
- LazyRow (androidx.compose.foundation.lazy.LazyRow): Creates a lazily laid out row with horizontally scrolling items.
- LazyColumnFor (androidx.compose.foundation.lazy.LazyColumnFor): Creates a lazily laid out column for a list of items.
- LazyRowFor (androidx.compose.foundation.lazy.LazyRowFor): Creates a lazily laid out row for a list of items.
- LazyVerticalGridFor (androidx.compose.foundation.lazy.LazyVerticalGridFor): Creates a lazily laid out vertical grid for a list of items.
- Clickable (androidx.compose.ui.Modifier.clickable): Adds a click listener to a composable.
- Icon (androidx.compose.material3.Icon): Displays an icon from the Material Icons font.
- IconButton (androidx.compose.material3.IconButton): Creates an icon button with optional click listener.
- Checkbox (androidx.compose.material3.Checkbox): Creates a checkbox.
- RadioButton (androidx.compose.material3.RadioButton): Creates a radio button.
- Switch (androidx.compose.material3.Switch): Creates a switch (on/off toggle).
- Slider (androidx.compose.material3.Slider): Creates a slider for selecting a value within a range.
- LinearProgressIndicator (androidx.compose.material3.LinearProgressIndicator): Creates a linear progress indicator.
- CircularProgressIndicator (androidx.compose.material3.CircularProgressIndicator): Creates a circular progress indicator.
- Divider (androidx.compose.material3.Divider): Creates a horizontal divider.
- Spacer (androidx.compose.foundation.layout.Spacer): Adds empty space between composables.
- AlertDialog (androidx.compose.material3.AlertDialog): Creates an alert dialog with customizable buttons and content.
- Snackbar (androidx.compose.material3.SnackbarHost): Creates a snackbar to display short messages.
- DropdownMenu (androidx.compose.material3.DropdownMenu): Creates a dropdown menu with a list of items.
- Drawer (androidx.compose.material3.Drawer): Creates a sliding drawer panel for navigation.
- MaterialTheme (androidx.compose.material3.MaterialTheme): Applies Material Design styles to its children.