1 Commits

10 changed files with 105 additions and 115 deletions

View File

@@ -1,7 +1,7 @@
plugins { plugins {
alias(libs.plugins.android.application) alias(libs.plugins.android.application)
alias(libs.plugins.kotlin.compose) alias(libs.plugins.kotlin.compose)
alias(libs.plugins.kotlin.serialization) alias(libs.plugins.jetbrains.kotlin.serialization)
alias(libs.plugins.ksp) alias(libs.plugins.ksp)
} }
@@ -61,7 +61,9 @@ dependencies {
implementation(libs.androidx.navigation3.runtime) implementation(libs.androidx.navigation3.runtime)
implementation(libs.androidx.lifecycle.viewmodel.navigation3) implementation(libs.androidx.lifecycle.viewmodel.navigation3)
implementation(libs.androidx.material3.adaptive.navigation3) implementation(libs.androidx.material3.adaptive.navigation3)
implementation(libs.kotlinx.serialization.json) implementation(libs.kotlinx.serialization.core)
implementation(libs.bundles.voyager)
// Room // Room
implementation(libs.androidx.room.runtime) implementation(libs.androidx.room.runtime)

View File

@@ -5,6 +5,8 @@ import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent import androidx.activity.compose.setContent
import androidx.activity.enableEdgeToEdge import androidx.activity.enableEdgeToEdge
import androidx.compose.material3.ExperimentalMaterial3Api import androidx.compose.material3.ExperimentalMaterial3Api
import cafe.adriel.voyager.navigator.Navigator
import com.fabisahne.cloudsync.ui.screens.HomeScreen
import com.fabisahne.cloudsync.ui.theme.CloudSyncTheme import com.fabisahne.cloudsync.ui.theme.CloudSyncTheme
class MainActivity : ComponentActivity() { class MainActivity : ComponentActivity() {
@@ -15,7 +17,7 @@ class MainActivity : ComponentActivity() {
enableEdgeToEdge() enableEdgeToEdge()
setContent { setContent {
CloudSyncTheme { CloudSyncTheme {
// Navigator(HomeScreen) Navigator(HomeScreen)
} }
} }
} }

View File

@@ -1,27 +0,0 @@
package com.fabisahne.cloudsync.navigation
import androidx.navigation3.runtime.NavKey
import kotlinx.serialization.Serializable
@Serializable
sealed interface Route : NavKey {
// Screens
@Serializable
data object HomeScreen : Route, NavKey
@Serializable
data class FolderPairScreen(val id: Long) : Route, NavKey
// Tabs
@Serializable
data object AccountTab : Route, NavKey
@Serializable
data object FolderPairTab : Route, NavKey
@Serializable
data object HistoryTab : Route, NavKey
@Serializable
data object SettingsTab : Route, NavKey
}

View File

@@ -0,0 +1,22 @@
package com.fabisahne.cloudsync.ui.navigation
import androidx.compose.foundation.layout.RowScope
import androidx.compose.material3.Icon
import androidx.compose.material3.NavigationBarItem
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import cafe.adriel.voyager.navigator.tab.LocalTabNavigator
import cafe.adriel.voyager.navigator.tab.Tab
@Composable
fun RowScope.TabNavigationItem(tab: Tab) {
val tabNavigator = LocalTabNavigator.current
NavigationBarItem(
selected = tabNavigator.current.key == tab.key,
onClick = { tabNavigator.current = tab },
label = { Text(tab.options.title) },
icon = { Icon(painter = tab.options.icon!!, contentDescription = tab.options.title) }
)
}

View File

@@ -8,16 +8,14 @@ import androidx.compose.material3.Text
import androidx.compose.material3.TopAppBar import androidx.compose.material3.TopAppBar
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.lifecycle.viewmodel.compose.viewModel import cafe.adriel.voyager.core.screen.Screen
import com.fabisahne.cloudsync.viewmodels.FolderPairScreenViewModel
data class FolderPairScreen(
val id: Long
) : Screen {
@OptIn(ExperimentalMaterial3Api::class) @OptIn(ExperimentalMaterial3Api::class)
@Composable @Composable
fun FolderPairScreen( override fun Content() {
modifier: Modifier = Modifier,
viewModel: FolderPairScreenViewModel = viewModel(),
id: Long
) {
Scaffold( Scaffold(
topBar = { topBar = {
TopAppBar( TopAppBar(
@@ -25,9 +23,10 @@ fun FolderPairScreen(
) )
}, },
content = { contentPadding -> content = { contentPadding ->
Surface(modifier = modifier.padding(contentPadding)) { Surface(modifier = Modifier.padding(contentPadding)) {
Text("ASDF: $id") Text("ASDF: $id")
} }
} }
) )
} }
}

View File

@@ -1,12 +1,7 @@
package com.fabisahne.cloudsync.ui.screens package com.fabisahne.cloudsync.ui.screens
import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.padding
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.filled.Route
import androidx.compose.material3.Button
import androidx.compose.material3.ExperimentalMaterial3Api import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.NavigationBar import androidx.compose.material3.NavigationBar
import androidx.compose.material3.NavigationBarDefaults import androidx.compose.material3.NavigationBarDefaults
import androidx.compose.material3.Scaffold import androidx.compose.material3.Scaffold
@@ -15,46 +10,55 @@ import androidx.compose.material3.Text
import androidx.compose.material3.TopAppBar import androidx.compose.material3.TopAppBar
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import cafe.adriel.voyager.core.screen.Screen
import cafe.adriel.voyager.navigator.LocalNavigator
import cafe.adriel.voyager.navigator.currentOrThrow
import cafe.adriel.voyager.navigator.tab.CurrentTab
import cafe.adriel.voyager.navigator.tab.TabNavigator
import com.fabisahne.cloudsync.ui.navigation.TabNavigationItem
import com.fabisahne.cloudsync.ui.tabs.AccountTab import com.fabisahne.cloudsync.ui.tabs.AccountTab
import com.fabisahne.cloudsync.ui.tabs.FolderPairTab import com.fabisahne.cloudsync.ui.tabs.FolderPairTab
import com.fabisahne.cloudsync.ui.tabs.HistoryTab import com.fabisahne.cloudsync.ui.tabs.HistoryTab
import com.fabisahne.cloudsync.ui.tabs.SettingsTab import com.fabisahne.cloudsync.ui.tabs.SettingsTab
@OptIn(ExperimentalMaterial3Api::class) object HomeScreen : Screen {
@Composable private fun readResolve(): Any = HomeScreen
fun HomeScreen(
modifier: Modifier = Modifier private val TABS = listOf(
) {
val tabs = listOf(
HistoryTab, HistoryTab,
FolderPairTab, FolderPairTab,
AccountTab, AccountTab,
SettingsTab, SettingsTab,
) )
@OptIn(ExperimentalMaterial3Api::class)
@Composable
override fun Content() {
val navigator = LocalNavigator.currentOrThrow
TabNavigator(
HistoryTab
) { tabNavigator ->
Scaffold( Scaffold(
topBar = { topBar = {
TopAppBar( TopAppBar(
title = { Text("Home Screen") } title = { Text(text = tabNavigator.current.options.title) }
) )
}, },
content = { padding -> content = { padding ->
Surface(modifier = modifier.padding(padding)) { Surface(modifier = Modifier.padding(padding)) {
Text("Home Screen Content") CurrentTab()
} }
}, },
bottomBar = { bottomBar = {
NavigationBar(windowInsets = NavigationBarDefaults.windowInsets) { NavigationBar(windowInsets = NavigationBarDefaults.windowInsets) {
tabs.forEach { _ -> TABS.forEach {
Button({}) { TabNavigationItem(it)
Icon(
Icons.Default.Route,
null
)
}
} }
} }
} }
) )
} }
}
}

View File

@@ -1,6 +1,7 @@
package com.fabisahne.cloudsync.ui.tabs package com.fabisahne.cloudsync.ui.tabs
import androidx.compose.material.icons.Icons import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.filled.AccountBox
import androidx.compose.material.icons.filled.Settings import androidx.compose.material.icons.filled.Settings
import androidx.compose.material3.Text import androidx.compose.material3.Text
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
@@ -20,7 +21,7 @@ object SettingsTab : Tab {
return remember { return remember {
TabOptions( TabOptions(
index = 3u, index = 2u,
title = title, title = title,
icon = icon icon = icon
) )

View File

@@ -1,20 +0,0 @@
package com.fabisahne.cloudsync.viewmodels
import androidx.lifecycle.ViewModel
import com.fabisahne.cloudsync.data.FolderPair
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.asStateFlow
class FolderPairScreenViewModel(
private val id: Long
) : ViewModel() {
private val _state = MutableStateFlow(
FolderPair(
id = id,
name = "ASDF",
localDir = "asdf/asdf",
cloudPath = "cloud/folder"
)
)
val state = _state.asStateFlow()
}

View File

@@ -2,5 +2,4 @@
plugins { plugins {
alias(libs.plugins.android.application) apply false alias(libs.plugins.android.application) apply false
alias(libs.plugins.kotlin.compose) apply false alias(libs.plugins.kotlin.compose) apply false
alias(libs.plugins.kotlin.serialization) apply false
} }

View File

@@ -5,18 +5,18 @@ junit = "4.13.2"
junitVersion = "1.3.0" junitVersion = "1.3.0"
espressoCore = "3.7.0" espressoCore = "3.7.0"
lifecycleRuntimeKtx = "2.10.0" lifecycleRuntimeKtx = "2.10.0"
activityCompose = "1.12.4" activityCompose = "1.12.2"
kotlin = "2.3.10" kotlin = "2.2.10"
composeBom = "2026.02.00" composeBom = "2024.09.00"
voyager = "1.1.0-beta03" voyager = "1.1.0-beta03"
room = "2.8.4" room = "2.7.1"
ksp = "2.3.6" ksp = "2.3.6"
nav3Core = "1.0.1" nav3Core = "1.0.0"
lifeCycleViewmodelNav3 = "2.10.0" lifeCycleViewmodelNav3 = "2.10.0"
kotlinSerialization = "2.3.10" kotlinSerialization = "2.2.21"
kotlinxSerializationCore = "1.10.0" kotlinxSerializationCore = "1.9.0"
material3AdaptiveNav3 = "1.3.0-alpha08" material3AdaptiveNav3 = "1.3.0-alpha06"
[libraries] [libraries]
androidx-core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "coreKtx" } androidx-core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "coreKtx" }
@@ -40,9 +40,14 @@ androidx-navigation3-ui = { module = "androidx.navigation3:navigation3-ui", vers
androidx-lifecycle-viewmodel-navigation3 = { module = "androidx.lifecycle:lifecycle-viewmodel-navigation3", version.ref = "lifeCycleViewmodelNav3" } androidx-lifecycle-viewmodel-navigation3 = { module = "androidx.lifecycle:lifecycle-viewmodel-navigation3", version.ref = "lifeCycleViewmodelNav3" }
kotlinx-serialization-core = { module = "org.jetbrains.kotlinx:kotlinx-serialization-core", version.ref = "kotlinxSerializationCore" } kotlinx-serialization-core = { module = "org.jetbrains.kotlinx:kotlinx-serialization-core", version.ref = "kotlinxSerializationCore" }
kotlinx-serialization-json = { module = "org.jetbrains.kotlinx:kotlinx-serialization-json", version.ref = "kotlinxSerializationCore" }
androidx-material3-adaptive-navigation3 = { group = "androidx.compose.material3.adaptive", name = "adaptive-navigation3", version.ref = "material3AdaptiveNav3" } androidx-material3-adaptive-navigation3 = { group = "androidx.compose.material3.adaptive", name = "adaptive-navigation3", version.ref = "material3AdaptiveNav3" }
# Voyager
voyager-navigator = { module = "cafe.adriel.voyager:voyager-navigator", version.ref = "voyager" }
voyager-screenmodel = { module = "cafe.adriel.voyager:voyager-screenmodel", version.ref = "voyager" }
voyager-tab-navigator = { module = "cafe.adriel.voyager:voyager-tab-navigator", version.ref = "voyager" }
voyager-transitions = { module = "cafe.adriel.voyager:voyager-transitions", version.ref = "voyager" }
# Room # Room
androidx-room-runtime = { group = "androidx.room", name = "room-runtime", version.ref = "room" } androidx-room-runtime = { group = "androidx.room", name = "room-runtime", version.ref = "room" }
androidx-room-ktx = { group = "androidx.room", name = "room-ktx", version.ref = "room" } androidx-room-ktx = { group = "androidx.room", name = "room-ktx", version.ref = "room" }
@@ -51,5 +56,8 @@ androidx-room-compiler = { group = "androidx.room", name = "room-compiler", vers
[plugins] [plugins]
android-application = { id = "com.android.application", version.ref = "agp" } android-application = { id = "com.android.application", version.ref = "agp" }
kotlin-compose = { id = "org.jetbrains.kotlin.plugin.compose", version.ref = "kotlin" } kotlin-compose = { id = "org.jetbrains.kotlin.plugin.compose", version.ref = "kotlin" }
kotlin-serialization = { id = "org.jetbrains.kotlin.plugin.serialization", version.ref = "kotlinSerialization" } jetbrains-kotlin-serialization = { id = "org.jetbrains.kotlin.plugin.serialization", version.ref = "kotlinSerialization" }
ksp = { id = "com.google.devtools.ksp", version.ref = "ksp" } ksp = { id = "com.google.devtools.ksp", version.ref = "ksp" }
[bundles]
voyager = ["voyager-navigator", "voyager-screenmodel", "voyager-tab-navigator", "voyager-transitions"]