From 0ef8a1e2551c7b7a4bed733135946f15784ec8f1 Mon Sep 17 00:00:00 2001 From: Amk Date: Tue, 24 Feb 2026 13:56:54 +0000 Subject: [PATCH] Initial commit --- .editorconfig | 21 ++ .gitattributes | 3 + .gitignore | 126 +++++++++ .idea/codeStyles/Project.xml | 175 ++++++++++++ .idea/codeStyles/codeStyleConfig.xml | 5 + README.md | 87 ++++++ build.gradle.kts | 7 + composeApp/build.gradle.kts | 105 ++++++++ .../1.json | 89 +++++++ .../src/androidMain/AndroidManifest.xml | 25 ++ .../contacts/ContactsApplication.kt | 34 +++ .../io/assessment/contacts/MainActivity.kt | 18 ++ .../presentation/ContactsScreenPreview.kt | 149 +++++++++++ .../data/database/DatabaseFactory.android.kt | 19 ++ .../assessment/contacts/di/DatabaseModule.kt | 18 ++ .../presentation/GroupsScreenPreview.kt | 73 +++++ .../res/mipmap-hdpi/ic_launcher.webp | Bin 0 -> 968 bytes .../res/mipmap-hdpi/ic_launcher_round.webp | Bin 0 -> 2470 bytes .../res/mipmap-mdpi/ic_launcher.webp | Bin 0 -> 748 bytes .../res/mipmap-mdpi/ic_launcher_round.webp | Bin 0 -> 1564 bytes .../res/mipmap-xhdpi/ic_launcher.webp | Bin 0 -> 1444 bytes .../res/mipmap-xhdpi/ic_launcher_round.webp | Bin 0 -> 3462 bytes .../res/mipmap-xxhdpi/ic_launcher.webp | Bin 0 -> 2226 bytes .../res/mipmap-xxhdpi/ic_launcher_round.webp | Bin 0 -> 5520 bytes .../res/mipmap-xxxhdpi/ic_launcher.webp | Bin 0 -> 2912 bytes .../res/mipmap-xxxhdpi/ic_launcher_round.webp | Bin 0 -> 7806 bytes .../src/androidMain/res/values/strings.xml | 4 + .../composeResources/values/strings.xml | 21 ++ .../kotlin/io/assessment/contacts/App.kt | 12 + .../io/assessment/contacts/MainViewModel.kt | 5 + .../contacts/contacts/data/ContactMapper.kt | 27 ++ .../data/ObserveContactsUseCaseImpl.kt | 17 ++ .../contacts/contacts/di/ContactsModule.kt | 14 + .../contacts/domain/ObserveContactsUseCase.kt | 7 + .../contacts/presentation/ContactVO.kt | 22 ++ .../contacts/presentation/ContactsScreen.kt | 200 ++++++++++++++ .../contacts/presentation/ContactsState.kt | 6 + .../presentation/ContactsViewModel.kt | 26 ++ .../navigation/ContactsNavDestinations.kt | 9 + .../contacts/core/data/database/ContactDao.kt | 39 +++ .../core/data/database/ContactEntity.kt | 16 ++ .../core/data/database/ContactsDatabase.kt | 17 ++ .../core/data/database/DatabaseSeeder.kt | 108 ++++++++ .../contacts/core/data/database/GroupDao.kt | 26 ++ .../core/data/database/GroupEntity.kt | 12 + .../core/data/datasource/ContactDataSource.kt | 23 ++ .../core/data/datasource/GroupDataSource.kt | 18 ++ .../data/repository/ContactRepositoryImpl.kt | 29 ++ .../data/repository/GroupRepositoryImpl.kt | 22 ++ .../contacts/core/domain/contact/Contact.kt | 10 + .../core/domain/contact/ContactError.kt | 9 + .../core/domain/contact/ContactRepository.kt | 9 + .../core/domain/contact/ContactStatus.kt | 7 + .../contacts/core/domain/group/Group.kt | 7 + .../contacts/core/domain/group/GroupError.kt | 9 + .../core/domain/group/GroupRepository.kt | 8 + .../contacts/core/domain/result/Error.kt | 3 + .../domain/result/FlowResultExtensions.kt | 60 +++++ .../contacts/core/domain/result/Result.kt | 77 ++++++ .../contacts/core/domain/usecase/UseCase.kt | 24 ++ .../core/presentation/ObserveAsEvents.kt | 27 ++ .../core/presentation/StateFlowExt.kt | 12 + .../assessment/contacts/designsystem/Color.kt | 63 +++++ .../contacts/designsystem/ExtendedColors.kt | 137 ++++++++++ .../assessment/contacts/designsystem/Theme.kt | 20 ++ .../assessment/contacts/designsystem/Type.kt | 115 ++++++++ .../assessment/contacts/di/CoreDataModule.kt | 33 +++ .../assessment/contacts/di/SharedAppModule.kt | 13 + .../contacts/groups/data/GroupMapper.kt | 20 ++ .../groups/data/ObserveGroupsUseCaseImpl.kt | 17 ++ .../contacts/groups/di/GroupsModule.kt | 14 + .../groups/domain/ObserveGroupsUseCase.kt | 7 + .../contacts/groups/presentation/GroupVO.kt | 17 ++ .../groups/presentation/GroupsScreen.kt | 140 ++++++++++ .../groups/presentation/GroupsState.kt | 6 + .../groups/presentation/GroupsViewModel.kt | 26 ++ .../navigation/GroupsNavDestinations.kt | 9 + .../contacts/navigation/BottomNavDestItem.kt | 41 +++ .../contacts/navigation/NavigationRoot.kt | 73 +++++ gradle.properties | 12 + gradle/libs.versions.toml | 77 ++++++ gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 43504 bytes gradle/wrapper/gradle-wrapper.properties | 7 + gradlew | 252 ++++++++++++++++++ gradlew.bat | 94 +++++++ settings.gradle.kts | 35 +++ 86 files changed, 3124 insertions(+) create mode 100644 .editorconfig create mode 100644 .gitattributes create mode 100644 .gitignore create mode 100644 .idea/codeStyles/Project.xml create mode 100644 .idea/codeStyles/codeStyleConfig.xml create mode 100644 README.md create mode 100644 build.gradle.kts create mode 100644 composeApp/build.gradle.kts create mode 100644 composeApp/schemas/io.assessment.contacts.core.data.database.ContactsDatabase/1.json create mode 100644 composeApp/src/androidMain/AndroidManifest.xml create mode 100644 composeApp/src/androidMain/kotlin/io/assessment/contacts/ContactsApplication.kt create mode 100644 composeApp/src/androidMain/kotlin/io/assessment/contacts/MainActivity.kt create mode 100644 composeApp/src/androidMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsScreenPreview.kt create mode 100644 composeApp/src/androidMain/kotlin/io/assessment/contacts/core/data/database/DatabaseFactory.android.kt create mode 100644 composeApp/src/androidMain/kotlin/io/assessment/contacts/di/DatabaseModule.kt create mode 100644 composeApp/src/androidMain/kotlin/io/assessment/contacts/groups/presentation/GroupsScreenPreview.kt create mode 100644 composeApp/src/androidMain/res/mipmap-hdpi/ic_launcher.webp create mode 100644 composeApp/src/androidMain/res/mipmap-hdpi/ic_launcher_round.webp create mode 100644 composeApp/src/androidMain/res/mipmap-mdpi/ic_launcher.webp create mode 100644 composeApp/src/androidMain/res/mipmap-mdpi/ic_launcher_round.webp create mode 100644 composeApp/src/androidMain/res/mipmap-xhdpi/ic_launcher.webp create mode 100644 composeApp/src/androidMain/res/mipmap-xhdpi/ic_launcher_round.webp create mode 100644 composeApp/src/androidMain/res/mipmap-xxhdpi/ic_launcher.webp create mode 100644 composeApp/src/androidMain/res/mipmap-xxhdpi/ic_launcher_round.webp create mode 100644 composeApp/src/androidMain/res/mipmap-xxxhdpi/ic_launcher.webp create mode 100644 composeApp/src/androidMain/res/mipmap-xxxhdpi/ic_launcher_round.webp create mode 100644 composeApp/src/androidMain/res/values/strings.xml create mode 100644 composeApp/src/commonMain/composeResources/values/strings.xml create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/App.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/MainViewModel.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/data/ContactMapper.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/data/ObserveContactsUseCaseImpl.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/di/ContactsModule.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/domain/ObserveContactsUseCase.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactVO.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsScreen.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsState.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsViewModel.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/navigation/ContactsNavDestinations.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactDao.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactEntity.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactsDatabase.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/DatabaseSeeder.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/GroupDao.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/GroupEntity.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/datasource/ContactDataSource.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/datasource/GroupDataSource.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/repository/ContactRepositoryImpl.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/repository/GroupRepositoryImpl.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/Contact.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactError.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactRepository.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactStatus.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/Group.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/GroupError.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/GroupRepository.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/Error.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/FlowResultExtensions.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/Result.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/usecase/UseCase.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/presentation/ObserveAsEvents.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/core/presentation/StateFlowExt.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Color.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/ExtendedColors.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Theme.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Type.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/di/CoreDataModule.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/di/SharedAppModule.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/data/GroupMapper.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/data/ObserveGroupsUseCaseImpl.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/di/GroupsModule.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/domain/ObserveGroupsUseCase.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupVO.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsScreen.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsState.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsViewModel.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/navigation/GroupsNavDestinations.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/navigation/BottomNavDestItem.kt create mode 100644 composeApp/src/commonMain/kotlin/io/assessment/contacts/navigation/NavigationRoot.kt create mode 100644 gradle.properties create mode 100644 gradle/libs.versions.toml create mode 100644 gradle/wrapper/gradle-wrapper.jar create mode 100644 gradle/wrapper/gradle-wrapper.properties create mode 100755 gradlew create mode 100644 gradlew.bat create mode 100644 settings.gradle.kts diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 0000000..3c19c17 --- /dev/null +++ b/.editorconfig @@ -0,0 +1,21 @@ +root = true + +[*] +end_of_line = lf +insert_final_newline = true +trim_trailing_whitespace = true +charset = utf-8 +indent_style = space +indent_size = 4 + +[*.{kt,kts}] +indent_size = 4 + +[*.{xml,json,yaml,yml,toml}] +indent_size = 4 + +[*.md] +trim_trailing_whitespace = false + +[*.bat] +end_of_line = crlf diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..9f95d99 --- /dev/null +++ b/.gitattributes @@ -0,0 +1,3 @@ +* text=auto eol=lf +*.bat text eol=crlf +*.jar binary diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..ef14ec5 --- /dev/null +++ b/.gitignore @@ -0,0 +1,126 @@ +# ---> Kotlin +# Compiled class file +*.class + +# Log file +*.log + +# BlueJ files +*.ctxt + +# Mobile Tools for Java (J2ME) +.mtj.tmp/ + +# Package Files # +*.jar +!gradle/wrapper/gradle-wrapper.jar +*.war +*.nar +*.ear +*.zip +*.tar.gz +*.rar + +# virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml +hs_err_pid* +replay_pid* + +# Kotlin Gradle plugin data, see https://kotlinlang.org/docs/whatsnew20.html#new-directory-for-kotlin-data-in-gradle-projects +.kotlin/ +# ---> Swift +# Xcode +# +# gitignore contributors: remember to update Global/Xcode.gitignore, Objective-C.gitignore & Swift.gitignore + +## User settings +xcuserdata/ + +## Obj-C/Swift specific +*.hmap + +## App packaging +*.ipa +*.dSYM.zip +*.dSYM + +## Playgrounds +timeline.xctimeline +playground.xcworkspace + +# Swift Package Manager +# +# Add this line if you want to avoid checking in source code from Swift Package Manager dependencies. +# Packages/ +# Package.pins +# Package.resolved +# *.xcodeproj +# +# Xcode automatically generates this directory with a .xcworkspacedata file and xcuserdata +# hence it is not needed unless you have added a package configuration file to your project +# .swiftpm + +.build/ + +# CocoaPods +# +# We recommend against adding the Pods directory to your .gitignore. However +# you should judge for yourself, the pros and cons are mentioned at: +# https://guides.cocoapods.org/using/using-cocoapods.html#should-i-check-the-pods-directory-into-source-control +# +# Pods/ +# +# Add this line if you want to avoid checking in source code from the Xcode workspace +# *.xcworkspace + +# Carthage +# +# Add this line if you want to avoid checking in source code from Carthage dependencies. +# Carthage/Checkouts + +Carthage/Build/ + +# fastlane +# +# It is recommended to not store the screenshots in the git repo. +# Instead, use fastlane to re-generate the screenshots whenever they are needed. +# For more information about the recommended setup visit: +# https://docs.fastlane.tools/best-practices/source-control/#source-control + +fastlane/report.xml +fastlane/Preview.html +fastlane/screenshots/**/*.png +fastlane/test_output + +*.iml +.kotlin +.gradle +**/build/ +xcuserdata +!src/**/build/ +local.properties +.idea/* +!.idea/codeStyles/ +.DS_Store +captures +.externalNativeBuild +.cxx +*.xcodeproj/* +!*.xcodeproj/project.pbxproj +!*.xcodeproj/xcshareddata/ +!*.xcodeproj/project.xcworkspace/ +!*.xcworkspace/contents.xcworkspacedata +**/xcshareddata/WorkspaceSettings.xcsettings + +# Database files +*.sqlite +*.sqlite-wal +*.sqlite-shm +*.db +*.db-wal +*.db-shm + +keystore.properties +release-keystore.jks +release-keystore.jks.base64 +/composeApp/src/androidMain/res/xml/locales_config.xml + diff --git a/.idea/codeStyles/Project.xml b/.idea/codeStyles/Project.xml new file mode 100644 index 0000000..96bbd61 --- /dev/null +++ b/.idea/codeStyles/Project.xml @@ -0,0 +1,175 @@ + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/.idea/codeStyles/codeStyleConfig.xml b/.idea/codeStyles/codeStyleConfig.xml new file mode 100644 index 0000000..79ee123 --- /dev/null +++ b/.idea/codeStyles/codeStyleConfig.xml @@ -0,0 +1,5 @@ + + + + \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..1cd5617 --- /dev/null +++ b/README.md @@ -0,0 +1,87 @@ +# Contacts Assessment App + +A simplified Kotlin Multiplatform (KMP) contacts management app designed for technical interviews. +This app mirrors GART architecture patterns and contains intentional bugs for candidates to identify +and fix. + +## Project Overview + +This is an Android-focused KMP app that displays contacts and groups with the following features: + +- Bottom navigation with Groups and Contacts tabs +- Status badges for contacts (Active, Pending, Inactive) +- Nostr-style npub identifiers for contacts +- Avatar images loaded from network +- Room database with reactive Flow-based queries +- Clean Architecture with domain, data, and presentation layers +- ViewObjects (VOs) separating domain from presentation +- Dependency injection with Koin + +## Getting Started + +### Prerequisites + +- Android Studio Ladybug (2024.2.1) or newer +- Android SDK 26+ +- JDK 17 is **auto-provisioned** by the Gradle toolchain — no manual install needed + +### Build & Run + +```bash +# Build debug APK +./gradlew assembleDebug + +# On Windows, use: +gradlew.bat assembleDebug + +# Install on connected device +adb install -r composeApp/build/outputs/apk/debug/composeApp-debug.apk + +# Or run directly from Android Studio +``` + +## Troubleshooting + +| Problem | Fix | +|---------|-----| +| `Unsupported class file major version` | The toolchain should handle this automatically. If not, ensure JDK 17+ is installed. | +| Out of memory during build | Increase heap in `gradle.properties`: `org.gradle.jvmargs=-Xmx3072M` | +| `SDK location not found` | Create a `local.properties` file with `sdk.dir=/path/to/your/Android/Sdk` | +| Gradle sync fails in Android Studio | File -> Invalidate Caches -> Restart, then re-sync. | + +## Architecture + +``` +composeApp/src/ +├── androidMain/kotlin/io/assessment/contacts/ +│ ├── ContactsApplication.kt +│ ├── MainActivity.kt +│ └── di/DatabaseModule.kt # Android-specific Room setup +└── commonMain/kotlin/io/assessment/contacts/ + ├── App.kt + ├── MainViewModel.kt + ├── core/ + │ ├── data/ + │ │ ├── database/ # Room entities, DAOs, DatabaseSeeder + │ │ ├── datasource/ # Data sources + │ │ └── repository/ # Repository implementations + │ ├── domain/ + │ │ ├── contact/ # Contact, ContactStatus, ContactError + │ │ ├── group/ # Group, GroupError + │ │ ├── result/ # Result type utilities + │ │ └── usecase/ # Use case interfaces + │ └── presentation/ # Shared presentation utilities + ├── contacts/ + │ ├── data/ # ContactMapper + │ ├── domain/ # ObserveContactsUseCase + │ ├── di/ # ContactsModule + │ └── presentation/ # ViewModel, Screen, State, ContactVO + ├── groups/ + │ ├── data/ # GroupMapper + │ ├── domain/ # ObserveGroupsUseCase + │ ├── di/ # GroupsModule + │ └── presentation/ # ViewModel, Screen, State, GroupVO + ├── designsystem/ # Theme, Colors, Typography + ├── navigation/ # NavigationRoot, BottomNavDestItem + └── di/ # App-level DI modules +``` diff --git a/build.gradle.kts b/build.gradle.kts new file mode 100644 index 0000000..8f973f0 --- /dev/null +++ b/build.gradle.kts @@ -0,0 +1,7 @@ +plugins { + alias(libs.plugins.androidApplication) apply false + alias(libs.plugins.composeMultiplatform) apply false + alias(libs.plugins.composeCompiler) apply false + alias(libs.plugins.kotlinMultiplatform) apply false + alias(libs.plugins.kotlinSerialization) apply false +} diff --git a/composeApp/build.gradle.kts b/composeApp/build.gradle.kts new file mode 100644 index 0000000..ee97208 --- /dev/null +++ b/composeApp/build.gradle.kts @@ -0,0 +1,105 @@ +plugins { + alias(libs.plugins.kotlinMultiplatform) + alias(libs.plugins.androidApplication) + alias(libs.plugins.composeMultiplatform) + alias(libs.plugins.composeCompiler) + alias(libs.plugins.kotlinSerialization) + alias(libs.plugins.ksp) + alias(libs.plugins.room) +} + +kotlin { + jvmToolchain(17) + + androidTarget() + + sourceSets { + androidMain.dependencies { + implementation(libs.compose.ui.tooling.preview) + implementation(libs.androidx.activity.compose) + implementation(libs.koin.android) + implementation(libs.ktor.client.okhttp) + } + + commonMain.dependencies { + implementation(libs.compose.runtime) + implementation(libs.compose.foundation) + implementation(compose.material3) + implementation(compose.materialIconsExtended) + implementation(compose.components.resources) + implementation(libs.compose.ui) + implementation(libs.compose.ui.tooling.preview) + + api(libs.koin.core) + implementation(libs.koin.compose) + implementation(libs.koin.compose.viewmodel) + implementation(libs.androidx.lifecycle.runtime.compose) + implementation(libs.androidx.lifecycle.viewmodel.compose) + implementation(libs.androidx.navigation.compose) + + implementation(libs.room.runtime) + implementation(libs.sqlite.bundled) + + implementation(libs.kotlinx.coroutines.core) + + implementation(libs.coil.compose) + implementation(libs.coil.network.ktor3) + } + + commonTest.dependencies { + implementation(libs.kotlin.test) + } + } +} + +android { + namespace = "io.assessment.contacts" + compileSdk = libs.versions.android.compileSdk.get().toInt() + + buildFeatures { + buildConfig = true + } + + lint { + abortOnError = false + } + + defaultConfig { + applicationId = "io.assessment.contacts" + minSdk = libs.versions.android.minSdk.get().toInt() + targetSdk = libs.versions.android.targetSdk.get().toInt() + versionCode = 1 + versionName = "1.0.0" + } + + packaging { + resources { + excludes += "/META-INF/{AL2.0,LGPL2.1}" + } + } + + buildTypes { + getByName("debug") { + isDebuggable = true + } + getByName("release") { + isMinifyEnabled = false + } + } + +} + +dependencies { + add("kspAndroid", libs.room.compiler) + debugImplementation(libs.compose.ui.tooling) +} + +room { + schemaDirectory("$projectDir/schemas") +} + +tasks.withType().configureEach { + compilerOptions { + freeCompilerArgs.add("-Xexpect-actual-classes") + } +} diff --git a/composeApp/schemas/io.assessment.contacts.core.data.database.ContactsDatabase/1.json b/composeApp/schemas/io.assessment.contacts.core.data.database.ContactsDatabase/1.json new file mode 100644 index 0000000..64bffaf --- /dev/null +++ b/composeApp/schemas/io.assessment.contacts.core.data.database.ContactsDatabase/1.json @@ -0,0 +1,89 @@ +{ + "formatVersion": 1, + "database": { + "version": 1, + "identityHash": "c812bdba76dc08f1c83d24558a84ae29", + "entities": [ + { + "tableName": "contacts", + "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `name` TEXT NOT NULL, `npub` TEXT NOT NULL, `avatarUrl` TEXT, `status` TEXT NOT NULL, `groupId` TEXT, PRIMARY KEY(`id`))", + "fields": [ + { + "fieldPath": "id", + "columnName": "id", + "affinity": "TEXT", + "notNull": true + }, + { + "fieldPath": "name", + "columnName": "name", + "affinity": "TEXT", + "notNull": true + }, + { + "fieldPath": "npub", + "columnName": "npub", + "affinity": "TEXT", + "notNull": true + }, + { + "fieldPath": "avatarUrl", + "columnName": "avatarUrl", + "affinity": "TEXT" + }, + { + "fieldPath": "status", + "columnName": "status", + "affinity": "TEXT", + "notNull": true + }, + { + "fieldPath": "groupId", + "columnName": "groupId", + "affinity": "TEXT" + } + ], + "primaryKey": { + "autoGenerate": false, + "columnNames": [ + "id" + ] + } + }, + { + "tableName": "groups", + "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, `name` TEXT NOT NULL, `memberCount` INTEGER NOT NULL, PRIMARY KEY(`id`))", + "fields": [ + { + "fieldPath": "id", + "columnName": "id", + "affinity": "TEXT", + "notNull": true + }, + { + "fieldPath": "name", + "columnName": "name", + "affinity": "TEXT", + "notNull": true + }, + { + "fieldPath": "memberCount", + "columnName": "memberCount", + "affinity": "INTEGER", + "notNull": true + } + ], + "primaryKey": { + "autoGenerate": false, + "columnNames": [ + "id" + ] + } + } + ], + "setupQueries": [ + "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", + "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, 'c812bdba76dc08f1c83d24558a84ae29')" + ] + } +} \ No newline at end of file diff --git a/composeApp/src/androidMain/AndroidManifest.xml b/composeApp/src/androidMain/AndroidManifest.xml new file mode 100644 index 0000000..212f678 --- /dev/null +++ b/composeApp/src/androidMain/AndroidManifest.xml @@ -0,0 +1,25 @@ + + + + + + + + + + + + + + + diff --git a/composeApp/src/androidMain/kotlin/io/assessment/contacts/ContactsApplication.kt b/composeApp/src/androidMain/kotlin/io/assessment/contacts/ContactsApplication.kt new file mode 100644 index 0000000..1aa9212 --- /dev/null +++ b/composeApp/src/androidMain/kotlin/io/assessment/contacts/ContactsApplication.kt @@ -0,0 +1,34 @@ +package io.assessment.contacts + +import android.app.Application +import io.assessment.contacts.core.data.database.DatabaseSeeder +import io.assessment.contacts.di.databaseModule +import io.assessment.contacts.di.sharedAppModule +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.SupervisorJob +import kotlinx.coroutines.launch +import org.koin.android.ext.android.inject +import org.koin.android.ext.koin.androidContext +import org.koin.android.ext.koin.androidLogger +import org.koin.core.context.startKoin + +class ContactsApplication : Application() { + + private val applicationScope = CoroutineScope(SupervisorJob() + Dispatchers.IO) + private val databaseSeeder: DatabaseSeeder by inject() + + override fun onCreate() { + super.onCreate() + + startKoin { + androidContext(this@ContactsApplication) + androidLogger() + modules(databaseModule, sharedAppModule) + } + + applicationScope.launch { + databaseSeeder.seedIfEmpty() + } + } +} diff --git a/composeApp/src/androidMain/kotlin/io/assessment/contacts/MainActivity.kt b/composeApp/src/androidMain/kotlin/io/assessment/contacts/MainActivity.kt new file mode 100644 index 0000000..de5e7a7 --- /dev/null +++ b/composeApp/src/androidMain/kotlin/io/assessment/contacts/MainActivity.kt @@ -0,0 +1,18 @@ +package io.assessment.contacts + +import android.os.Bundle +import androidx.activity.ComponentActivity +import androidx.activity.compose.setContent +import androidx.activity.enableEdgeToEdge + +class MainActivity : ComponentActivity() { + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + enableEdgeToEdge() + + setContent { + App() + } + } +} diff --git a/composeApp/src/androidMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsScreenPreview.kt b/composeApp/src/androidMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsScreenPreview.kt new file mode 100644 index 0000000..bd20f1e --- /dev/null +++ b/composeApp/src/androidMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsScreenPreview.kt @@ -0,0 +1,149 @@ +package io.assessment.contacts.contacts.presentation + +import androidx.compose.runtime.Composable +import androidx.compose.ui.tooling.preview.Preview +import io.assessment.contacts.core.domain.contact.ContactStatus +import io.assessment.contacts.designsystem.ContactsTheme + +private val sampleContacts = listOf( + ContactVO( + id = "1", + name = "Alice Johnson", + npub = "npub1a2b3c4d5e6f7g8h9i0j1k2l3m4n5o6p7q8r9s0", + avatarUrl = null, + status = ContactStatus.ACTIVE, + ), + ContactVO( + id = "2", + name = "Bob Smith", + npub = "npub1z9y8x7w6v5u4t3s2r1q0p9o8n7m6l5k4j3i2h", + avatarUrl = null, + status = ContactStatus.PENDING, + ), + ContactVO( + id = "3", + name = "Charlie Brown", + npub = "npub1m2n3o4p5q6r7s8t9u0v1w2x3y4z5a6b7c8d9e", + avatarUrl = null, + status = ContactStatus.INACTIVE, + ), +) + +@Preview +@Composable +private fun ContactsScreenContentPreview() { + ContactsTheme { + ContactsScreenContent( + state = ContactsState( + contacts = sampleContacts, + isLoading = false, + ), + ) + } +} + +@Preview +@Composable +private fun ContactsScreenLoadingPreview() { + ContactsTheme { + ContactsScreenContent( + state = ContactsState(isLoading = true), + ) + } +} + +@Preview +@Composable +private fun ContactsScreenEmptyPreview() { + ContactsTheme { + ContactsScreenContent( + state = ContactsState( + contacts = emptyList(), + isLoading = false, + ), + ) + } +} + +@Preview +@Composable +private fun ContactListItemActivePreview() { + ContactsTheme { + ContactListItem( + contact = ContactVO( + id = "1", + name = "Alice Johnson", + npub = "npub1a2b3c4d5e6f7g8h9i0j1k2l3m4n5o6p7q8r9s0", + avatarUrl = null, + status = ContactStatus.ACTIVE, + ), + ) + } +} + +@Preview +@Composable +private fun ContactListItemPendingPreview() { + ContactsTheme { + ContactListItem( + contact = ContactVO( + id = "2", + name = "Bob Smith", + npub = "npub1z9y8x7w6v5u4t3s2r1q0p9o8n7m6l5k4j3i2h", + avatarUrl = null, + status = ContactStatus.PENDING, + ), + ) + } +} + +@Preview +@Composable +private fun ContactListItemInactivePreview() { + ContactsTheme { + ContactListItem( + contact = ContactVO( + id = "3", + name = "Charlie Brown", + npub = "npub1m2n3o4p5q6r7s8t9u0v1w2x3y4z5a6b7c8d9e", + avatarUrl = null, + status = ContactStatus.INACTIVE, + ), + ) + } +} + +@Preview +@Composable +private fun ContactAvatarWithoutImagePreview() { + ContactsTheme { + ContactAvatar( + avatarUrl = null, + name = "Alice Johnson", + ) + } +} + +@Preview +@Composable +private fun StatusBadgeActivePreview() { + ContactsTheme { + StatusBadge(status = ContactStatus.ACTIVE) + } +} + +@Preview +@Composable +private fun StatusBadgePendingPreview() { + ContactsTheme { + StatusBadge(status = ContactStatus.PENDING) + } +} + +@Preview +@Composable +private fun StatusBadgeInactivePreview() { + ContactsTheme { + StatusBadge(status = ContactStatus.INACTIVE) + } +} diff --git a/composeApp/src/androidMain/kotlin/io/assessment/contacts/core/data/database/DatabaseFactory.android.kt b/composeApp/src/androidMain/kotlin/io/assessment/contacts/core/data/database/DatabaseFactory.android.kt new file mode 100644 index 0000000..793eed1 --- /dev/null +++ b/composeApp/src/androidMain/kotlin/io/assessment/contacts/core/data/database/DatabaseFactory.android.kt @@ -0,0 +1,19 @@ +package io.assessment.contacts.core.data.database + +import android.content.Context +import androidx.room.Room +import androidx.room.RoomDatabase + +class DatabaseFactory( + private val context: Context, +) { + fun create(): RoomDatabase.Builder { + val appContext = context.applicationContext + val dbFile = appContext.getDatabasePath("contacts.db") + return Room.databaseBuilder( + context = appContext, + klass = ContactsDatabase::class.java, + name = dbFile.absolutePath, + ) + } +} diff --git a/composeApp/src/androidMain/kotlin/io/assessment/contacts/di/DatabaseModule.kt b/composeApp/src/androidMain/kotlin/io/assessment/contacts/di/DatabaseModule.kt new file mode 100644 index 0000000..d6ad7e5 --- /dev/null +++ b/composeApp/src/androidMain/kotlin/io/assessment/contacts/di/DatabaseModule.kt @@ -0,0 +1,18 @@ +package io.assessment.contacts.di + +import androidx.sqlite.driver.bundled.BundledSQLiteDriver +import io.assessment.contacts.core.data.database.ContactsDatabase +import io.assessment.contacts.core.data.database.DatabaseFactory +import kotlinx.coroutines.Dispatchers +import org.koin.android.ext.koin.androidContext +import org.koin.dsl.module + +val databaseModule = module { + single { + DatabaseFactory(androidContext()) + .create() + .setDriver(BundledSQLiteDriver()) + .setQueryCoroutineContext(Dispatchers.IO) + .build() + } +} diff --git a/composeApp/src/androidMain/kotlin/io/assessment/contacts/groups/presentation/GroupsScreenPreview.kt b/composeApp/src/androidMain/kotlin/io/assessment/contacts/groups/presentation/GroupsScreenPreview.kt new file mode 100644 index 0000000..fa1776f --- /dev/null +++ b/composeApp/src/androidMain/kotlin/io/assessment/contacts/groups/presentation/GroupsScreenPreview.kt @@ -0,0 +1,73 @@ +package io.assessment.contacts.groups.presentation + +import androidx.compose.runtime.Composable +import androidx.compose.ui.tooling.preview.Preview +import io.assessment.contacts.designsystem.ContactsTheme + +private val sampleGroups = listOf( + GroupVO( + id = "1", + name = "Engineering Team", + memberCount = 12, + ), + GroupVO( + id = "2", + name = "Design Team", + memberCount = 5, + ), + GroupVO( + id = "3", + name = "Marketing", + memberCount = 8, + ), +) + +@Preview +@Composable +private fun GroupsScreenContentPreview() { + ContactsTheme { + GroupsScreenContent( + state = GroupsState( + groups = sampleGroups, + isLoading = false, + ), + ) + } +} + +@Preview +@Composable +private fun GroupsScreenLoadingPreview() { + ContactsTheme { + GroupsScreenContent( + state = GroupsState(isLoading = true), + ) + } +} + +@Preview +@Composable +private fun GroupsScreenEmptyPreview() { + ContactsTheme { + GroupsScreenContent( + state = GroupsState( + groups = emptyList(), + isLoading = false, + ), + ) + } +} + +@Preview +@Composable +private fun GroupListItemPreview() { + ContactsTheme { + GroupListItem( + group = GroupVO( + id = "1", + name = "Engineering Team", + memberCount = 12, + ), + ) + } +} diff --git a/composeApp/src/androidMain/res/mipmap-hdpi/ic_launcher.webp b/composeApp/src/androidMain/res/mipmap-hdpi/ic_launcher.webp new file mode 100644 index 0000000000000000000000000000000000000000..0834f207dc34b5f828795e34d158779b9e534bea GIT binary patch literal 968 zcmV;(12_CqNk&G%0{{S5MM6+kP&iDp0{{RoN5Byf?I}iHZCkZ6_xWzn5O_M1R* zcWwZJAc*`~nV^Fp2nu>=BWT{ZZ6K8-Nkoz)BAKZqYm#J5a!vBge}Nm)0G=dT%ZouGu8$5}(JkTuB~a!qm#Su>eatP`%sn&f|vb&_X3WN9=T_>gwy3YvR__n%Zu?jKJ9y2FTNFg# zUy|2{4|%!Kh9Z%QYGrK!X}I&(Lh5^^n%v_1_36snYxcT&-MAs)!MdO+qP}B&Yo6c-MK)>BWqug}splz8 zOcshP#dMHZZ>m_ZG{`%|h}04yU=dQ&vTKzFCX>4?VjHubN;cDbU^1VXvY9`E^HSGv zAI@_v58+g`D~dT413*s5_xS2rechyI|M}}K^C=W0bZF zbB40pgR)exDHNO$YvBB7mAX~1%{14EM==Sn?|p(D^`ju7dlwg#aqvySa1M*c zHAJ+N-8l`25H4jcAOF2XIEI`QRBVBW4ILx)uP45R-zbN3v3TPhVw$p12|P$o`A`EtEGmdXHbeoD5B_T zIiJlNSuD(K>M}8%T9Dr21?g0jukFk%SmYPW|F3ehGKEL|%|FYp3i~fxr|M7dKE4zwMhBx`;VW(>qb<#ngCQfQvm>v6y_fQ literal 0 HcmV?d00001 diff --git a/composeApp/src/androidMain/res/mipmap-hdpi/ic_launcher_round.webp b/composeApp/src/androidMain/res/mipmap-hdpi/ic_launcher_round.webp new file mode 100644 index 0000000000000000000000000000000000000000..3d1a04f1976ea46756f63a079ec135c2e62f556a GIT binary patch literal 2470 zcmV;X30d}1Nk&GV2><|BMM6+kP&iDH2><{uN5Byf)rNw$Z5W3?>>UgdF#+7t@9~QJn*@@!c-v1VZR5D= zrfqK+xNX}=PS;&-=Q_T?vci8O|^cj)vv|oR9jbV`}=G2ZSl7hulvgp$$%23 zNPS9|T!%x=%g3I<1RD3{NtdMw zmB8&hGlTzMka)eHb1f6~+C#)SD72#A+bI}nQTFwiH-?YjgH58(#_LWX>R^j{tk9XvH6G*M#V=Y@J5zv$l;1zN=!q}@DUE^fc0F)#>o8LESO{-+6 z$6hQ#wvD`X{u-CB@p(zX`7D%(viila&iO)b0J<6fT2p&zs5Cjh@Ai?|3K1+8$@gXC z8in0tidF5@yKeb9PpB{;-<$S4Zr-D1NN|O5Eh8+Q@P!uc8?Pe)DMI;1Gvr##M zMxN!cCw0@iziNtv2w)_+pQY#=mJtZS1U*y^;*fnktfO1#FTi(2K80qY`t_QINC_5- z;rkq-(;-4|6GPV->Le@cM-63m=ta{#p`&5b9$!^qH#vp^(MvVXMd=VMYtyh0t+GHW z+sX)cpt#M9&M{()9DeyVQ)$@)r|(QKi0Y$SCl&ZOS1Ny_FViES({K<>*6A;cX0|f8 z_t}maeEDpgz8fAJy4@@rO=-CRV~y(NQ$G*J8MU4T;;dTN>WZ*kgv;JW|JLg=X7KT| z^+D1cl_$6X6NXu z-?SHrC)&sa&K@GZi104Mh#89wJ!rh%VnZG(Hx`(qXyjjru+&KwH-Ksy1s$L$`F>gJ zgH2@yxbz#OQzTZkL5AUcw*P=iQKN(bTwG`DC_JfUuK{bcYdurN%?{^*ICZfouRa`+SM!n+O!0$q6KQ4&sz|DFSep zBo{{a%6UUCw)p-8=;1!5lH7#tdAe0fXJB9{J056uQL6>F*wkP%n^i{=y;wk1R(k23 z;g~n3w^t0O;ph^QMSVn|J^4@{0JWknCIAK3w-&1nl;88rQh;WrQvU@)H6n$Ozp(EN z4o4PUfa~vJukkbE+ZFjk`mAFs>}Eu;wk$n!y)E1xI*x^2JfI;n-7Jy1m7EIEbDf2v z&|Gd~EY*kXcMk!L4^#O7pfI>qQZ(<3-?lOn^-_Q50j?N<5eSuJNz$QzuB@|(8Im|k z0QB5KpG#Ch53`qmYuQX!-1)$B%53<40{S>g;1nI8Jh~}cli*)Hs$bN#Jzm4S@N#%g zaf3%VP74tcFmMxlt`htcf6uo$jKMWyT)@+y%K-Hu`_-HMmsxorE@GPB4+zrCR{S?2 z9iqCrzh1ipJ&Vk5<@fEv8$?*n67CQIz3jt|X*{SVd6Ri}_7QNU?+v*N#$AwX7(VUn z6E2eHOaKa9lmQDj();K=^tH@*4&bnpjHTS`9NfF?mTq9IZR^-Ijh8yM}9etJ=a zud$Z~V2X0y`R_g)sD($BWMz4MOb7Y*KB0`y_Z$BpH`q=UFa(_3{pYmIZ7CLwlC8MN z?>PrJ1BTd6bwidhnkb-#K43FgFE;z$D{(JW^`J`}U~_d*E=U3_ zx@iI?faCCJ<$3SEpJ)=2T4Y(zUH)3PpHe*eksVRFSscA_LL~=LM>ha2!>FRsmmav; zrX&|Mgp^d1mTj#*QWyGOx66Fs4X`*bMk^7(pqdqy+y+jIE7Dtk)_06SvXGynhq literal 0 HcmV?d00001 diff --git a/composeApp/src/androidMain/res/mipmap-mdpi/ic_launcher.webp b/composeApp/src/androidMain/res/mipmap-mdpi/ic_launcher.webp new file mode 100644 index 0000000000000000000000000000000000000000..7b5013bacb0b331b2302ac0b6100174618de7df2 GIT binary patch literal 748 zcmViTPe|-lhjlGid2(P z>iEBnL-3S8P$|`&Kdn$&qN+PzqN=!6#dtk!w_1}SiXobeTD3yN5KYEierkof%15bM zMWR~UEk-SnK=kGmL!=UkKmvqFLuRsHJzY4NpBW)qf4b(m1&?@A$TOC_tPQ>28^6293 zt=?~|zWq|jrL`t}-hsy=4ox?XtFH+-I@P$Aue{54Wub9T>|opDt=XGT9bA<#sygR3 z*J7ipGfxt5Z@rZ`!t`>MwbT=U^HYqf?qhlpvV!ajvm{`?wcrYX=X;I4pKG0^%g;Q# zwQHJHd6w%~h2^^F8P~%VIaHk>E?k)Fw&4li%fnSJ%Wi-V4>mc>J4X_5XMMzs<0J!D zmV_jBK9z#gRo0sO5c6oGQJry_@4 ei_1Q|d(RwyEGteS+J7l2WrzB}Ihkhn!DI;_`f_jp literal 0 HcmV?d00001 diff --git a/composeApp/src/androidMain/res/mipmap-mdpi/ic_launcher_round.webp b/composeApp/src/androidMain/res/mipmap-mdpi/ic_launcher_round.webp new file mode 100644 index 0000000000000000000000000000000000000000..df591b7f7232f47e81071d594d73063254b327d2 GIT binary patch literal 1564 zcmV+%2IKisNk&E#1^@t8MM6+kP&iBn1^@srFTe{B)rLSIIe+dQB4PqsD!6DR*tV@k z%Ed|Y-^FzbjEgZaU8Yi!pPv~#?8B-QtL@?1lC)KO8Y#(wTy5Ji=P6-=0<_A>o&`AB z8C%=N*tTaryzd*g@Hg=HG?3d!awG@Y8LQR34z6U|rmc&;@3Ul0KI%hI)+elHed`z?V207Zc9ecC{+qTo&Y;B`UI%fxI&Ca%s%C>DggXeqRRZRaM z@9~sV%>w>5spfzp006;gwr$(Cm6mPW$!D{~5S#q{z#>2d~gV(Zj_z zK3ufXoXy9k)sn)C2M5uC_r8gz|1XsrhR)j2t`3YJOSk0nZYZB6%IBs125*hjIb}EG zo_=q?`Ty%=zgz<@s=7;AETwr;Z{*h97LiSo<$+O=PP}T4I0hGJq2YJEv~mx7_-yGYs%{PY^_dN)HIU0pTG0{euj%vZbHeHp&VE z+TMIcMoPiMMD((_T=O~GqdD;&paAPMY)%VLpaiyibFa+$pRP)4$>w|gdes2;P#pHr z?~of45GOPuL7>Fa1eaWv_CI=Q*X4WW6911&1$M{~;IRz8l%#sdw&Y`v`k_)*3Koi|728?G&P!Y?WNo& zd{u1*Z4b)+OzN-uO+Uf`FYP}GBkkQ>KcuaCBS4R~M=!p!I1IZaDxH=F>d|lyg9B51 z)nCJ~y(c+ktjbpgezcw+mirEX9Ws*=%Iz`hkv!0(-i;TD|AU8pGfha;eH=5CD-%eFc!+ zp*e_U-Bx`h+dj&KL&Qmq4| zwCyi&@5?2HqAn?ZzD5VgPAKs|jucCYr5duE6))l-4q%jqw034i@%JmS-W`lz7T9XM zM7UFP<<2ogTR*C51PG_zN@&AUQ6=~NBwx;-!K4&pl=TwTkY2BH8i(T;Kt;+sDqKw7 z^Iw7RB^JsbGsj)36M(v64SzVSj?!HL(%Dbpkk<1RF6TKUG-mZtf;9G%tO^~WhN=!H zfOL9a{bVU1$}^5w5*3o`OLH4NaefL=b$%l>ffBP`D*u{fuV;KbZGH0yfC6-~a1+#_ zI}PWwo234x0J;D=_0LlyfL;=4bk-e^V#AmnbfQA?-Z(&i1TUg~+YvxJySy|@Yu~mM z(wH4AnKbvEovmf3FspvVIKlYpN|9N*BdWoc4rpV%uRy(iuw=6=+0oUdMUR_l$UF`& z0YC;|;!aoN3Z01x)^*i-&Z_I|{GB^lEx>ehE%0|_J_0Di!#v$nlEtB^mFnaiGg zH$MLfFz$9yLrea-OaPMI&I$%uULNCmc_O#v_s1(c-G@p1s3mRTMet3qHrWM;0VJBY OyOTAzD7;AVK@v75= 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 literal 0 HcmV?d00001 diff --git a/composeApp/src/androidMain/res/mipmap-xxhdpi/ic_launcher.webp b/composeApp/src/androidMain/res/mipmap-xxhdpi/ic_launcher.webp new file mode 100644 index 0000000000000000000000000000000000000000..7c4a06dda47eeba6338c9fa7f10e334de7a8d8cc GIT binary patch literal 2226 zcmV;j2u=4=Nk&Gh2mkZJ35X?EM2EA|^oNBycs3 zpCEPsM+PH_XH~^-!?`cT{y#@jew%>as{Rx1f8116Q#E9$_we(u=@^aOQ}EW3b4GKB zJws_SsMdP|Gzz&7nJt1g=Rn!mo~zO^<{qHL4+uQ8H#NxLTP zgS0==el>sP!IJ|66I)iGY`O1hA?=hjgg_F6#~OfqlEDcNVFlN#)t04|Etv*GAc;~k z{gcEEI3i7hqvws-vZTcB2%*V|&RpSUPB2M-fxZTIcXiuD$}Nj0Qqs;slQZZx7eDM0 z&6M_*j06zkuO>q3j7l2CZ-MV`%KwzH2oTe6d!|Y0C-9TXfCSCN5^j0S|Fk6Ch7lrS z4cf>AjBxz2N7Q_+{+75!(`iSPq))e5gw8hyes_Jb0r~0g#bJ}=7hjc5O+zLg_REyy+*C~8+%k+2bvvss01%QTdAQ`DlN5zS;W>Sq+YDb@>4$1l4jYSyTh z{-ptmZh2{lTp<${4_VQnyh`JX4pSCy7M81jrNwqjB_L+fr&ht%d#iSlr%Gx_@Sw z{?8p8R@`ex0kgr(1Asaf(Tt~-T zARPZlJV_d8D!C$|yH1z!=XGa^1}!^KX+h;jnu z#k^{<kXdqRj+v*sS zHbVQADd!A)j3I8j50JLdY*Bw2NOD8(V@Ot)BPXv(Gp$}skhqoWF|0@XlV)%PZHb7a z{qHeY&>mB*6CX59+oz$Zrl`h|B|st!M72T@(Rc-WMqZarbiMTi6ljx>INObCAxFi) z%R01jwKUIsEcO*gA5KJ#Y97{_3I_v!C(u+z-d0NEevXt=;BbuXg#y){|A&Lm1^RWa zq#?#K^YGoj6Serq7_1!UgYW4EIa6X0_+Q8SQ|{Br2#Bj;6ey?q4{6h(cw35g@WGda zRkXF8dK&4AlJ$wxy?}aP^673INs@p}+)QKuteXjupxSy3zYQ#S_AWUi@jBigm;eUf z+f2%XaD$1A07nN;g-gi)3Sb5N{0$Ku1*!o%pkTxY1Wbdu4_Km7-K21T1(*()0+z3D zEm5d%QdAWg1D_5iBg})&RWJ)Yu7;!kgk{n1cuvoj|Y~ie1RcSzarD%-p5F_ z^lAXxAWPN<2%7hSb*3Kg%u~7-$?5@ZGWFp1Z=~wVbrW&yA(G|_K;Ks{OCyiiFA&Eb zF47Pzo&i;g<+^?+Z>5e@nkZFH0wnNcK%-1O_~n5zoq22`ZoLw&0F6K2PNXuiI|AZA zY6{}DtJENhHqMh7zU#_mz6WVbi82G&Ir+rG9dBgHsRptGXt~_p(`vg!kR=vabuuHo zRLYz^GR7AjomVT<+RK8_`b$@AbY%~ySb~N!wdyC8QY7tv4E!!?*Q7}XDoAoRD*w$q z{JO8GR(+0aQgLV|&Qbh!(N7OK`NJlN3oF1LFJ|;DS{Ifikej0l7L94EXQe*5PBX z<!;^{4zL0qAPB2CERt}^Xn#DcV(I=3bMxum%J$O%8K9b50&IW-aJG!qdP~d6 zIVC{fynMR1^1Yv4X#|wbfV!bo>kSRf+Z!v>y|+qwziQQV1Fl--y;ay7(Dtbm0M3{` AnE(I) literal 0 HcmV?d00001 diff --git a/composeApp/src/androidMain/res/mipmap-xxhdpi/ic_launcher_round.webp b/composeApp/src/androidMain/res/mipmap-xxhdpi/ic_launcher_round.webp new file mode 100644 index 0000000000000000000000000000000000000000..465a18ee2526a8a9a28e8d5a6a28743f51cd806b GIT binary patch literal 5520 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 SN?NMBoktP3k{)HVR_8k5w8dBe literal 0 HcmV?d00001 diff --git a/composeApp/src/androidMain/res/mipmap-xxxhdpi/ic_launcher.webp b/composeApp/src/androidMain/res/mipmap-xxxhdpi/ic_launcher.webp new file mode 100644 index 0000000000000000000000000000000000000000..884efbe8c35d3ce8ed2b4474600572655040452c GIT binary patch literal 2912 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 KhCsb#7Xbh}TFAcu literal 0 HcmV?d00001 diff --git a/composeApp/src/androidMain/res/mipmap-xxxhdpi/ic_launcher_round.webp b/composeApp/src/androidMain/res/mipmap-xxxhdpi/ic_launcher_round.webp new file mode 100644 index 0000000000000000000000000000000000000000..0e7063365aebf396073d8ec4a48a37e535f23133 GIT binary patch literal 7806 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 + + Contacts Assessment + diff --git a/composeApp/src/commonMain/composeResources/values/strings.xml b/composeApp/src/commonMain/composeResources/values/strings.xml new file mode 100644 index 0000000..9f64935 --- /dev/null +++ b/composeApp/src/commonMain/composeResources/values/strings.xml @@ -0,0 +1,21 @@ + + + Contacts Assessment + + + Groups + Contacts + + + Contacts + Avatar of %1$s + + + Groups + %1$d members + + + Active + Pending + Inactive + diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/App.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/App.kt new file mode 100644 index 0000000..f6f027d --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/App.kt @@ -0,0 +1,12 @@ +package io.assessment.contacts + +import androidx.compose.runtime.Composable +import io.assessment.contacts.designsystem.ContactsTheme +import io.assessment.contacts.navigation.NavigationRoot + +@Composable +fun App() { + ContactsTheme { + NavigationRoot() + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/MainViewModel.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/MainViewModel.kt new file mode 100644 index 0000000..ddee944 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/MainViewModel.kt @@ -0,0 +1,5 @@ +package io.assessment.contacts + +import androidx.lifecycle.ViewModel + +class MainViewModel : ViewModel() diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/data/ContactMapper.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/data/ContactMapper.kt new file mode 100644 index 0000000..d6fe2d5 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/data/ContactMapper.kt @@ -0,0 +1,27 @@ +package io.assessment.contacts.contacts.data + +import io.assessment.contacts.core.data.database.ContactEntity +import io.assessment.contacts.core.domain.contact.Contact +import io.assessment.contacts.core.domain.contact.ContactStatus + +fun ContactEntity.toDomain(): Contact { + return Contact( + id = id, + name = name, + npub = npub, + avatarUrl = avatarUrl, + status = ContactStatus.INACTIVE, + groupId = groupId, + ) +} + +fun Contact.toEntity(): ContactEntity { + return ContactEntity( + id = id, + name = name, + npub = npub, + avatarUrl = avatarUrl, + status = status, + groupId = groupId, + ) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/data/ObserveContactsUseCaseImpl.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/data/ObserveContactsUseCaseImpl.kt new file mode 100644 index 0000000..672de6b --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/data/ObserveContactsUseCaseImpl.kt @@ -0,0 +1,17 @@ +package io.assessment.contacts.contacts.data + +import io.assessment.contacts.contacts.domain.ObserveContactsUseCase +import io.assessment.contacts.core.domain.contact.Contact +import io.assessment.contacts.core.domain.contact.ContactError +import io.assessment.contacts.core.domain.contact.ContactRepository +import io.assessment.contacts.core.domain.result.Result +import kotlinx.coroutines.flow.Flow + +class ObserveContactsUseCaseImpl( + private val contactRepository: ContactRepository, +) : ObserveContactsUseCase { + + override fun invoke(): Flow, ContactError>> { + return contactRepository.observeContacts() + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/di/ContactsModule.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/di/ContactsModule.kt new file mode 100644 index 0000000..c0b3514 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/di/ContactsModule.kt @@ -0,0 +1,14 @@ +package io.assessment.contacts.contacts.di + +import io.assessment.contacts.contacts.domain.ObserveContactsUseCase +import io.assessment.contacts.contacts.data.ObserveContactsUseCaseImpl +import io.assessment.contacts.contacts.presentation.ContactsViewModel +import org.koin.core.module.dsl.factoryOf +import org.koin.core.module.dsl.viewModelOf +import org.koin.dsl.bind +import org.koin.dsl.module + +val contactsModule = module { + factoryOf(::ObserveContactsUseCaseImpl) bind ObserveContactsUseCase::class + viewModelOf(::ContactsViewModel) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/domain/ObserveContactsUseCase.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/domain/ObserveContactsUseCase.kt new file mode 100644 index 0000000..59b7b40 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/domain/ObserveContactsUseCase.kt @@ -0,0 +1,7 @@ +package io.assessment.contacts.contacts.domain + +import io.assessment.contacts.core.domain.contact.Contact +import io.assessment.contacts.core.domain.contact.ContactError +import io.assessment.contacts.core.domain.usecase.NoParamsFlowUseCase + +interface ObserveContactsUseCase : NoParamsFlowUseCase, ContactError> diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactVO.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactVO.kt new file mode 100644 index 0000000..3b536ef --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactVO.kt @@ -0,0 +1,22 @@ +package io.assessment.contacts.contacts.presentation + +import io.assessment.contacts.core.domain.contact.Contact +import io.assessment.contacts.core.domain.contact.ContactStatus + +data class ContactVO( + val id: String, + val name: String, + val npub: String, + val avatarUrl: String?, + val status: ContactStatus, +) + +fun Contact.toVO(): ContactVO { + return ContactVO( + id = id, + name = name, + npub = npub, + avatarUrl = avatarUrl, + status = status, + ) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsScreen.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsScreen.kt new file mode 100644 index 0000000..dac7a83 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsScreen.kt @@ -0,0 +1,200 @@ +package io.assessment.contacts.contacts.presentation + +import androidx.compose.foundation.background +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.Spacer +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.foundation.layout.width +import androidx.compose.foundation.lazy.LazyColumn +import androidx.compose.foundation.lazy.items +import androidx.compose.foundation.shape.CircleShape +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.material.icons.Icons +import androidx.compose.material.icons.filled.Person +import androidx.compose.material3.Card +import androidx.compose.material3.CardDefaults +import androidx.compose.material3.CircularProgressIndicator +import androidx.compose.material3.ExperimentalMaterial3Api +import androidx.compose.material3.Icon +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Scaffold +import androidx.compose.material3.Text +import androidx.compose.material3.TopAppBar +import androidx.compose.runtime.Composable +import androidx.compose.runtime.getValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.clip +import androidx.compose.ui.layout.ContentScale +import androidx.compose.ui.text.style.TextOverflow +import androidx.compose.ui.unit.dp +import androidx.lifecycle.compose.collectAsStateWithLifecycle +import coil3.compose.AsyncImage +import contactsassessment.composeapp.generated.resources.Res +import contactsassessment.composeapp.generated.resources.contacts_avatar_description +import contactsassessment.composeapp.generated.resources.contacts_title +import contactsassessment.composeapp.generated.resources.status_active +import contactsassessment.composeapp.generated.resources.status_inactive +import contactsassessment.composeapp.generated.resources.status_pending +import io.assessment.contacts.core.domain.contact.ContactStatus +import io.assessment.contacts.designsystem.extendedColors +import org.jetbrains.compose.resources.stringResource +import org.koin.compose.viewmodel.koinViewModel + +@OptIn(ExperimentalMaterial3Api::class) +@Composable +fun ContactsScreen( + viewModel: ContactsViewModel = koinViewModel(), +) { + val state by viewModel.state.collectAsStateWithLifecycle() + + ContactsScreenContent(state = state) +} + +@OptIn(ExperimentalMaterial3Api::class) +@Composable +internal fun ContactsScreenContent( + state: ContactsState, +) { + Scaffold( + topBar = { + TopAppBar( + title = { Text(stringResource(Res.string.contacts_title)) }, + ) + }, + ) { paddingValues -> + if (state.isLoading) { + Box( + modifier = Modifier + .fillMaxSize() + .padding(paddingValues), + contentAlignment = Alignment.Center, + ) { + CircularProgressIndicator() + } + } else { + LazyColumn( + modifier = Modifier + .fillMaxSize() + .padding(paddingValues), + verticalArrangement = Arrangement.spacedBy(8.dp), + ) { + items( + items = state.contacts, + key = { it.id }, + ) { contact -> + ContactListItem(contact = contact) + } + } + } + } +} + +@Composable +internal fun ContactListItem( + contact: ContactVO, + modifier: Modifier = Modifier, +) { + Card( + modifier = modifier + .fillMaxWidth() + .padding(horizontal = 16.dp), + shape = RoundedCornerShape(12.dp), + elevation = CardDefaults.cardElevation(defaultElevation = 2.dp), + ) { + Row( + modifier = Modifier + .fillMaxWidth() + .padding(16.dp), + verticalAlignment = Alignment.CenterVertically, + ) { + ContactAvatar( + avatarUrl = contact.avatarUrl, + name = contact.name, + ) + + Spacer(modifier = Modifier.width(16.dp)) + + Column( + modifier = Modifier.weight(1f), + ) { + Text( + text = contact.name, + style = MaterialTheme.typography.titleMedium, + ) + Text( + text = contact.npub, + style = MaterialTheme.typography.bodySmall, + color = MaterialTheme.colorScheme.onSurfaceVariant, + maxLines = 1, + overflow = TextOverflow.Ellipsis, + ) + } + + Spacer(modifier = Modifier.width(8.dp)) + + StatusBadge(status = contact.status) + } + } +} + +@Composable +internal fun ContactAvatar( + avatarUrl: String?, + name: String, + modifier: Modifier = Modifier, +) { + Box( + modifier = modifier + .size(48.dp) + .clip(CircleShape) + .background(MaterialTheme.colorScheme.primaryContainer), + contentAlignment = Alignment.Center, + ) { + if (avatarUrl != null) { + AsyncImage( + model = avatarUrl, + contentDescription = stringResource(Res.string.contacts_avatar_description, name), + modifier = Modifier.fillMaxSize(), + contentScale = ContentScale.Crop, + ) + } else { + Icon( + imageVector = Icons.Default.Person, + contentDescription = null, + tint = MaterialTheme.colorScheme.onPrimaryContainer, + ) + } + } +} + +@Composable +internal fun StatusBadge( + status: ContactStatus, + modifier: Modifier = Modifier, +) { + val (backgroundColor, text) = when (status) { + ContactStatus.ACTIVE -> extendedColors.statusActive to stringResource(Res.string.status_active) + ContactStatus.PENDING -> extendedColors.statusPending to stringResource(Res.string.status_pending) + ContactStatus.INACTIVE -> extendedColors.statusInactive to stringResource(Res.string.status_inactive) + } + + Box( + modifier = modifier + .clip(RoundedCornerShape(16.dp)) + .background(backgroundColor.copy(alpha = 0.2f)) + .padding(horizontal = 12.dp, vertical = 4.dp), + ) { + Text( + text = text, + style = MaterialTheme.typography.labelSmall, + color = backgroundColor, + ) + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsState.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsState.kt new file mode 100644 index 0000000..483044d --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsState.kt @@ -0,0 +1,6 @@ +package io.assessment.contacts.contacts.presentation + +data class ContactsState( + val contacts: List = emptyList(), + val isLoading: Boolean = true, +) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsViewModel.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsViewModel.kt new file mode 100644 index 0000000..1873728 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/ContactsViewModel.kt @@ -0,0 +1,26 @@ +package io.assessment.contacts.contacts.presentation + +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import io.assessment.contacts.contacts.domain.ObserveContactsUseCase +import io.assessment.contacts.core.domain.result.filterSuccess +import io.assessment.contacts.core.presentation.stateInWhileSubscribed +import kotlinx.coroutines.flow.map + +class ContactsViewModel( + observeContactsUseCase: ObserveContactsUseCase, +) : ViewModel() { + + val state = observeContactsUseCase() + .filterSuccess() + .map { contacts -> + ContactsState( + contacts = contacts.take(1).map { it.toVO() }, + isLoading = false, + ) + } + .stateInWhileSubscribed( + scope = viewModelScope, + initialValue = ContactsState(), + ) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/navigation/ContactsNavDestinations.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/navigation/ContactsNavDestinations.kt new file mode 100644 index 0000000..3fe8fdb --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/contacts/presentation/navigation/ContactsNavDestinations.kt @@ -0,0 +1,9 @@ +package io.assessment.contacts.contacts.presentation.navigation + +import kotlinx.serialization.Serializable + +@Serializable +object ContactsNavDestinations { + @Serializable + object ContactsList +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactDao.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactDao.kt new file mode 100644 index 0000000..b80c6e0 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactDao.kt @@ -0,0 +1,39 @@ +package io.assessment.contacts.core.data.database + +import androidx.room.Dao +import androidx.room.Insert +import androidx.room.OnConflictStrategy +import androidx.room.Query +import io.assessment.contacts.core.domain.contact.ContactStatus +import kotlinx.coroutines.flow.Flow + +@Dao +interface ContactDao { + + @Query("SELECT * FROM contacts ORDER BY name ASC") + fun observeAll(): Flow> + + @Query("SELECT * FROM contacts WHERE name LIKE '%' || :query || '%' ORDER BY name ASC") + fun observeByNameQuery(query: String): Flow> + + @Query("SELECT * FROM contacts WHERE status = :status ORDER BY name ASC") + fun observeByStatus(status: ContactStatus): Flow> + + @Query("SELECT * FROM contacts WHERE groupId = :groupId ORDER BY name ASC") + fun observeByGroupId(groupId: String): Flow> + + @Query("SELECT * FROM contacts WHERE id = :id") + suspend fun getById(id: String): ContactEntity? + + @Insert(onConflict = OnConflictStrategy.REPLACE) + suspend fun insertAll(contacts: List) + + @Insert(onConflict = OnConflictStrategy.REPLACE) + suspend fun insert(contact: ContactEntity) + + @Query("DELETE FROM contacts") + suspend fun deleteAll() + + @Query("DELETE FROM contacts WHERE id = :id") + suspend fun deleteById(id: String) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactEntity.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactEntity.kt new file mode 100644 index 0000000..03de494 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactEntity.kt @@ -0,0 +1,16 @@ +package io.assessment.contacts.core.data.database + +import androidx.room.Entity +import androidx.room.PrimaryKey +import io.assessment.contacts.core.domain.contact.ContactStatus + +@Entity(tableName = "contacts") +data class ContactEntity( + @PrimaryKey + val id: String, + val name: String, + val npub: String, + val avatarUrl: String?, + val status: ContactStatus, + val groupId: String?, +) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactsDatabase.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactsDatabase.kt new file mode 100644 index 0000000..5ffdba5 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/ContactsDatabase.kt @@ -0,0 +1,17 @@ +package io.assessment.contacts.core.data.database + +import androidx.room.Database +import androidx.room.RoomDatabase + +@Database( + entities = [ + ContactEntity::class, + GroupEntity::class, + ], + version = 1, + exportSchema = true, +) +abstract class ContactsDatabase : RoomDatabase() { + abstract val contactDao: ContactDao + abstract val groupDao: GroupDao +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/DatabaseSeeder.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/DatabaseSeeder.kt new file mode 100644 index 0000000..000c669 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/DatabaseSeeder.kt @@ -0,0 +1,108 @@ +package io.assessment.contacts.core.data.database + +import io.assessment.contacts.core.domain.contact.ContactStatus + +class DatabaseSeeder( + private val contactDao: ContactDao, + private val groupDao: GroupDao, +) { + suspend fun seedIfEmpty() { + if (contactDao.getById("c1") == null) { + seedContacts() + } + if (groupDao.getById("g1") == null) { + seedGroups() + } + } + + private suspend fun seedContacts() { + val contacts = listOf( + ContactEntity( + id = "c1", + name = "Alice Johnson", + npub = "npub1alice...7x8q", + avatarUrl = "https://i.pravatar.cc/150?u=alice", + status = ContactStatus.ACTIVE, + groupId = "g1", + ), + ContactEntity( + id = "c2", + name = "Bob Smith", + npub = "npub1bob...3k9m", + avatarUrl = "https://i.pravatar.cc/150?u=bob", + status = ContactStatus.PENDING, + groupId = "g1", + ), + ContactEntity( + id = "c3", + name = "Charlie Brown", + npub = "npub1charlie...5n2p", + avatarUrl = "https://i.pravatar.cc/150?u=charlie", + status = ContactStatus.ACTIVE, + groupId = "g1", + ), + ContactEntity( + id = "c4", + name = "Diana Ross", + npub = "npub1diana...8w4r", + avatarUrl = "https://i.pravatar.cc/150?u=diana", + status = ContactStatus.INACTIVE, + groupId = "g2", + ), + ContactEntity( + id = "c5", + name = "Edward Norton", + npub = "npub1edward...2j6t", + avatarUrl = "https://i.pravatar.cc/150?u=edward", + status = ContactStatus.PENDING, + groupId = "g2", + ), + ContactEntity( + id = "c6", + name = "Fiona Apple", + npub = "npub1fiona...9v3s", + avatarUrl = "https://i.pravatar.cc/150?u=fiona", + status = ContactStatus.ACTIVE, + groupId = "g3", + ), + ContactEntity( + id = "c7", + name = "George Clooney", + npub = "npub1george...4h8y", + avatarUrl = "https://i.pravatar.cc/150?u=george", + status = ContactStatus.ACTIVE, + groupId = null, + ), + ContactEntity( + id = "c8", + name = "Helen Mirren", + npub = "npub1helen...6m1z", + avatarUrl = "https://i.pravatar.cc/150?u=helen", + status = ContactStatus.PENDING, + groupId = null, + ), + ) + contactDao.insertAll(contacts) + } + + private suspend fun seedGroups() { + val groups = listOf( + GroupEntity( + id = "g1", + name = "Alice's group", + memberCount = 3, + ), + GroupEntity( + id = "g2", + name = "Close friends", + memberCount = 2, + ), + GroupEntity( + id = "g3", + name = "Colleagues", + memberCount = 1, + ), + ) + groupDao.insertAll(groups) + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/GroupDao.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/GroupDao.kt new file mode 100644 index 0000000..85f3817 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/GroupDao.kt @@ -0,0 +1,26 @@ +package io.assessment.contacts.core.data.database + +import androidx.room.Dao +import androidx.room.Insert +import androidx.room.OnConflictStrategy +import androidx.room.Query +import kotlinx.coroutines.flow.Flow + +@Dao +interface GroupDao { + + @Query("SELECT * FROM groups ORDER BY name ASC") + fun observeAll(): Flow> + + @Query("SELECT * FROM groups WHERE id = :id") + suspend fun getById(id: String): GroupEntity? + + @Insert(onConflict = OnConflictStrategy.REPLACE) + suspend fun insertAll(groups: List) + + @Insert(onConflict = OnConflictStrategy.REPLACE) + suspend fun insert(group: GroupEntity) + + @Query("DELETE FROM groups") + suspend fun deleteAll() +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/GroupEntity.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/GroupEntity.kt new file mode 100644 index 0000000..fceaf42 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/database/GroupEntity.kt @@ -0,0 +1,12 @@ +package io.assessment.contacts.core.data.database + +import androidx.room.Entity +import androidx.room.PrimaryKey + +@Entity(tableName = "groups") +data class GroupEntity( + @PrimaryKey + val id: String, + val name: String, + val memberCount: Int, +) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/datasource/ContactDataSource.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/datasource/ContactDataSource.kt new file mode 100644 index 0000000..bc53809 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/datasource/ContactDataSource.kt @@ -0,0 +1,23 @@ +package io.assessment.contacts.core.data.datasource + +import io.assessment.contacts.core.data.database.ContactDao +import io.assessment.contacts.core.data.database.ContactEntity +import kotlinx.coroutines.flow.Flow + +interface ContactDataSource { + fun observeContacts(): Flow> + fun observeContactsByName(query: String): Flow> +} + +class RoomContactDataSource( + private val contactDao: ContactDao, +) : ContactDataSource { + + override fun observeContacts(): Flow> { + return contactDao.observeAll() + } + + override fun observeContactsByName(query: String): Flow> { + return contactDao.observeByNameQuery(query) + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/datasource/GroupDataSource.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/datasource/GroupDataSource.kt new file mode 100644 index 0000000..f788800 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/datasource/GroupDataSource.kt @@ -0,0 +1,18 @@ +package io.assessment.contacts.core.data.datasource + +import io.assessment.contacts.core.data.database.GroupDao +import io.assessment.contacts.core.data.database.GroupEntity +import kotlinx.coroutines.flow.Flow + +interface GroupDataSource { + fun observeGroups(): Flow> +} + +class RoomGroupDataSource( + private val groupDao: GroupDao, +) : GroupDataSource { + + override fun observeGroups(): Flow> { + return groupDao.observeAll() + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/repository/ContactRepositoryImpl.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/repository/ContactRepositoryImpl.kt new file mode 100644 index 0000000..5acb35e --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/repository/ContactRepositoryImpl.kt @@ -0,0 +1,29 @@ +package io.assessment.contacts.core.data.repository + +import io.assessment.contacts.contacts.data.toDomain +import io.assessment.contacts.core.data.datasource.ContactDataSource +import io.assessment.contacts.core.domain.contact.Contact +import io.assessment.contacts.core.domain.contact.ContactError +import io.assessment.contacts.core.domain.contact.ContactRepository +import io.assessment.contacts.core.domain.result.Result +import kotlinx.coroutines.flow.Flow +import kotlinx.coroutines.flow.map + +class ContactRepositoryImpl( + private val contactDataSource: ContactDataSource, +) : ContactRepository { + + override fun observeContacts(): Flow, ContactError>> { + return contactDataSource.observeContacts() + .map { entities -> + Result.Success(entities.map { it.toDomain() }) + } + } + + override fun observeContactsByName(query: String): Flow, ContactError>> { + return contactDataSource.observeContactsByName(query) + .map { entities -> + Result.Success(entities.map { it.toDomain() }) + } + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/repository/GroupRepositoryImpl.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/repository/GroupRepositoryImpl.kt new file mode 100644 index 0000000..2dd2bde --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/data/repository/GroupRepositoryImpl.kt @@ -0,0 +1,22 @@ +package io.assessment.contacts.core.data.repository + +import io.assessment.contacts.core.data.datasource.GroupDataSource +import io.assessment.contacts.core.domain.group.Group +import io.assessment.contacts.core.domain.group.GroupError +import io.assessment.contacts.core.domain.group.GroupRepository +import io.assessment.contacts.core.domain.result.Result +import io.assessment.contacts.groups.data.toDomain +import kotlinx.coroutines.flow.Flow +import kotlinx.coroutines.flow.map + +class GroupRepositoryImpl( + private val groupDataSource: GroupDataSource, +) : GroupRepository { + + override fun observeGroups(): Flow, GroupError>> { + return groupDataSource.observeGroups() + .map { entities -> + Result.Success(entities.map { it.toDomain() }) + } + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/Contact.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/Contact.kt new file mode 100644 index 0000000..175da2c --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/Contact.kt @@ -0,0 +1,10 @@ +package io.assessment.contacts.core.domain.contact + +data class Contact( + val id: String, + val name: String, + val npub: String, + val avatarUrl: String?, + val status: ContactStatus, + val groupId: String?, +) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactError.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactError.kt new file mode 100644 index 0000000..9ddbe8d --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactError.kt @@ -0,0 +1,9 @@ +package io.assessment.contacts.core.domain.contact + +import io.assessment.contacts.core.domain.result.Error + +sealed interface ContactError : Error { + data object NotFound : ContactError + data object NetworkError : ContactError + data class Unknown(val message: String) : ContactError +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactRepository.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactRepository.kt new file mode 100644 index 0000000..069e6d3 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactRepository.kt @@ -0,0 +1,9 @@ +package io.assessment.contacts.core.domain.contact + +import io.assessment.contacts.core.domain.result.Result +import kotlinx.coroutines.flow.Flow + +interface ContactRepository { + fun observeContacts(): Flow, ContactError>> + fun observeContactsByName(query: String): Flow, ContactError>> +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactStatus.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactStatus.kt new file mode 100644 index 0000000..6adba68 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/contact/ContactStatus.kt @@ -0,0 +1,7 @@ +package io.assessment.contacts.core.domain.contact + +enum class ContactStatus { + ACTIVE, + PENDING, + INACTIVE, +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/Group.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/Group.kt new file mode 100644 index 0000000..84965d8 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/Group.kt @@ -0,0 +1,7 @@ +package io.assessment.contacts.core.domain.group + +data class Group( + val id: String, + val name: String, + val memberCount: Int, +) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/GroupError.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/GroupError.kt new file mode 100644 index 0000000..9ad4fb5 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/GroupError.kt @@ -0,0 +1,9 @@ +package io.assessment.contacts.core.domain.group + +import io.assessment.contacts.core.domain.result.Error + +sealed interface GroupError : Error { + data object NotFound : GroupError + data object NetworkError : GroupError + data class Unknown(val message: String) : GroupError +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/GroupRepository.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/GroupRepository.kt new file mode 100644 index 0000000..fec81e3 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/group/GroupRepository.kt @@ -0,0 +1,8 @@ +package io.assessment.contacts.core.domain.group + +import io.assessment.contacts.core.domain.result.Result +import kotlinx.coroutines.flow.Flow + +interface GroupRepository { + fun observeGroups(): Flow, GroupError>> +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/Error.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/Error.kt new file mode 100644 index 0000000..9e1f3f2 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/Error.kt @@ -0,0 +1,3 @@ +package io.assessment.contacts.core.domain.result + +interface Error diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/FlowResultExtensions.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/FlowResultExtensions.kt new file mode 100644 index 0000000..0e6a16a --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/FlowResultExtensions.kt @@ -0,0 +1,60 @@ +package io.assessment.contacts.core.domain.result + +import kotlinx.coroutines.flow.Flow +import kotlinx.coroutines.flow.map +import kotlinx.coroutines.flow.transform + +fun Flow>.mapSuccess(transform: suspend (T) -> R): Flow> = + transform { result -> + when (result) { + is Result.Success -> emit(Result.Success(transform(result.data))) + is Result.Error -> emit(Result.Error(result.error)) + } + } + +fun Flow>.mapError(transform: (E1) -> E2): Flow> = + map { result -> result.mapError(transform) } + +fun Flow>.flatMapSuccess(transform: suspend (T) -> Result): Flow> = + transform { result -> + when (result) { + is Result.Success -> emit(transform(result.data)) + is Result.Error -> emit(Result.Error(result.error)) + } + } + +fun Flow>.filterSuccess(): Flow = + transform { result -> + if (result is Result.Success) { + emit(result.data) + } + } + +fun Flow>.filterError(): Flow = + transform { result -> + if (result is Result.Error) { + emit(result.error) + } + } + +fun Flow>.biMap( + onSuccess: suspend (T) -> Result, + onError: (E1) -> E2, +): Flow> = + transform { result -> + when (result) { + is Result.Success -> emit(onSuccess(result.data)) + is Result.Error -> emit(Result.Error(onError(result.error))) + } + } + +fun Flow>.transformResult( + onSuccess: suspend (T) -> R, + onError: (E1) -> E2, +): Flow> = + transform { result -> + when (result) { + is Result.Success -> emit(Result.Success(onSuccess(result.data))) + is Result.Error -> emit(Result.Error(onError(result.error))) + } + } diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/Result.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/Result.kt new file mode 100644 index 0000000..80aee09 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/result/Result.kt @@ -0,0 +1,77 @@ +package io.assessment.contacts.core.domain.result + +import kotlinx.coroutines.flow.Flow +import kotlinx.coroutines.flow.flowOf + +sealed interface Result { + data class Success( + val data: D, + ) : Result + + data class Error( + val error: E, + ) : Result +} + +inline fun Result.map(map: (T) -> R): Result = + when (this) { + is Result.Error -> Result.Error(error) + is Result.Success -> Result.Success(map(data)) + } + +fun Result.asEmptyDataResult(): EmptyResult = map { } + +inline fun Result.onSuccess(action: (T) -> Unit): Result = + when (this) { + is Result.Error -> this + is Result.Success -> { + action(data) + this + } + } + +inline fun Result.onError(action: (E) -> Unit): Result = + when (this) { + is Result.Error -> { + action(error) + this + } + is Result.Success -> this + } + +fun Result.successValue(): T? = + when (this) { + is Result.Success -> data + is Result.Error -> null + } + +inline fun Result.mapSuccess(transform: (T) -> R): Result = + when (this) { + is Result.Success -> Result.Success(transform(data)) + is Result.Error -> Result.Error(error) + } + +inline fun Result.mapError(transform: (E1) -> E2): Result = + when (this) { + is Result.Success -> Result.Success(data) + is Result.Error -> Result.Error(transform(error)) + } + +inline fun Result.flatMapSuccess(transform: (T) -> Result): Result = + when (this) { + is Result.Success -> transform(data) + is Result.Error -> Result.Error(error) + } + +inline fun Result.fold( + onSuccess: (T) -> R, + onError: (E) -> R, +): R = + when (this) { + is Result.Success -> onSuccess(data) + is Result.Error -> onError(error) + } + +typealias EmptyResult = Result + +fun Result.asFlow(): Flow> = flowOf(this) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/usecase/UseCase.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/usecase/UseCase.kt new file mode 100644 index 0000000..7f1e665 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/domain/usecase/UseCase.kt @@ -0,0 +1,24 @@ +package io.assessment.contacts.core.domain.usecase + +import io.assessment.contacts.core.domain.result.Error +import io.assessment.contacts.core.domain.result.Result +import kotlinx.coroutines.flow.Flow + +interface UseCase { + suspend operator fun invoke(params: P): Result +} + +interface NoParamsUseCase { + suspend operator fun invoke(): Result +} + +interface FlowUseCase { + operator fun invoke(params: P): Flow> +} + +interface NoParamsFlowUseCase { + operator fun invoke(): Flow> +} + +typealias ActionUseCase = UseCase +typealias NoParamsActionUseCase = NoParamsUseCase diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/presentation/ObserveAsEvents.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/presentation/ObserveAsEvents.kt new file mode 100644 index 0000000..9ad950f --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/presentation/ObserveAsEvents.kt @@ -0,0 +1,27 @@ +package io.assessment.contacts.core.presentation + +import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect +import androidx.lifecycle.Lifecycle +import androidx.lifecycle.compose.LocalLifecycleOwner +import androidx.lifecycle.repeatOnLifecycle +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.flow.Flow +import kotlinx.coroutines.withContext + +@Composable +fun ObserveAsEvents( + flow: Flow, + key1: Any? = null, + key2: Any? = null, + onEvent: (T) -> Unit, +) { + val lifecycleOwner = LocalLifecycleOwner.current + LaunchedEffect(flow, lifecycleOwner.lifecycle, key1, key2) { + lifecycleOwner.repeatOnLifecycle(Lifecycle.State.STARTED) { + withContext(Dispatchers.Main.immediate) { + flow.collect(onEvent) + } + } + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/presentation/StateFlowExt.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/presentation/StateFlowExt.kt new file mode 100644 index 0000000..8f4b9d9 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/core/presentation/StateFlowExt.kt @@ -0,0 +1,12 @@ +package io.assessment.contacts.core.presentation + +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.flow.Flow +import kotlinx.coroutines.flow.SharingStarted +import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.flow.stateIn + +fun Flow.stateInWhileSubscribed( + scope: CoroutineScope, + initialValue: T, +): StateFlow = this.stateIn(scope, SharingStarted.WhileSubscribed(5000), initialValue) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Color.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Color.kt new file mode 100644 index 0000000..25f64e0 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Color.kt @@ -0,0 +1,63 @@ +package io.assessment.contacts.designsystem + +import androidx.compose.ui.graphics.Color + +// Common Red Palette Colors +val RedDarkest = Color(0xFF410001) +val RedSecondaryDarkest = Color(0xFF410004) +val RedVeryDark = Color(0xFF690003) +val RedDark = Color(0xFF8B0006) +val RedMediumDark = Color(0xFF930007) +val RedMedium = Color(0xFFB50E11) +val RedBright = Color(0xFFBA1A1A) +val RedSecondaryDark = Color(0xFF68000C) +val RedSecondaryMedium = Color(0xFFB51824) +val RedSecondaryBright = Color(0xFFD93539) +val RedTertiaryDark = Color(0xFF5A0009) +val RedTertiaryMedium = Color(0xFF930015) +val RedAccent = Color(0xFFBC1515) +val RedLight = Color(0xFFFF5353) +val RedVeryLight = Color(0xFFFFB3AE) +val RedPastel = Color(0xFFFFB4AA) +val RedSoft = Color(0xFFFFB4AB) +val RedPale = Color(0xFFFFC3BC) +val RedMuted = Color(0xFFE5BDB8) +val RedVeryPale = Color(0xFFFFDAD5) +val RedSecondaryPale = Color(0xFFFFDAD7) +val RedAlmostWhite = Color(0xFFFFDAD6) +val RedUltraPale = Color(0xFFFFEDEB) + +// Surface Colors +val SurfaceDarkest = Color(0xFF0F0E0E) +val SurfaceVeryDark = Color(0xFF141313) +val SurfaceDark = Color(0xFF1D1B1B) +val SurfaceMediumDark = Color(0xFF211F1F) +val SurfaceMedium = Color(0xFF2B2A29) +val SurfaceGray = Color(0xFF323030) +val SurfaceLightGray = Color(0xFF363434) +val SurfaceBright = Color(0xFF3B3938) +val SurfaceOutline = Color(0xFF444748) +val SurfaceVariant = Color(0xFF747878) +val SurfaceOutlineLight = Color(0xFF8E9192) +val SurfaceLight = Color(0xFFC4C7C7) +val SurfacePale = Color(0xFFDED9D8) +val SurfaceVeryPale = Color(0xFFE6E1E1) +val SurfaceLightest = Color(0xFFECE7E6) +val SurfaceNearWhite = Color(0xFFF2EDEC) +val SurfaceAlmostWhite = Color(0xFFF5F0EF) +val SurfaceOffWhite = Color(0xFFF8F2F2) +val SurfaceWhiteTinted = Color(0xFFFDF8F7) +val SurfaceAlmostPure = Color(0xFFFFFBFF) + +// Pure Colors +val PureBlack = Color(0xFF000000) +val PureWhite = Color(0xFFFFFFFF) + +// Accent Colors +val AccentGreen = Color(0xFF34C759) +val AccentOrange = Color(0xFFFF9500) + +// Status Colors +val StatusGreen = Color(0xFF4CAF50) +val StatusOrange = Color(0xFFFFA726) +val StatusGray = Color(0xFF9E9E9E) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/ExtendedColors.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/ExtendedColors.kt new file mode 100644 index 0000000..9ad6711 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/ExtendedColors.kt @@ -0,0 +1,137 @@ +package io.assessment.contacts.designsystem + +import androidx.compose.material3.ColorScheme +import androidx.compose.material3.darkColorScheme +import androidx.compose.material3.lightColorScheme +import androidx.compose.runtime.Composable +import androidx.compose.runtime.Immutable +import androidx.compose.runtime.ReadOnlyComposable +import androidx.compose.runtime.staticCompositionLocalOf +import androidx.compose.ui.graphics.Color + +val LocalExtendedColors = staticCompositionLocalOf { extendedColors } + +val ColorScheme.extended: ExtendedColors + @ReadOnlyComposable + @Composable + get() = LocalExtendedColors.current + +@Immutable +data class ExtendedColors( + val primaryFixed: Color, + val primaryFixedDim: Color, + val onPrimaryFixed: Color, + val onPrimaryFixedVariant: Color, + val secondaryFixed: Color, + val secondaryFixedDim: Color, + val onSecondaryFixed: Color, + val onSecondaryFixedVariant: Color, + val tertiaryFixed: Color, + val tertiaryFixedDim: Color, + val onTertiaryFixed: Color, + val onTertiaryFixedVariant: Color, + val shadow: Color, + val accentGreen: Color, + val accentOrange: Color, + val disabledContainer: Color, + val statusActive: Color, + val statusPending: Color, + val statusInactive: Color, +) + +val LightColorScheme = + lightColorScheme( + primary = RedDark, + onPrimary = PureWhite, + primaryContainer = RedMedium, + onPrimaryContainer = RedPale, + secondary = RedSecondaryMedium, + onSecondary = PureWhite, + secondaryContainer = RedSecondaryBright, + onSecondaryContainer = SurfaceAlmostPure, + tertiary = RedDark, + onTertiary = PureWhite, + tertiaryContainer = RedMedium, + onTertiaryContainer = RedPale, + error = RedBright, + onError = PureWhite, + errorContainer = RedAlmostWhite, + onErrorContainer = RedMediumDark, + surfaceDim = SurfacePale, + surface = SurfaceWhiteTinted, + surfaceBright = SurfaceDark, + surfaceTint = RedUltraPale, + surfaceContainerLowest = PureWhite, + surfaceContainerLow = SurfaceOffWhite, + surfaceContainer = SurfaceNearWhite, + surfaceContainerHigh = SurfaceLightest, + surfaceContainerHighest = SurfaceVeryPale, + onSurface = SurfaceDark, + onSurfaceVariant = SurfaceOutline, + outline = SurfaceVariant, + outlineVariant = RedMuted, + inverseSurface = SurfaceGray, + inverseOnSurface = SurfaceAlmostWhite, + inversePrimary = RedPastel, + scrim = PureBlack, + ) + +val DarkColorScheme = + darkColorScheme( + primary = RedPastel, + onPrimary = RedVeryDark, + primaryContainer = RedMedium, + onPrimaryContainer = RedPale, + secondary = RedVeryLight, + onSecondary = RedSecondaryDark, + secondaryContainer = RedLight, + onSecondaryContainer = RedTertiaryDark, + tertiary = RedPastel, + onTertiary = RedVeryDark, + tertiaryContainer = RedMedium, + onTertiaryContainer = RedPale, + error = RedSoft, + onError = RedVeryDark, + errorContainer = RedMediumDark, + onErrorContainer = RedAlmostWhite, + surfaceDim = SurfaceVeryDark, + surface = SurfaceVeryDark, + surfaceBright = SurfaceBright, + surfaceTint = RedMediumDark, + surfaceContainerLowest = SurfaceDarkest, + surfaceContainerLow = SurfaceDark, + surfaceContainer = SurfaceMediumDark, + surfaceContainerHigh = SurfaceMedium, + surfaceContainerHighest = RedMediumDark, + onSurface = SurfaceVeryPale, + onSurfaceVariant = SurfaceLight, + outline = SurfaceOutlineLight, + outlineVariant = SurfaceOutline, + inverseSurface = SurfaceVeryPale, + inverseOnSurface = SurfaceGray, + inversePrimary = RedAccent, + scrim = PureBlack, + ) + +val extendedColors = + ExtendedColors( + primaryFixed = RedVeryPale, + primaryFixedDim = RedPastel, + onPrimaryFixed = RedDarkest, + onPrimaryFixedVariant = RedMediumDark, + secondaryFixed = RedSecondaryPale, + secondaryFixedDim = RedVeryLight, + onSecondaryFixed = RedSecondaryDarkest, + onSecondaryFixedVariant = RedTertiaryMedium, + tertiaryFixed = RedVeryPale, + tertiaryFixedDim = RedPastel, + onTertiaryFixed = RedDarkest, + onTertiaryFixedVariant = RedMediumDark, + shadow = PureBlack, + accentGreen = AccentGreen, + accentOrange = AccentOrange, + disabledContainer = SurfaceVeryPale, + statusActive = StatusGreen, + statusPending = StatusOrange, + statusInactive = StatusGray, + ) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Theme.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Theme.kt new file mode 100644 index 0000000..6effa28 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Theme.kt @@ -0,0 +1,20 @@ +package io.assessment.contacts.designsystem + +import androidx.compose.foundation.isSystemInDarkTheme +import androidx.compose.material3.MaterialTheme +import androidx.compose.runtime.Composable +import androidx.compose.ui.platform.LocalInspectionMode + +@Composable +fun ContactsTheme( + darkTheme: Boolean = if (LocalInspectionMode.current) isSystemInDarkTheme() else false, + content: @Composable () -> Unit, +) { + val colorScheme = if (darkTheme) DarkColorScheme else LightColorScheme + + MaterialTheme( + colorScheme = colorScheme, + typography = ContactsTypography, + content = content, + ) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Type.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Type.kt new file mode 100644 index 0000000..9573bad --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/designsystem/Type.kt @@ -0,0 +1,115 @@ +package io.assessment.contacts.designsystem + +import androidx.compose.material3.Typography +import androidx.compose.ui.text.TextStyle +import androidx.compose.ui.text.font.FontFamily +import androidx.compose.ui.text.font.FontWeight +import androidx.compose.ui.unit.sp + +val ContactsTypography = Typography( + displayLarge = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 57.sp, + lineHeight = 64.sp, + letterSpacing = (-0.25).sp, + ), + displayMedium = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 45.sp, + lineHeight = 52.sp, + letterSpacing = 0.sp, + ), + displaySmall = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 36.sp, + lineHeight = 44.sp, + letterSpacing = 0.sp, + ), + headlineLarge = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 32.sp, + lineHeight = 40.sp, + letterSpacing = 0.sp, + ), + headlineMedium = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 28.sp, + lineHeight = 36.sp, + letterSpacing = 0.sp, + ), + headlineSmall = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 24.sp, + lineHeight = 32.sp, + letterSpacing = 0.sp, + ), + titleLarge = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 22.sp, + lineHeight = 28.sp, + letterSpacing = 0.sp, + ), + titleMedium = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Medium, + fontSize = 16.sp, + lineHeight = 24.sp, + letterSpacing = 0.15.sp, + ), + titleSmall = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Medium, + fontSize = 14.sp, + lineHeight = 20.sp, + letterSpacing = 0.1.sp, + ), + bodyLarge = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 16.sp, + lineHeight = 24.sp, + letterSpacing = 0.5.sp, + ), + bodyMedium = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 14.sp, + lineHeight = 20.sp, + letterSpacing = 0.25.sp, + ), + bodySmall = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Normal, + fontSize = 12.sp, + lineHeight = 16.sp, + letterSpacing = 0.4.sp, + ), + labelLarge = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Medium, + fontSize = 14.sp, + lineHeight = 20.sp, + letterSpacing = 0.1.sp, + ), + labelMedium = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Medium, + fontSize = 12.sp, + lineHeight = 16.sp, + letterSpacing = 0.5.sp, + ), + labelSmall = TextStyle( + fontFamily = FontFamily.Default, + fontWeight = FontWeight.Medium, + fontSize = 11.sp, + lineHeight = 16.sp, + letterSpacing = 0.5.sp, + ), +) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/di/CoreDataModule.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/di/CoreDataModule.kt new file mode 100644 index 0000000..bffaa00 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/di/CoreDataModule.kt @@ -0,0 +1,33 @@ +package io.assessment.contacts.di + +import io.assessment.contacts.core.data.database.ContactsDatabase +import io.assessment.contacts.core.data.database.DatabaseSeeder +import io.assessment.contacts.core.data.datasource.ContactDataSource +import io.assessment.contacts.core.data.datasource.GroupDataSource +import io.assessment.contacts.core.data.datasource.RoomContactDataSource +import io.assessment.contacts.core.data.datasource.RoomGroupDataSource +import io.assessment.contacts.core.data.repository.ContactRepositoryImpl +import io.assessment.contacts.core.data.repository.GroupRepositoryImpl +import io.assessment.contacts.core.domain.contact.ContactRepository +import io.assessment.contacts.core.domain.group.GroupRepository +import org.koin.core.module.dsl.singleOf +import org.koin.dsl.bind +import org.koin.dsl.module + +val coreDataModule = module { + single { + get().contactDao + } + + single { + get().groupDao + } + + singleOf(::DatabaseSeeder) + + singleOf(::RoomContactDataSource) bind ContactDataSource::class + singleOf(::RoomGroupDataSource) bind GroupDataSource::class + + singleOf(::ContactRepositoryImpl) bind ContactRepository::class + singleOf(::GroupRepositoryImpl) bind GroupRepository::class +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/di/SharedAppModule.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/di/SharedAppModule.kt new file mode 100644 index 0000000..35ae6c6 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/di/SharedAppModule.kt @@ -0,0 +1,13 @@ +package io.assessment.contacts.di + +import io.assessment.contacts.contacts.di.contactsModule +import io.assessment.contacts.groups.di.groupsModule +import org.koin.dsl.module + +val sharedAppModule = module { + includes( + coreDataModule, + contactsModule, + groupsModule, + ) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/data/GroupMapper.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/data/GroupMapper.kt new file mode 100644 index 0000000..df1dd39 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/data/GroupMapper.kt @@ -0,0 +1,20 @@ +package io.assessment.contacts.groups.data + +import io.assessment.contacts.core.data.database.GroupEntity +import io.assessment.contacts.core.domain.group.Group + +fun GroupEntity.toDomain(): Group { + return Group( + id = id, + name = name, + memberCount = memberCount, + ) +} + +fun Group.toEntity(): GroupEntity { + return GroupEntity( + id = id, + name = name, + memberCount = memberCount, + ) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/data/ObserveGroupsUseCaseImpl.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/data/ObserveGroupsUseCaseImpl.kt new file mode 100644 index 0000000..59ea92c --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/data/ObserveGroupsUseCaseImpl.kt @@ -0,0 +1,17 @@ +package io.assessment.contacts.groups.data + +import io.assessment.contacts.core.domain.group.Group +import io.assessment.contacts.core.domain.group.GroupError +import io.assessment.contacts.core.domain.group.GroupRepository +import io.assessment.contacts.core.domain.result.Result +import io.assessment.contacts.groups.domain.ObserveGroupsUseCase +import kotlinx.coroutines.flow.Flow + +class ObserveGroupsUseCaseImpl( + private val groupRepository: GroupRepository, +) : ObserveGroupsUseCase { + + override fun invoke(): Flow, GroupError>> { + return groupRepository.observeGroups() + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/di/GroupsModule.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/di/GroupsModule.kt new file mode 100644 index 0000000..4fb07c3 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/di/GroupsModule.kt @@ -0,0 +1,14 @@ +package io.assessment.contacts.groups.di + +import io.assessment.contacts.groups.domain.ObserveGroupsUseCase +import io.assessment.contacts.groups.data.ObserveGroupsUseCaseImpl +import io.assessment.contacts.groups.presentation.GroupsViewModel +import org.koin.core.module.dsl.factoryOf +import org.koin.core.module.dsl.viewModelOf +import org.koin.dsl.bind +import org.koin.dsl.module + +val groupsModule = module { + factoryOf(::ObserveGroupsUseCaseImpl) bind ObserveGroupsUseCase::class + viewModelOf(::GroupsViewModel) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/domain/ObserveGroupsUseCase.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/domain/ObserveGroupsUseCase.kt new file mode 100644 index 0000000..16979c3 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/domain/ObserveGroupsUseCase.kt @@ -0,0 +1,7 @@ +package io.assessment.contacts.groups.domain + +import io.assessment.contacts.core.domain.group.Group +import io.assessment.contacts.core.domain.group.GroupError +import io.assessment.contacts.core.domain.usecase.NoParamsFlowUseCase + +interface ObserveGroupsUseCase : NoParamsFlowUseCase, GroupError> diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupVO.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupVO.kt new file mode 100644 index 0000000..b80bbdf --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupVO.kt @@ -0,0 +1,17 @@ +package io.assessment.contacts.groups.presentation + +import io.assessment.contacts.core.domain.group.Group + +data class GroupVO( + val id: String, + val name: String, + val memberCount: Int, +) + +fun Group.toVO(): GroupVO { + return GroupVO( + id = id, + name = name, + memberCount = memberCount, + ) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsScreen.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsScreen.kt new file mode 100644 index 0000000..6a87e74 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsScreen.kt @@ -0,0 +1,140 @@ +package io.assessment.contacts.groups.presentation + +import androidx.compose.foundation.background +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.Spacer +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.foundation.layout.width +import androidx.compose.foundation.lazy.LazyColumn +import androidx.compose.foundation.lazy.items +import androidx.compose.foundation.shape.CircleShape +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.material.icons.Icons +import androidx.compose.material.icons.filled.Group +import androidx.compose.material3.Card +import androidx.compose.material3.CardDefaults +import androidx.compose.material3.CircularProgressIndicator +import androidx.compose.material3.ExperimentalMaterial3Api +import androidx.compose.material3.Icon +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Scaffold +import androidx.compose.material3.Text +import androidx.compose.material3.TopAppBar +import androidx.compose.runtime.Composable +import androidx.compose.runtime.getValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.clip +import androidx.compose.ui.unit.dp +import androidx.lifecycle.compose.collectAsStateWithLifecycle +import contactsassessment.composeapp.generated.resources.Res +import contactsassessment.composeapp.generated.resources.groups_member_count +import contactsassessment.composeapp.generated.resources.groups_title +import org.jetbrains.compose.resources.stringResource +import org.koin.compose.viewmodel.koinViewModel + +@OptIn(ExperimentalMaterial3Api::class) +@Composable +fun GroupsScreen( + viewModel: GroupsViewModel = koinViewModel(), +) { + val state by viewModel.state.collectAsStateWithLifecycle() + + GroupsScreenContent(state = state) +} + +@OptIn(ExperimentalMaterial3Api::class) +@Composable +internal fun GroupsScreenContent( + state: GroupsState, +) { + Scaffold( + topBar = { + TopAppBar( + title = { Text(stringResource(Res.string.groups_title)) }, + ) + }, + ) { paddingValues -> + if (state.isLoading) { + Box( + modifier = Modifier + .fillMaxSize() + .padding(paddingValues), + contentAlignment = Alignment.Center, + ) { + CircularProgressIndicator() + } + } else { + LazyColumn( + modifier = Modifier + .fillMaxSize() + .padding(paddingValues), + verticalArrangement = Arrangement.spacedBy(8.dp), + ) { + items( + items = state.groups, + key = { it.id }, + ) { group -> + GroupListItem(group = group) + } + } + } + } +} + +@Composable +internal fun GroupListItem( + group: GroupVO, + modifier: Modifier = Modifier, +) { + Card( + modifier = modifier + .fillMaxWidth() + .padding(horizontal = 16.dp), + shape = RoundedCornerShape(12.dp), + elevation = CardDefaults.cardElevation(defaultElevation = 2.dp), + ) { + Row( + modifier = Modifier + .fillMaxWidth() + .padding(16.dp), + verticalAlignment = Alignment.CenterVertically, + ) { + Box( + modifier = Modifier + .size(48.dp) + .clip(CircleShape) + .background(MaterialTheme.colorScheme.secondaryContainer), + contentAlignment = Alignment.Center, + ) { + Icon( + imageVector = Icons.Default.Group, + contentDescription = null, + tint = MaterialTheme.colorScheme.onSecondaryContainer, + ) + } + + Spacer(modifier = Modifier.width(16.dp)) + + Column( + modifier = Modifier.weight(1f), + ) { + Text( + text = group.name, + style = MaterialTheme.typography.titleMedium, + ) + Text( + text = stringResource(Res.string.groups_member_count, group.memberCount), + style = MaterialTheme.typography.bodyMedium, + color = MaterialTheme.colorScheme.onSurfaceVariant, + ) + } + } + } +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsState.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsState.kt new file mode 100644 index 0000000..916e084 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsState.kt @@ -0,0 +1,6 @@ +package io.assessment.contacts.groups.presentation + +data class GroupsState( + val groups: List = emptyList(), + val isLoading: Boolean = true, +) diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsViewModel.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsViewModel.kt new file mode 100644 index 0000000..3387b61 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/GroupsViewModel.kt @@ -0,0 +1,26 @@ +package io.assessment.contacts.groups.presentation + +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import io.assessment.contacts.core.domain.result.filterSuccess +import io.assessment.contacts.core.presentation.stateInWhileSubscribed +import io.assessment.contacts.groups.domain.ObserveGroupsUseCase +import kotlinx.coroutines.flow.map + +class GroupsViewModel( + observeGroupsUseCase: ObserveGroupsUseCase, +) : ViewModel() { + + val state = observeGroupsUseCase() + .filterSuccess() + .map { groups -> + GroupsState( + groups = groups.map { it.toVO() }, + isLoading = false, + ) + } + .stateInWhileSubscribed( + scope = viewModelScope, + initialValue = GroupsState(), + ) +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/navigation/GroupsNavDestinations.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/navigation/GroupsNavDestinations.kt new file mode 100644 index 0000000..9c8d0d0 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/groups/presentation/navigation/GroupsNavDestinations.kt @@ -0,0 +1,9 @@ +package io.assessment.contacts.groups.presentation.navigation + +import kotlinx.serialization.Serializable + +@Serializable +object GroupsNavDestinations { + @Serializable + object GroupsList +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/navigation/BottomNavDestItem.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/navigation/BottomNavDestItem.kt new file mode 100644 index 0000000..bd68385 --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/navigation/BottomNavDestItem.kt @@ -0,0 +1,41 @@ +package io.assessment.contacts.navigation + +import androidx.compose.material.icons.Icons +import androidx.compose.material.icons.filled.Group +import androidx.compose.material.icons.filled.Person +import androidx.compose.material.icons.outlined.Group +import androidx.compose.material.icons.outlined.Person +import androidx.compose.ui.graphics.vector.ImageVector +import contactsassessment.composeapp.generated.resources.Res +import contactsassessment.composeapp.generated.resources.nav_contacts +import contactsassessment.composeapp.generated.resources.nav_groups +import kotlinx.serialization.Serializable +import org.jetbrains.compose.resources.StringResource + +enum class BottomNavDestItem( + val route: Any, + val labelRes: StringResource, + val selectedIcon: ImageVector, + val unselectedIcon: ImageVector, +) { + Groups( + route = BottomNavRoute.Groups, + labelRes = Res.string.nav_groups, + selectedIcon = Icons.Filled.Group, + unselectedIcon = Icons.Outlined.Group, + ), + Contacts( + route = BottomNavRoute.Contacts, + labelRes = Res.string.nav_contacts, + selectedIcon = Icons.Filled.Person, + unselectedIcon = Icons.Outlined.Person, + ), +} + +@Serializable +sealed interface BottomNavRoute { + @Serializable + data object Groups : BottomNavRoute + @Serializable + data object Contacts : BottomNavRoute +} diff --git a/composeApp/src/commonMain/kotlin/io/assessment/contacts/navigation/NavigationRoot.kt b/composeApp/src/commonMain/kotlin/io/assessment/contacts/navigation/NavigationRoot.kt new file mode 100644 index 0000000..b466dde --- /dev/null +++ b/composeApp/src/commonMain/kotlin/io/assessment/contacts/navigation/NavigationRoot.kt @@ -0,0 +1,73 @@ +package io.assessment.contacts.navigation + +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.padding +import androidx.compose.material3.Icon +import androidx.compose.material3.NavigationBar +import androidx.compose.material3.NavigationBarItem +import androidx.compose.material3.Scaffold +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.getValue +import androidx.compose.ui.Modifier +import androidx.navigation.NavDestination.Companion.hasRoute +import androidx.navigation.NavGraph.Companion.findStartDestination +import androidx.navigation.compose.NavHost +import androidx.navigation.compose.composable +import androidx.navigation.compose.currentBackStackEntryAsState +import androidx.navigation.compose.rememberNavController +import io.assessment.contacts.contacts.presentation.ContactsScreen +import io.assessment.contacts.groups.presentation.GroupsScreen +import org.jetbrains.compose.resources.stringResource + +@Composable +fun NavigationRoot() { + val navController = rememberNavController() + val navBackStackEntry by navController.currentBackStackEntryAsState() + val currentDestination = navBackStackEntry?.destination + + Scaffold( + bottomBar = { + NavigationBar { + BottomNavDestItem.entries.forEach { item -> + val selected = currentDestination?.hasRoute(item.route::class) == true + + NavigationBarItem( + selected = selected, + onClick = { + navController.navigate(item.route) { + popUpTo(navController.graph.findStartDestination().id) { + saveState = true + } + launchSingleTop = true + restoreState = true + } + }, + icon = { + Icon( + imageVector = if (selected) item.selectedIcon else item.unselectedIcon, + contentDescription = stringResource(item.labelRes), + ) + }, + label = { Text(stringResource(item.labelRes)) }, + ) + } + } + }, + ) { paddingValues -> + NavHost( + navController = navController, + startDestination = BottomNavRoute.Groups, + modifier = Modifier + .fillMaxSize() + .padding(paddingValues), + ) { + composable { + GroupsScreen() + } + composable { + ContactsScreen() + } + } + } +} diff --git a/gradle.properties b/gradle.properties new file mode 100644 index 0000000..ca79e37 --- /dev/null +++ b/gradle.properties @@ -0,0 +1,12 @@ +#Kotlin +kotlin.code.style=official +kotlin.daemon.jvmargs=-Xmx2048M + +#Gradle +org.gradle.jvmargs=-Xmx2048M -Dfile.encoding=UTF-8 +org.gradle.configuration-cache=true +org.gradle.caching=true + +#Android +android.nonTransitiveRClass=true +android.useAndroidX=true diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml new file mode 100644 index 0000000..289f1b7 --- /dev/null +++ b/gradle/libs.versions.toml @@ -0,0 +1,77 @@ +[versions] +# Core +agp = "8.7.3" +kotlin = "2.2.0" +ksp = "2.2.0-2.0.2" + +# Android +android-compileSdk = "35" +android-minSdk = "26" +android-targetSdk = "35" + +# Compose & UI +compose-multiplatform = "1.10.0" + +# AndroidX - Core +androidx-activity = "1.10.1" +androidx-lifecycle = "2.9.1" +androidx-navigation = "2.9.0-beta03" + +# Database +room = "2.7.2" +sqlite-bundled = "2.5.2" + +# DI +koin = "4.1.0" + +# Coroutines +kotlinx-coroutines = "1.10.2" + +# Image Loading +coil = "3.3.0" +ktor = "3.1.3" + +[libraries] +# Kotlin +kotlin-test = { module = "org.jetbrains.kotlin:kotlin-test", version.ref = "kotlin" } + +# Coroutines +kotlinx-coroutines-core = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-core", version.ref = "kotlinx-coroutines" } + +# Compose +compose-runtime = { module = "org.jetbrains.compose.runtime:runtime", version.ref = "compose-multiplatform" } +compose-foundation = { module = "org.jetbrains.compose.foundation:foundation", version.ref = "compose-multiplatform" } +compose-ui = { module = "org.jetbrains.compose.ui:ui", version.ref = "compose-multiplatform" } +compose-ui-tooling = { module = "org.jetbrains.compose.ui:ui-tooling", version.ref = "compose-multiplatform" } +compose-ui-tooling-preview = { module = "org.jetbrains.compose.ui:ui-tooling-preview", version.ref = "compose-multiplatform" } + +# Database +room-runtime = { module = "androidx.room:room-runtime", version.ref = "room" } +room-compiler = { module = "androidx.room:room-compiler", version.ref = "room" } +sqlite-bundled = { module = "androidx.sqlite:sqlite-bundled", version.ref = "sqlite-bundled" } + +# DI +koin-core = { module = "io.insert-koin:koin-core", version.ref = "koin" } +koin-android = { module = "io.insert-koin:koin-android", version.ref = "koin" } +koin-compose = { module = "io.insert-koin:koin-compose", version.ref = "koin" } +koin-compose-viewmodel = { module = "io.insert-koin:koin-compose-viewmodel", version.ref = "koin" } + +# AndroidX +androidx-activity-compose = { module = "androidx.activity:activity-compose", version.ref = "androidx-activity" } +androidx-lifecycle-runtime-compose = { module = "org.jetbrains.androidx.lifecycle:lifecycle-runtime-compose", version.ref = "androidx-lifecycle" } +androidx-lifecycle-viewmodel-compose = { module = "org.jetbrains.androidx.lifecycle:lifecycle-viewmodel-compose", version.ref = "androidx-lifecycle" } +androidx-navigation-compose = { module = "org.jetbrains.androidx.navigation:navigation-compose", version.ref = "androidx-navigation" } + +# Image Loading +coil-compose = { module = "io.coil-kt.coil3:coil-compose", version.ref = "coil" } +coil-network-ktor3 = { module = "io.coil-kt.coil3:coil-network-ktor3", version.ref = "coil" } +ktor-client-okhttp = { module = "io.ktor:ktor-client-okhttp", version.ref = "ktor" } + +[plugins] +androidApplication = { id = "com.android.application", version.ref = "agp" } +composeMultiplatform = { id = "org.jetbrains.compose", version.ref = "compose-multiplatform" } +composeCompiler = { id = "org.jetbrains.kotlin.plugin.compose", version.ref = "kotlin" } +kotlinMultiplatform = { id = "org.jetbrains.kotlin.multiplatform", version.ref = "kotlin" } +kotlinSerialization = { id = "org.jetbrains.kotlin.plugin.serialization", version.ref = "kotlin" } +ksp = { id = "com.google.devtools.ksp", version.ref = "ksp" } +room = { id = "androidx.room", version.ref = "room" } diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar new file mode 100644 index 0000000000000000000000000000000000000000..2c3521197d7c4586c843d1d3e9090525f1898cde GIT binary patch literal 43504 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 yu?e%!_ptzGB|8GrCfu%p?(_ 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 TZiJi*F`$tkWLn) literal 0 HcmV?d00001 diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 0000000..09523c0 --- /dev/null +++ b/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,7 @@ +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-bin.zip +networkTimeout=10000 +validateDistributionUrl=true +zipStoreBase=GRADLE_USER_HOME +zipStorePath=wrapper/dists diff --git a/gradlew b/gradlew new file mode 100755 index 0000000..f5feea6 --- /dev/null +++ b/gradlew @@ -0,0 +1,252 @@ +#!/bin/sh + +# +# Copyright © 2015-2021 the original authors. +# +# 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. +# +# SPDX-License-Identifier: Apache-2.0 +# + +############################################################################## +# +# Gradle start up script for POSIX generated by Gradle. +# +# Important for running: +# +# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is +# noncompliant, but you have some other compliant shell such as ksh or +# bash, then to run this script, type that shell name before the whole +# command line, like: +# +# ksh Gradle +# +# Busybox and similar reduced shells will NOT work, because this script +# requires all of these POSIX shell features: +# * functions; +# * expansions «$var», «${var}», «${var:-default}», «${var+SET}», +# «${var#prefix}», «${var%suffix}», and «$( cmd )»; +# * compound commands having a testable exit status, especially «case»; +# * various built-in commands including «command», «set», and «ulimit». +# +# Important for patching: +# +# (2) This script targets any POSIX shell, so it avoids extensions provided +# by Bash, Ksh, etc; in particular arrays are avoided. +# +# The "traditional" practice of packing multiple parameters into a +# space-separated string is a well documented source of bugs and security +# problems, so this is (mostly) avoided, by progressively accumulating +# options in "$@", and eventually passing that to Java. +# +# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, +# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; +# see the in-line comments for details. +# +# There are tweaks for specific operating systems such as AIX, CygWin, +# Darwin, MinGW, and NonStop. +# +# (3) This script is generated from the Groovy template +# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt +# within the Gradle project. +# +# You can find Gradle at https://github.com/gradle/gradle/. +# +############################################################################## + +# Attempt to set APP_HOME + +# Resolve links: $0 may be a link +app_path=$0 + +# Need this for daisy-chained symlinks. +while + APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path + [ -h "$app_path" ] +do + ls=$( ls -ld "$app_path" ) + link=${ls#*' -> '} + case $link in #( + /*) app_path=$link ;; #( + *) app_path=$APP_HOME$link ;; + esac +done + +# This is normally unused +# shellcheck disable=SC2034 +APP_BASE_NAME=${0##*/} +# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) +APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s +' "$PWD" ) || exit + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD=maximum + +warn () { + echo "$*" +} >&2 + +die () { + echo + echo "$*" + echo + exit 1 +} >&2 + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +nonstop=false +case "$( uname )" in #( + CYGWIN* ) cygwin=true ;; #( + Darwin* ) darwin=true ;; #( + MSYS* | MINGW* ) msys=true ;; #( + NONSTOP* ) nonstop=true ;; +esac + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD=$JAVA_HOME/jre/sh/java + else + JAVACMD=$JAVA_HOME/bin/java + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD=java + if ! command -v java >/dev/null 2>&1 + then + die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +fi + +# Increase the maximum file descriptors if we can. +if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then + case $MAX_FD in #( + max*) + # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC2039,SC3045 + MAX_FD=$( ulimit -H -n ) || + warn "Could not query maximum file descriptor limit" + esac + case $MAX_FD in #( + '' | soft) :;; #( + *) + # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC2039,SC3045 + ulimit -n "$MAX_FD" || + warn "Could not set maximum file descriptor limit to $MAX_FD" + esac +fi + +# Collect all arguments for the java command, stacking in reverse order: +# * args from the command line +# * the main class name +# * -classpath +# * -D...appname settings +# * --module-path (only if needed) +# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. + +# For Cygwin or MSYS, switch paths to Windows format before running java +if "$cygwin" || "$msys" ; then + APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) + CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) + + JAVACMD=$( cygpath --unix "$JAVACMD" ) + + # Now convert the arguments - kludge to limit ourselves to /bin/sh + for arg do + if + case $arg in #( + -*) false ;; # don't mess with options #( + /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath + [ -e "$t" ] ;; #( + *) false ;; + esac + then + arg=$( cygpath --path --ignore --mixed "$arg" ) + fi + # Roll the args list around exactly as many times as the number of + # args, so each arg winds up back in the position where it started, but + # possibly modified. + # + # NB: a `for` loop captures its iteration list before it begins, so + # changing the positional parameters here affects neither the number of + # iterations, nor the values presented in `arg`. + shift # remove old arg + set -- "$@" "$arg" # push replacement arg + done +fi + + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + +# Collect all arguments for the java command: +# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments, +# and any embedded shellness will be escaped. +# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be +# treated as '${Hostname}' itself on the command line. + +set -- \ + "-Dorg.gradle.appname=$APP_BASE_NAME" \ + -classpath "$CLASSPATH" \ + org.gradle.wrapper.GradleWrapperMain \ + "$@" + +# Stop when "xargs" is not available. +if ! command -v xargs >/dev/null 2>&1 +then + die "xargs is not available" +fi + +# Use "xargs" to parse quoted args. +# +# With -n1 it outputs one arg per line, with the quotes and backslashes removed. +# +# In Bash we could simply go: +# +# readarray ARGS < <( xargs -n1 <<<"$var" ) && +# set -- "${ARGS[@]}" "$@" +# +# but POSIX shell has neither arrays nor command substitution, so instead we +# post-process each arg (as a line of input to sed) to backslash-escape any +# character that might be a shell metacharacter, then use eval to reverse +# that process (while maintaining the separation between arguments), and wrap +# the whole thing up as a single "set" statement. +# +# This will of course break if any of these variables contains a newline or +# an unmatched quote. +# + +eval "set -- $( + printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | + xargs -n1 | + sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | + tr '\n' ' ' + )" '"$@"' + +exec "$JAVACMD" "$@" diff --git a/gradlew.bat b/gradlew.bat new file mode 100644 index 0000000..9d21a21 --- /dev/null +++ b/gradlew.bat @@ -0,0 +1,94 @@ +@rem +@rem Copyright 2015 the original author or authors. +@rem +@rem Licensed under the Apache License, Version 2.0 (the "License"); +@rem you may not use this file except in compliance with the License. +@rem You may obtain a copy of the License at +@rem +@rem https://www.apache.org/licenses/LICENSE-2.0 +@rem +@rem Unless required by applicable law or agreed to in writing, software +@rem distributed under the License is distributed on an "AS IS" BASIS, +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +@rem See the License for the specific language governing permissions and +@rem limitations under the License. +@rem +@rem SPDX-License-Identifier: Apache-2.0 +@rem + +@if "%DEBUG%"=="" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +set DIRNAME=%~dp0 +if "%DIRNAME%"=="" set DIRNAME=. +@rem This is normally unused +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Resolve any "." and ".." in APP_HOME to make it shorter. +for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if %ERRORLEVEL% equ 0 goto execute + +echo. 1>&2 +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2 +echo. 1>&2 +echo Please set the JAVA_HOME variable in your environment to match the 1>&2 +echo location of your Java installation. 1>&2 + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto execute + +echo. 1>&2 +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2 +echo. 1>&2 +echo Please set the JAVA_HOME variable in your environment to match the 1>&2 +echo location of your Java installation. 1>&2 + +goto fail + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* + +:end +@rem End local scope for the variables with windows NT shell +if %ERRORLEVEL% equ 0 goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +set EXIT_CODE=%ERRORLEVEL% +if %EXIT_CODE% equ 0 set EXIT_CODE=1 +if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% +exit /b %EXIT_CODE% + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/settings.gradle.kts b/settings.gradle.kts new file mode 100644 index 0000000..064bc88 --- /dev/null +++ b/settings.gradle.kts @@ -0,0 +1,35 @@ +rootProject.name = "ContactsAssessment" +enableFeaturePreview("TYPESAFE_PROJECT_ACCESSORS") + +pluginManagement { + repositories { + google { + mavenContent { + includeGroupAndSubgroups("androidx") + includeGroupAndSubgroups("com.android") + includeGroupAndSubgroups("com.google") + } + } + mavenCentral() + gradlePluginPortal() + } +} + +dependencyResolutionManagement { + repositories { + google { + mavenContent { + includeGroupAndSubgroups("androidx") + includeGroupAndSubgroups("com.android") + includeGroupAndSubgroups("com.google") + } + } + mavenCentral() + } +} + +plugins { + id("org.gradle.toolchains.foojay-resolver-convention") version "0.10.0" +} + +include(":composeApp")