Draw sleep timeline graph in Compose
Viktor Mykhailiv on 2025-01-06
Draw sleep timeline graph in Compose
data:image/s3,"s3://crabby-images/09e9e/09e9e978d049a8d00b7d4377637c77d79772a31f" alt=""
Custom drawing is useful for when the built-in components just don’t cover exactly what our app needs. This article provides a guide to create a custom sleep timeline graph, similar to those you can find in the Fitbit app.
data:image/s3,"s3://crabby-images/4e513/4e5133c680d123c897300e23aa37c7fdc3a2fe3d" alt=""
How do we draw in Compose?
To get started with drawing in Compose we can use drawing modifiers or Canvas
composable which gives us DrawScope
— a declarative, stateless API to draw shapes and paths without requiring consumers to maintain underlying state. DrawScope
implementations are also provided sizing information and transformations are done relative to the local translation.
Note: Jetpack Compose (Android only) and Compose Multiplatform (Desktop, Android, iOS, web) have similar drawing API. Screenshots below are made on Desktop (macOS), but the result is the same on all platforms (check the last screenshot).
data:image/s3,"s3://crabby-images/3e7e7/3e7e7188c2c058957889cc46533e827f012315d6" alt=""
What is the sleep timeline?
We can read or write sleep data in Health Connect. Sleep data is displayed as a session, and can be divided into sleep stages:
- Awake: the user is awake within a sleep cycle.
- Light sleep: the user is in a light sleep cycle.
- Deep sleep: the user is in a deep sleep cycle.
- REM: the user is in a REM sleep cycle.
The SleepSessionRecord
data type has two parts:
- The overall session, spanning the entire duration of sleep.
- Individual stages during the sleep session such as light sleep or deep sleep.
Math
During the sleep session we can be in the same stage many times at different moments in time. We need to calculate the start and end points relative to the sleep session.
To draw a rect in Compose we need topOffset
and size
.
data:image/s3,"s3://crabby-images/74441/744413ba22eb8bf0fa67fcbc8891b01a7265eb1c" alt=""
data:image/s3,"s3://crabby-images/c25ca/c25ca8d5f27858147e62e5056f3108a993c5ee38" alt=""
Drawing
Let’s build our custom Canvas to draw one stage of the sleep session, e.g. deep.
If we run the project with the previously defined sleep session, we will see 3 rects: 1 grey rect for background and 2 purple rects for deep sleep stage.
data:image/s3,"s3://crabby-images/2a75e/2a75e30e642246f3d57cccbde3fad2994fe9d557" alt=""
To draw all stages of the sleep session (awake, REM, light, and deep) we need to make a few adjustments to draw each stage type as Column component, vertically, by drawing line by line and applying some offset for the next line.
data:image/s3,"s3://crabby-images/21cf8/21cf82059836e7ef56c342ae0b28ff3d123eef7c" alt=""
Draw text
To draw a text in Compose, we can typically use the Text
composable. However, in our example we are in a DrawScope
and we can use the DrawScope.drawText()
method.
Drawing text works a bit differently from other drawing commands. Normally, we give the drawing command the size (width and height) to draw the shape/image as. With text, there are a few parameters that control the size of the rendered text, such as font size, font, ligatures, and letter spacing. We need to use a TextMeasurer
to get access to the measured size of text, depending on the above factors.
data:image/s3,"s3://crabby-images/d9928/d9928eda2f82705c0814a6541a5c6ea3f01ae2a6" alt=""
data:image/s3,"s3://crabby-images/689d0/689d0567888bd5752a73efccab5e271fcc5a0348" alt=""
Please find the full example in my repository: https://github.com/vitoksmile/Sleep-timeline-graph