Skip to content
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions gradle/libs.versions.toml
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,7 @@ tiles = "1.5.0"
tracing = "1.3.0"
validatorPush = "1.0.0-alpha06"
version-catalog-update = "1.0.0"
watchfaceComplicationsDataSourceKtx = "1.2.1"
wear = "1.3.0"
wearComposeFoundation = "1.5.0-rc02"
wearComposeMaterial = "1.5.0-rc02"
Expand Down Expand Up @@ -151,6 +152,7 @@ androidx-tiles-testing = { module = "androidx.wear.tiles:tiles-testing", version
androidx-tiles-tooling = { module = "androidx.wear.tiles:tiles-tooling", version.ref = "tiles" }
androidx-tiles-tooling-preview = { module = "androidx.wear.tiles:tiles-tooling-preview", version.ref = "tiles" }
androidx-tracing = { module = "androidx.tracing:tracing", version.ref = "tracing" }
androidx-watchface-complications-data-source-ktx = { module = "androidx.wear.watchface:watchface-complications-data-source-ktx", version.ref = "watchfaceComplicationsDataSourceKtx" }
androidx-wear = { module = "androidx.wear:wear", version.ref = "wear" }
androidx-wear-ongoing = { module = "androidx.wear:wear-ongoing", version.ref = "wearOngoing" }
androidx-wear-tooling-preview = { module = "androidx.wear:wear-tooling-preview", version.ref = "wearToolingPreview" }
Expand Down
3 changes: 3 additions & 0 deletions settings.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,9 @@ pluginManagement {
mavenCentral()
}
}
plugins {
id("org.gradle.toolchains.foojay-resolver-convention") version "0.8.0"
}
dependencyResolutionManagement {
repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)
repositories {
Expand Down
2 changes: 2 additions & 0 deletions wear/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,8 @@ dependencies {
implementation(libs.horologist.compose.layout)
implementation(libs.horologist.compose.material)
implementation(libs.androidx.material.icons.core)
implementation(libs.androidx.watchface.complications.data.source.ktx)

androidTestImplementation(libs.androidx.compose.ui.test.junit4)
debugImplementation(libs.androidx.compose.ui.tooling)
debugImplementation(libs.androidx.compose.ui.tooling.preview)
Expand Down
64 changes: 64 additions & 0 deletions wear/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
Expand Up @@ -201,6 +201,70 @@
android:value="For Ongoing Activity"/>
</service>

<!-- [START android_wear_complication_manifest] -->
<service
android:name=".snippets.complication.MyComplicationDataSourceService"
android:exported="true"
android:label="@string/my_complication_service_label"
android:icon="@drawable/complication_icon"
android:permission="com.google.android.wearable.permission.BIND_COMPLICATION_PROVIDER">
<intent-filter>
<action android:name="android.support.wearable.complications.ACTION_COMPLICATION_UPDATE_REQUEST" />
</intent-filter>

<!-- Supported types should be comma-separated e.g. SHORT_TEXT,SMALL_IMAGE -->
<meta-data
android:name="android.support.wearable.complications.SUPPORTED_TYPES"
android:value="SHORT_TEXT" />
<meta-data
android:name="android.support.wearable.complications.UPDATE_PERIOD_SECONDS"
android:value="300" />

<!--
Optionally, the complication can be configured by the user by specifying a
configuration activity.
-->
<meta-data
android:name="android.support.wearable.complications.PROVIDER_CONFIG_ACTION"
android:value="MY_CONFIG_ACTION" />
</service>
<!-- [END android_wear_complication_manifest] -->

<!-- [START android_wear_timeline_complication_manifest] -->
<service
android:name=".snippets.complication.MyTimelineComplicationDataSourceService"
android:exported="true"
android:label="@string/my_complication_timeline_service_label"
android:icon="@drawable/complication_icon"
android:permission="com.google.android.wearable.permission.BIND_COMPLICATION_PROVIDER">
<intent-filter>
<action android:name="android.support.wearable.complications.ACTION_COMPLICATION_UPDATE_REQUEST" />
</intent-filter>

<meta-data
android:name="android.support.wearable.complications.SUPPORTED_TYPES"
android:value="SHORT_TEXT" />
<meta-data
android:name="android.support.wearable.complications.UPDATE_PERIOD_SECONDS"
android:value="300" />
</service>
<!-- [END android_wear_timeline_complication_manifest] -->

<activity
android:name=".snippets.complication.ConfigurationActivity"
android:label="@string/configuration_activity_label"
android:exported="true"
android:taskAffinity=""
android:theme="@android:style/Theme.DeviceDefault">
<!-- [START android_wear_complication_intent_filter] -->
<intent-filter>
<action android:name="MY_CONFIG_ACTION" />
<category android:name="android.support.wearable.complications.category.PROVIDER_CONFIG" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
<!-- [END android_wear_complication_intent_filter] -->
</activity>

</application>

</manifest>
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
package com.example.wear.snippets.complication

import android.os.Bundle
import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Spacer
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.height
import androidx.compose.runtime.Composable
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.unit.dp
import androidx.wear.compose.material3.Button
import androidx.wear.compose.material3.Text
import androidx.wear.watchface.complications.datasource.ComplicationDataSourceService.Companion.EXTRA_CONFIG_COMPLICATION_ID
import androidx.wear.watchface.complications.datasource.ComplicationDataSourceService.Companion.EXTRA_CONFIG_COMPLICATION_TYPE
import androidx.wear.watchface.complications.datasource.ComplicationDataSourceService.Companion.EXTRA_CONFIG_DATA_SOURCE_COMPONENT

class ConfigurationActivity: ComponentActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
// [START android_wear_complication_configuration_intent]
// Keys defined on ComplicationDataSourceService
val id = intent.getIntExtra(EXTRA_CONFIG_COMPLICATION_ID, -1)
val type = intent.getIntExtra(EXTRA_CONFIG_COMPLICATION_TYPE, -1)
val source = intent.getStringExtra(EXTRA_CONFIG_DATA_SOURCE_COMPONENT)
// [END android_wear_complication_configuration_intent]
setContent {
ComplicationConfig(
id = id,
type = type,
source = source
)
}
}

@Composable
fun ComplicationConfig(
modifier: Modifier = Modifier,
id: Int,
type: Int,
source: String?
) {
Column(
modifier = Modifier.fillMaxSize(),
verticalArrangement = Arrangement.SpaceEvenly,
horizontalAlignment = Alignment.CenterHorizontally
) {
Text("ID: $id")
Text("Type: $type")
Text("Source: $source")
Spacer(modifier = Modifier.height(4.dp))
Button(onClick = {
// [START android_wear_complication_configuration_finish]
setResult(RESULT_OK) // Or RESULT_CANCELLED to cancel configuration
finish()
// [END android_wear_complication_configuration_finish]
}) {
Text("Done!")
}
}
}
}

Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
/*
* Copyright 2025 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package com.example.wear.snippets.complication

import androidx.wear.watchface.complications.data.ComplicationData
import androidx.wear.watchface.complications.data.ComplicationType
import androidx.wear.watchface.complications.data.PlainComplicationText
import androidx.wear.watchface.complications.data.ShortTextComplicationData
import androidx.wear.watchface.complications.datasource.ComplicationRequest
import androidx.wear.watchface.complications.datasource.SuspendingComplicationDataSourceService

// [START android_wear_complication]
class MyComplicationDataSourceService : SuspendingComplicationDataSourceService() {
override suspend fun onComplicationRequest(request: ComplicationRequest): ComplicationData? {
// Retrieve latest info for inclusion in the data
val text = getLatestData()
return shortTextComplicationData(text)
}

override fun getPreviewData(type: ComplicationType): ComplicationData? {
return shortTextComplicationData("Event 1")
}

private fun shortTextComplicationData(text: String) =
ShortTextComplicationData.Builder(
text = PlainComplicationText.Builder(text).build(),
contentDescription = PlainComplicationText.Builder(text).build()
)
// Add further optional details here such as icon, tap action, title etc
.build()

// [START_EXCLUDE]
private fun getLatestData() = "Test"
// [END_EXCLUDE]
}
// [END android_wear_complication]
Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
/*
* Copyright 2025 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package com.example.wear.snippets.complication

import androidx.wear.watchface.complications.data.ComplicationData
import androidx.wear.watchface.complications.data.ComplicationType
import androidx.wear.watchface.complications.data.NoDataComplicationData
import androidx.wear.watchface.complications.data.PlainComplicationText
import androidx.wear.watchface.complications.data.ShortTextComplicationData
import androidx.wear.watchface.complications.datasource.ComplicationDataTimeline
import androidx.wear.watchface.complications.datasource.ComplicationRequest
import androidx.wear.watchface.complications.datasource.SuspendingTimelineComplicationDataSourceService
import androidx.wear.watchface.complications.datasource.TimeInterval
import androidx.wear.watchface.complications.datasource.TimelineEntry
import java.time.Instant
import java.time.temporal.ChronoUnit

data class CalendarEntry(
val start: Instant,
val end: Instant,
val name: String
)

// [START android_wear_timeline_complication]
class MyTimelineComplicationDataSourceService : SuspendingTimelineComplicationDataSourceService() {
override suspend fun onComplicationRequest(request: ComplicationRequest): ComplicationDataTimeline? {
if (request.complicationType != ComplicationType.SHORT_TEXT) {
return ComplicationDataTimeline(
defaultComplicationData = NoDataComplicationData(),
timelineEntries = emptyList()
)
}
// Retrieve list of events from your own datasource / database.
val events = getCalendarEvents()
return ComplicationDataTimeline(
defaultComplicationData = shortTextComplicationData("No event"),
timelineEntries = events.map {
TimelineEntry(
validity = TimeInterval(it.start, it.end),
complicationData = shortTextComplicationData(it.name)
)
}
)
}

override fun getPreviewData(type: ComplicationType): ComplicationData? {
return shortTextComplicationData("Event 1")
}

private fun shortTextComplicationData(text: String) =
ShortTextComplicationData.Builder(
text = PlainComplicationText.Builder(text).build(),
contentDescription = PlainComplicationText.Builder(text).build()
)
// Add further optional details here such as icon, tap action, title etc
.build()

// [START_EXCLUDE]
private fun getCalendarEvents(): List<CalendarEntry> {
val now = Instant.now()
return listOf(
CalendarEntry(now, now.plus(1, ChronoUnit.HOURS), "Event 1"),
CalendarEntry(now.plus(2, ChronoUnit.HOURS), now.plus(3, ChronoUnit.HOURS), "Event 2"),
CalendarEntry(now.plus(4, ChronoUnit.HOURS), now.plus(5, ChronoUnit.HOURS), "Event 3"),
)
}
// [END_EXCLUDE]
}
// [END android_wear_timeline_complication]
5 changes: 5 additions & 0 deletions wear/src/main/res/drawable/complication_icon.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android" android:height="24dp" android:tint="#FFFFFF" android:viewportHeight="960" android:viewportWidth="960" android:width="24dp">

<path android:fillColor="@android:color/white" android:pathData="M480,880Q397,880 324,848.5Q251,817 197,763Q143,709 111.5,636Q80,563 80,480Q80,397 111.5,324Q143,251 197,197Q251,143 324,111.5Q397,80 480,80Q563,80 636,111.5Q709,143 763,197Q817,251 848.5,324Q880,397 880,480Q880,563 848.5,636Q817,709 763,763Q709,817 636,848.5Q563,880 480,880ZM253,707L480,480L480,160Q346,160 253,253Q160,346 160,480Q160,544 184,603Q208,662 253,707Z"/>

</vector>
3 changes: 3 additions & 0 deletions wear/src/main/res/values/strings.xml
Original file line number Diff line number Diff line change
Expand Up @@ -6,4 +6,7 @@
<string name="message_detail">Message Detail</string>
<string name="tile_label">Hello Tile</string>
<string name="tile_description">Hello Tile Description</string>
<string name="my_complication_service_label">My Complication</string>
<string name="my_complication_timeline_service_label">My Timeline Complication</string>
<string name="configuration_activity_label">Configuration activity</string>
</resources>