├── .gitignore ├── .idea └── .gitignore ├── README.md ├── build.gradle.kts ├── buildSrc ├── build.gradle.kts └── src │ └── main │ └── kotlin │ └── Dependencies.kt ├── data ├── .gitignore ├── build.gradle.kts ├── consumer-rules.pro ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── com │ │ └── charlezz │ │ └── data │ │ └── ExampleInstrumentedTest.kt │ ├── main │ ├── AndroidManifest.xml │ └── java │ │ └── com │ │ └── charlezz │ │ └── data │ │ └── flickr │ │ ├── FlickrPagingSource.kt │ │ ├── FlickrPhoto.kt │ │ ├── FlickrPhotoMapper.kt │ │ ├── FlickrPhotos.kt │ │ ├── FlickrPhotosRepository.kt │ │ ├── FlickrResult.kt │ │ ├── FlickrRetrofitModule.kt │ │ └── FlickrService.kt │ └── test │ └── java │ └── com │ └── charlezz │ └── data │ ├── FlickrPagingSourceTest.kt │ ├── FlickrServiceTest.kt │ └── fake │ └── FakeFlickrService.kt ├── domain ├── .gitignore ├── build.gradle.kts ├── consumer-rules.pro ├── proguard-rules.pro └── src │ └── main │ ├── AndroidManifest.xml │ └── java │ └── com │ └── charlezz │ └── domain │ ├── Photo.kt │ ├── repository │ └── PhotosRepository.kt │ └── usecase │ └── SearchUseCase.kt ├── gradle.properties ├── gradle.properties.kts ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat ├── mvi ├── .gitignore ├── build.gradle.kts ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── com │ │ └── charlezz │ │ └── mvi │ │ └── ExampleInstrumentedTest.kt │ ├── main │ ├── AndroidManifest.xml │ ├── java │ │ └── com │ │ │ └── charlezz │ │ │ └── mvi │ │ │ ├── App.kt │ │ │ ├── di │ │ │ └── AppModules.kt │ │ │ ├── ui │ │ │ ├── BindableViewHolder.kt │ │ │ ├── PhotoAction.kt │ │ │ ├── PhotoActivity.kt │ │ │ ├── PhotoAdapter.kt │ │ │ ├── PhotoState.kt │ │ │ ├── PhotoUiModel.kt │ │ │ ├── PhotoViewModel.kt │ │ │ └── base │ │ │ │ ├── Action.kt │ │ │ │ ├── BaseActivity.kt │ │ │ │ ├── BaseViewModel.kt │ │ │ │ └── State.kt │ │ │ └── util │ │ │ └── ImageViewBindingAdapter.kt │ └── res │ │ ├── drawable-v24 │ │ └── ic_launcher_foreground.xml │ │ ├── drawable │ │ └── ic_launcher_background.xml │ │ ├── layout │ │ ├── activity_photo.xml │ │ └── view_photo.xml │ │ ├── mipmap-anydpi-v26 │ │ ├── ic_launcher.xml │ │ └── ic_launcher_round.xml │ │ ├── mipmap-hdpi │ │ ├── ic_launcher.webp │ │ └── ic_launcher_round.webp │ │ ├── mipmap-mdpi │ │ ├── ic_launcher.webp │ │ └── ic_launcher_round.webp │ │ ├── mipmap-xhdpi │ │ ├── ic_launcher.webp │ │ └── ic_launcher_round.webp │ │ ├── mipmap-xxhdpi │ │ ├── ic_launcher.webp │ │ └── ic_launcher_round.webp │ │ ├── mipmap-xxxhdpi │ │ ├── ic_launcher.webp │ │ └── ic_launcher_round.webp │ │ ├── values-night │ │ └── themes.xml │ │ └── values │ │ ├── colors.xml │ │ ├── strings.xml │ │ └── themes.xml │ └── test │ └── java │ └── com │ └── charlezz │ └── mvi │ └── ExampleUnitTest.kt ├── mvvm ├── .gitignore ├── build.gradle.kts ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── com │ │ └── charlezz │ │ └── mvvm │ │ └── ExampleInstrumentedTest.kt │ ├── main │ ├── AndroidManifest.xml │ ├── java │ │ └── com │ │ │ └── charlezz │ │ │ └── mvvm │ │ │ ├── App.kt │ │ │ ├── di │ │ │ └── AppModules.kt │ │ │ ├── ui │ │ │ ├── BindableViewHolder.kt │ │ │ ├── PhotoActivity.kt │ │ │ ├── PhotoAdapter.kt │ │ │ ├── PhotoUiModel.kt │ │ │ └── PhotoViewModel.kt │ │ │ └── util │ │ │ └── ImageViewBindingAdapter.kt │ └── res │ │ ├── drawable-v24 │ │ └── ic_launcher_foreground.xml │ │ ├── drawable │ │ └── ic_launcher_background.xml │ │ ├── layout │ │ ├── activity_photo.xml │ │ └── view_photo.xml │ │ ├── mipmap-anydpi-v26 │ │ ├── ic_launcher.xml │ │ └── ic_launcher_round.xml │ │ ├── mipmap-hdpi │ │ ├── ic_launcher.webp │ │ └── ic_launcher_round.webp │ │ ├── mipmap-mdpi │ │ ├── ic_launcher.webp │ │ └── ic_launcher_round.webp │ │ ├── mipmap-xhdpi │ │ ├── ic_launcher.webp │ │ └── ic_launcher_round.webp │ │ ├── mipmap-xxhdpi │ │ ├── ic_launcher.webp │ │ └── ic_launcher_round.webp │ │ ├── mipmap-xxxhdpi │ │ ├── ic_launcher.webp │ │ └── ic_launcher_round.webp │ │ ├── values-night │ │ └── themes.xml │ │ └── values │ │ ├── colors.xml │ │ ├── strings.xml │ │ └── themes.xml │ └── test │ └── java │ └── com │ └── charlezz │ └── mvvm │ └── ExampleUnitTest.kt └── settings.gradle.kts /.gitignore: -------------------------------------------------------------------------------- 1 | 2 | # Created by https://www.toptal.com/developers/gitignore/api/androidstudio 3 | # Edit at https://www.toptal.com/developers/gitignore?templates=androidstudio 4 | 5 | ### AndroidStudio ### 6 | # Covers files to be ignored for android development using Android Studio. 7 | 8 | # Built application files 9 | *.apk 10 | *.ap_ 11 | *.aab 12 | 13 | # Files for the ART/Dalvik VM 14 | *.dex 15 | 16 | # Java class files 17 | *.class 18 | 19 | # Generated files 20 | bin/ 21 | gen/ 22 | out/ 23 | 24 | # Gradle files 25 | .gradle 26 | .gradle/ 27 | build/ 28 | 29 | # Signing files 30 | .signing/ 31 | 32 | # Local configuration file (sdk path, etc) 33 | local.properties 34 | 35 | # Proguard folder generated by Eclipse 36 | proguard/ 37 | 38 | # Log Files 39 | *.log 40 | 41 | # Android Studio 42 | /*/build/ 43 | /*/local.properties 44 | /*/out 45 | /*/*/build 46 | /*/*/production 47 | captures/ 48 | .navigation/ 49 | *.ipr 50 | *~ 51 | *.swp 52 | 53 | # Keystore files 54 | *.jks 55 | *.keystore 56 | 57 | # Google Services (e.g. APIs or Firebase) 58 | # google-services.json 59 | 60 | # Android Patch 61 | gen-external-apklibs 62 | 63 | # External native build folder generated in Android Studio 2.2 and later 64 | .externalNativeBuild 65 | 66 | # NDK 67 | obj/ 68 | 69 | # IntelliJ IDEA 70 | *.iml 71 | *.iws 72 | /out/ 73 | 74 | # User-specific configurations 75 | .idea/caches/ 76 | .idea/libraries/ 77 | .idea/shelf/ 78 | .idea/workspace.xml 79 | .idea/tasks.xml 80 | .idea/.name 81 | .idea/compiler.xml 82 | .idea/copyright/profiles_settings.xml 83 | .idea/encodings.xml 84 | .idea/misc.xml 85 | .idea/modules.xml 86 | .idea/scopes/scope_settings.xml 87 | .idea/dictionaries 88 | .idea/vcs.xml 89 | .idea/jsLibraryMappings.xml 90 | .idea/datasources.xml 91 | .idea/dataSources.ids 92 | .idea/sqlDataSources.xml 93 | .idea/dynamic.xml 94 | .idea/uiDesigner.xml 95 | .idea/assetWizardSettings.xml 96 | .idea/gradle.xml 97 | .idea/jarRepositories.xml 98 | .idea/navEditor.xml 99 | 100 | # OS-specific files 101 | .DS_Store 102 | .DS_Store? 103 | ._* 104 | .Spotlight-V100 105 | .Trashes 106 | ehthumbs.db 107 | Thumbs.db 108 | 109 | # Legacy Eclipse project files 110 | .classpath 111 | .project 112 | .cproject 113 | .settings/ 114 | 115 | # Mobile Tools for Java (J2ME) 116 | .mtj.tmp/ 117 | 118 | # Package Files # 119 | *.war 120 | *.ear 121 | 122 | # virtual machine crash logs (Reference: http://www.java.com/en/download/help/error_hotspot.xml) 123 | hs_err_pid* 124 | 125 | ## Plugin-specific files: 126 | 127 | # mpeltonen/sbt-idea plugin 128 | .idea_modules/ 129 | 130 | # JIRA plugin 131 | atlassian-ide-plugin.xml 132 | 133 | # Mongo Explorer plugin 134 | .idea/mongoSettings.xml 135 | 136 | # Crashlytics plugin (for Android Studio and IntelliJ) 137 | com_crashlytics_export_strings.xml 138 | crashlytics.properties 139 | crashlytics-build.properties 140 | fabric.properties 141 | 142 | ### AndroidStudio Patch ### 143 | 144 | !/gradle/wrapper/gradle-wrapper.jar 145 | 146 | # End of https://www.toptal.com/developers/gitignore/api/androidstudio -------------------------------------------------------------------------------- /.idea/.gitignore: -------------------------------------------------------------------------------- 1 | # Default ignored files 2 | /shelf/ 3 | /workspace.xml 4 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | You need an API-Key to build. Please refer the link below 2 | 3 | https://www.flickr.com/services/api/misc.api_keys.html 4 | 5 | The API key must be in local.properties 6 | 7 | ``` 8 | #local.properties 9 | apiKey=YOUR_FLICKR_API_KEY 10 | ``` 11 | 12 | -------------------------------------------------------------------------------- /build.gradle.kts: -------------------------------------------------------------------------------- 1 | buildscript { 2 | repositories { 3 | google() 4 | mavenCentral() 5 | } 6 | dependencies { 7 | classpath("com.android.tools.build:gradle:7.0.2") 8 | classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:${Dependencies.Kotlin.VERSION}") 9 | classpath ("com.google.dagger:hilt-android-gradle-plugin:${Dependencies.Hilt.VERSION}") 10 | } 11 | } 12 | -------------------------------------------------------------------------------- /buildSrc/build.gradle.kts: -------------------------------------------------------------------------------- 1 | plugins { 2 | `kotlin-dsl` 3 | } 4 | 5 | repositories { 6 | mavenCentral() 7 | } -------------------------------------------------------------------------------- /buildSrc/src/main/kotlin/Dependencies.kt: -------------------------------------------------------------------------------- 1 | import org.gradle.kotlin.dsl.DependencyHandlerScope 2 | import org.gradle.kotlin.dsl.exclude 3 | import org.gradle.kotlin.dsl.project 4 | 5 | object Dependencies { 6 | 7 | const val COMPILE_SDK = 31 8 | const val MIN_SDK = 26 9 | const val TARGET_SDK = 31 10 | 11 | private const val implementation = "implementation" 12 | private const val testImplementation = "testImplementation" 13 | private const val androidTestImplementation = "androidTestImplementation" 14 | 15 | private const val kapt = "kapt" 16 | private const val kaptTest = "kaptTest" 17 | private const val kaptAndroidTest = "kaptAndroidTest" 18 | 19 | 20 | 21 | object Kotlin{ 22 | const val VERSION = "1.5.21" 23 | const val STD_LIB = "org.jetbrains.kotlin:kotlin-stdlib:$VERSION" 24 | const val TEST = "org.jetbrains.kotlin:kotlin-test-junit:$VERSION" 25 | 26 | const val COROUTINE = "org.jetbrains.kotlinx:kotlinx-coroutines-android:$VERSION" 27 | const val COROUTINE_TEST = "org.jetbrains.kotlinx:kotlinx-coroutines-test:1.5.1" 28 | const val COROUTINE_CORE = "org.jetbrains.kotlinx:kotlinx-coroutines-core:$VERSION" 29 | } 30 | object Google{ 31 | const val MATERIAL = "com.google.android.material:material:1.4.0" 32 | } 33 | 34 | object AndroidX { 35 | const val APPCOMPAT = "androidx.appcompat:appcompat:1.3.1" 36 | const val RECYCLER_VIEW = "androidx.recyclerview:recyclerview:1.2.0" 37 | const val CONSTRAINT_LAYOUT = "androidx.constraintlayout:constraintlayout:2.1.0" 38 | const val ACTIVITY = "androidx.activity:activity-ktx:1.2.2" 39 | const val FRAGMENT = "androidx.fragment:fragment-ktx:1.3.2" 40 | const val CORE = "androidx.core:core-ktx:1.6.0" 41 | 42 | private const val VERSION_PAGING = "3.0.0" 43 | const val PAGING_RUNTIME = "androidx.paging:paging-runtime-ktx:$VERSION_PAGING" 44 | const val PAGING_COMMON = "androidx.paging:paging-common:$VERSION_PAGING" 45 | 46 | } 47 | 48 | fun DependencyHandlerScope.applyAndroidX(){ 49 | implementation(AndroidX.APPCOMPAT) 50 | implementation(AndroidX.CORE) 51 | implementation(AndroidX.ACTIVITY) 52 | implementation(AndroidX.FRAGMENT) 53 | implementation(AndroidX.RECYCLER_VIEW) 54 | implementation(AndroidX.CONSTRAINT_LAYOUT) 55 | implementation(AndroidX.PAGING_RUNTIME) 56 | } 57 | 58 | object Retrofit2{ 59 | const val VERSION = "2.9.0" 60 | const val CORE = "com.squareup.retrofit2:retrofit:${VERSION}" 61 | const val MOSHI = "com.squareup.retrofit2:converter-moshi:${VERSION}" 62 | const val SCALARS = "com.squareup.retrofit2:converter-scalars:${VERSION}" 63 | } 64 | 65 | fun DependencyHandlerScope.applyRetrofit2(){ 66 | implementation(Retrofit2.CORE) 67 | implementation(Retrofit2.MOSHI) 68 | implementation(Retrofit2.SCALARS) 69 | } 70 | 71 | 72 | // object Lifecycle{ 73 | // const val VIEWMODEL = "androidx.lifecycle:lifecycle-viewmodel-ktx:${Versions.lifecycle}" 74 | // const val VIEWMODEL_SAVEDSTATE = "androidx.lifecycle:lifecycle-viewmodel-savedstate:${Versions.lifecycle}" 75 | // const val RUNTIME = "androidx.lifecycle:lifecycle-runtime-ktx:${Versions.lifecycle}" 76 | // const val EXTENSIONS = "androidx.lifecycle:lifecycle-extensions:${Versions.lifecycleExt}" 77 | // const val COMMON_JAVA8 = "androidx.lifecycle:lifecycle-common-java8:${Versions.lifecycle}" 78 | // } 79 | // 80 | // fun DependencyHandlerScope.applyLifecycle(){ 81 | // implementation(Lifecycle.VIEWMODEL) 82 | // implementation(Lifecycle.VIEWMODEL_SAVEDSTATE) 83 | // implementation(Lifecycle.COMMON_JAVA8) 84 | // implementation(Lifecycle.EXTENSIONS) 85 | // implementation(Lifecycle.RUNTIME) 86 | // } 87 | 88 | // object Navigation{ 89 | // const val FRAGMENT = "androidx.navigation:navigation-fragment-ktx:${Versions.navigation}" 90 | // const val UI = "androidx.navigation:navigation-ui-ktx:${Versions.navigation}" 91 | // const val DYNAMIC_FEATURE_FRAGMENT = "androidx.navigation:navigation-dynamic-features-fragment:${Versions.navigation}" 92 | // } 93 | // 94 | // fun DependencyHandlerScope.applyNavigation(){ 95 | // implementation(Navigation.FRAGMENT) 96 | // implementation(Navigation.UI) 97 | // implementation(Navigation.DYNAMIC_FEATURE_FRAGMENT) 98 | // } 99 | 100 | object Glide{ 101 | const val VERSION = "4.12.0" 102 | const val CORE = "com.github.bumptech.glide:glide:$VERSION" 103 | const val COMPILER = "com.github.bumptech.glide:compiler:$VERSION" 104 | const val OKHTTP3_INTEGRATION = "com.github.bumptech.glide:okhttp3-integration:$VERSION" 105 | const val TRANSFORMATIONS = "jp.wasabeef:glide-transformations:4.3.0" 106 | } 107 | 108 | fun DependencyHandlerScope.applyGlide(){ 109 | implementation(Glide.CORE) 110 | kapt(Glide.COMPILER) 111 | implementation(Glide.OKHTTP3_INTEGRATION) 112 | } 113 | 114 | // object JakeWharton{ 115 | // const val THREE_TEN_ABP = "com.jakewharton.threetenabp:threetenabp:${Versions.threetenabp}" 116 | // const val TIMBER = "com.jakewharton.timber:timber:${Versions.timber}" 117 | // const val RXRELAY = "com.jakewharton.rxrelay2:rxrelay:${Versions.rxrelay2}" 118 | // const val RXBINDING = "com.jakewharton.rxbinding2:rxbinding:${Versions.rxbinding}" 119 | // } 120 | 121 | object Test{ 122 | const val JUNIT = "junit:junit:4.13.2" 123 | const val ANDROID_EXT_JUNIT = "androidx.test.ext:junit:1.1.3" 124 | const val ANDROID_ESPRESSO_CORE = "androidx.test.espresso:espresso-core:3.4.0" 125 | } 126 | 127 | fun DependencyHandlerScope.applyTest(){ 128 | Dependencies.testImplementation(Test.JUNIT) 129 | androidTestImplementation(Test.ANDROID_EXT_JUNIT) 130 | androidTestImplementation(Test.ANDROID_ESPRESSO_CORE) 131 | } 132 | 133 | object Hilt{ 134 | const val VERSION = "2.38.1" 135 | 136 | const val CORE = "com.google.dagger:hilt-android:$VERSION" 137 | const val COMPILER = "com.google.dagger:hilt-compiler:$VERSION" 138 | 139 | // For instrumentation tests 140 | const val ANDROID_TESTING = "com.google.dagger:hilt-android-testing:$VERSION" 141 | const val ANDROID_TESTING_COMPILER = "com.google.dagger:hilt-compiler:$VERSION" 142 | 143 | // For local unit tests 144 | const val LOCAL_TESTING = "com.google.dagger:hilt-android-testing:$VERSION" 145 | const val LOCAL_TESTING_COMPILER = "com.google.dagger:hilt-compiler:$VERSION" 146 | } 147 | 148 | fun DependencyHandlerScope.applyHilt(){ 149 | implementation(Hilt.CORE) 150 | kapt(Hilt.COMPILER) 151 | 152 | kaptTest(Hilt.LOCAL_TESTING_COMPILER) 153 | testImplementation(Hilt.LOCAL_TESTING) 154 | 155 | androidTestImplementation(Hilt.ANDROID_TESTING) 156 | kaptAndroidTest(Hilt.ANDROID_TESTING_COMPILER) 157 | } 158 | 159 | 160 | } -------------------------------------------------------------------------------- /data/.gitignore: -------------------------------------------------------------------------------- 1 | /build -------------------------------------------------------------------------------- /data/build.gradle.kts: -------------------------------------------------------------------------------- 1 | import Dependencies.applyAndroidX 2 | import Dependencies.applyHilt 3 | import Dependencies.applyRetrofit2 4 | import Dependencies.applyTest 5 | import com.android.build.gradle.internal.cxx.configure.gradleLocalProperties 6 | 7 | plugins { 8 | id ("com.android.library") 9 | id ("kotlin-android") 10 | id("kotlin-kapt") 11 | } 12 | 13 | android { 14 | compileSdk = Dependencies.COMPILE_SDK 15 | 16 | buildFeatures { 17 | dataBinding = true 18 | } 19 | 20 | defaultConfig { 21 | minSdk = Dependencies.MIN_SDK 22 | targetSdk = Dependencies.TARGET_SDK 23 | 24 | buildConfigField(String::class.java.canonicalName, "API_KEY", "\"${gradleLocalProperties(rootDir)["apiKey"] as String}\"") 25 | 26 | testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" 27 | consumerProguardFiles("consumer-rules.pro") 28 | } 29 | 30 | buildTypes { 31 | release { 32 | isMinifyEnabled = false 33 | proguardFiles(getDefaultProguardFile("proguard-android-optimize.txt"), "proguard-rules.pro") 34 | } 35 | } 36 | compileOptions { 37 | sourceCompatibility = JavaVersion.VERSION_11 38 | targetCompatibility = JavaVersion.VERSION_11 39 | } 40 | } 41 | 42 | dependencies { 43 | implementation (project(":domain")) 44 | 45 | testImplementation(Dependencies.Kotlin.COROUTINE_TEST) 46 | testImplementation(Dependencies.Kotlin.TEST) 47 | 48 | implementation(Dependencies.Google.MATERIAL) 49 | implementation(Dependencies.AndroidX.PAGING_RUNTIME) 50 | applyAndroidX() 51 | applyTest() 52 | applyRetrofit2() 53 | applyHilt() 54 | } -------------------------------------------------------------------------------- /data/consumer-rules.pro: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AndroidDeepDive/ADD-Architecture-Pattern-Sample/20149a5eaca1aedffb5c63002ef9d39aed9b34f2/data/consumer-rules.pro -------------------------------------------------------------------------------- /data/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # You can control the set of applied configuration files using the 3 | # proguardFiles setting in build.gradle. 4 | # 5 | # For more details, see 6 | # http://developer.android.com/guide/developing/tools/proguard.html 7 | 8 | # If your project uses WebView with JS, uncomment the following 9 | # and specify the fully qualified class name to the JavaScript interface 10 | # class: 11 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 12 | # public *; 13 | #} 14 | 15 | # Uncomment this to preserve the line number information for 16 | # debugging stack traces. 17 | #-keepattributes SourceFile,LineNumberTable 18 | 19 | # If you keep the line number information, uncomment this to 20 | # hide the original source file name. 21 | #-renamesourcefileattribute SourceFile -------------------------------------------------------------------------------- /data/src/androidTest/java/com/charlezz/data/ExampleInstrumentedTest.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data 2 | 3 | import androidx.test.platform.app.InstrumentationRegistry 4 | import androidx.test.ext.junit.runners.AndroidJUnit4 5 | 6 | import org.junit.Test 7 | import org.junit.runner.RunWith 8 | 9 | import org.junit.Assert.* 10 | 11 | /** 12 | * Instrumented test, which will execute on an Android device. 13 | * 14 | * See [testing documentation](http://d.android.com/tools/testing). 15 | */ 16 | @RunWith(AndroidJUnit4::class) 17 | class ExampleInstrumentedTest { 18 | @Test 19 | fun useAppContext(){ 20 | // Context of the app under test. 21 | val appContext = InstrumentationRegistry.getInstrumentation().targetContext 22 | assertEquals("com.charlezz.data.test", appContext.packageName) 23 | } 24 | } -------------------------------------------------------------------------------- /data/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | -------------------------------------------------------------------------------- /data/src/main/java/com/charlezz/data/flickr/FlickrPagingSource.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data.flickr 2 | 3 | import androidx.paging.PagingSource 4 | import androidx.paging.PagingState 5 | import com.charlezz.domain.Photo 6 | import kotlinx.coroutines.Dispatchers 7 | import kotlinx.coroutines.withContext 8 | 9 | class FlickrPagingSource( 10 | private val service: FlickrService, 11 | private val keyword: String? 12 | ) : PagingSource() { 13 | 14 | companion object { 15 | private const val FIRST_PAGE = 1 16 | } 17 | 18 | override fun getRefreshKey(state: PagingState): Int? { 19 | return state.anchorPosition?.let { anchorPosition -> 20 | val anchorPage = state.closestPageToPosition(anchorPosition) 21 | anchorPage?.prevKey?.plus(1) ?: anchorPage?.nextKey?.minus(1) 22 | } 23 | } 24 | 25 | override suspend fun load(params: LoadParams): LoadResult { 26 | 27 | val loadSize = params.loadSize 28 | val key = params.key ?: FIRST_PAGE 29 | 30 | return withContext(Dispatchers.IO){ 31 | val result = if (keyword.isNullOrEmpty()) { 32 | service.getRecent(key, loadSize) 33 | } else { 34 | service.search(keyword, key, loadSize) 35 | } 36 | val data = result.photos.photo.map { FlickrPhotoMapper.toModel(it) } 37 | 38 | val prevKey: Int? = if (result.photos.page == 1) { // first page 39 | null 40 | } else { 41 | result.photos.page - 1 42 | } 43 | 44 | val nextKey: Int? = if (result.photos.page < result.photos.pages) { // last page 45 | result.photos.page + 1 46 | } else { 47 | null 48 | } 49 | LoadResult.Page(data, prevKey, nextKey) 50 | } 51 | 52 | } 53 | } -------------------------------------------------------------------------------- /data/src/main/java/com/charlezz/data/flickr/FlickrPhoto.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data.flickr 2 | 3 | import com.squareup.moshi.Json 4 | 5 | data class FlickrPhoto( 6 | val id:Long, 7 | val owner:String, 8 | val secret:String, 9 | val server:String, 10 | val farm: Int, 11 | val title:String, 12 | val ispublic:Int, 13 | @Json(name = "ispublic") val isPublic:Int, 14 | @Json(name = "isfriend") val isFriend:Int, 15 | @Json(name = "isfamily") val isFamily:Int, 16 | ) -------------------------------------------------------------------------------- /data/src/main/java/com/charlezz/data/flickr/FlickrPhotoMapper.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data.flickr 2 | 3 | import com.charlezz.domain.Photo 4 | 5 | object FlickrPhotoMapper{ 6 | fun toModel(dto:FlickrPhoto):Photo{ 7 | return Photo( 8 | title = dto.title, 9 | url = "https://live.staticflickr.com/${dto.server}/${dto.id}_${dto.secret}.jpg" 10 | ) 11 | } 12 | } 13 | 14 | -------------------------------------------------------------------------------- /data/src/main/java/com/charlezz/data/flickr/FlickrPhotos.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data.flickr 2 | 3 | class FlickrPhotos( 4 | val page: Int, 5 | val pages: Int, 6 | val perpage: Int, 7 | val total: Int, 8 | val photo: List, 9 | ) -------------------------------------------------------------------------------- /data/src/main/java/com/charlezz/data/flickr/FlickrPhotosRepository.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data.flickr 2 | 3 | import androidx.paging.Pager 4 | import androidx.paging.PagingConfig 5 | import androidx.paging.PagingData 6 | import com.charlezz.domain.Photo 7 | import com.charlezz.domain.repository.PhotosRepository 8 | import kotlinx.coroutines.flow.Flow 9 | 10 | class FlickrPhotosRepository(private val service: FlickrService) : PhotosRepository { 11 | override suspend fun search(keyword: String?): Flow> { 12 | return Pager(PagingConfig(30)){ 13 | FlickrPagingSource(service, keyword) 14 | }.flow 15 | } 16 | 17 | } -------------------------------------------------------------------------------- /data/src/main/java/com/charlezz/data/flickr/FlickrResult.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data.flickr 2 | 3 | /** 4 | * "photos": { 5 | "page": 1, 6 | "pages": 1154, 7 | "perpage": 100, 8 | "total": 115323, 9 | "photo": [ 10 | { 11 | "id": "51405312918", 12 | "owner": "153499399@N02", 13 | "secret": "b11c94e3d9", 14 | "server": "65535", 15 | "farm": 66, 16 | "title": "Dollar Tree-Spirit Lake, Iowa", 17 | "ispublic": 1, 18 | "isfriend": 0, 19 | "isfamily": 0 20 | }, ... 21 | ] 22 | }, 23 | "stat": "ok" 24 | */ 25 | data class FlickrResult( 26 | val photos:FlickrPhotos, 27 | val stat: String, 28 | ) -------------------------------------------------------------------------------- /data/src/main/java/com/charlezz/data/flickr/FlickrRetrofitModule.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data.flickr 2 | 3 | import com.charlezz.data.BuildConfig 4 | import dagger.Module 5 | import dagger.Provides 6 | import dagger.hilt.InstallIn 7 | import dagger.hilt.components.SingletonComponent 8 | import okhttp3.Interceptor 9 | import okhttp3.OkHttpClient 10 | import retrofit2.Retrofit 11 | import retrofit2.converter.moshi.MoshiConverterFactory 12 | import retrofit2.converter.scalars.ScalarsConverterFactory 13 | 14 | 15 | @InstallIn(SingletonComponent::class) 16 | @Module 17 | object FlickrRetrofitModule { 18 | 19 | @Provides 20 | fun providesIntercepter():Interceptor{ 21 | 22 | return Interceptor { chain-> 23 | var request = chain.request() 24 | val newUrl = request.url().newBuilder() 25 | .addQueryParameter("api_key", "${BuildConfig.API_KEY}") 26 | .addQueryParameter("format", "json") 27 | .addQueryParameter("nojsoncallback","1") 28 | .build() 29 | 30 | request = request.newBuilder().url(newUrl).build() 31 | chain.proceed(request) 32 | } 33 | } 34 | 35 | @Provides 36 | fun providesOkHttpClient(interceptor: Interceptor):OkHttpClient{ 37 | return OkHttpClient.Builder() 38 | .addInterceptor(interceptor) 39 | .build() 40 | } 41 | 42 | @Provides 43 | fun providesRetrofit(okHttpClient: OkHttpClient):Retrofit{ 44 | return Retrofit.Builder() 45 | .baseUrl("https://www.flickr.com/services/rest/") 46 | .client(okHttpClient) 47 | .addConverterFactory(MoshiConverterFactory.create()) 48 | .addConverterFactory(ScalarsConverterFactory.create()) 49 | .build() 50 | } 51 | } -------------------------------------------------------------------------------- /data/src/main/java/com/charlezz/data/flickr/FlickrService.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data.flickr 2 | 3 | import retrofit2.Call 4 | import retrofit2.http.GET 5 | import retrofit2.http.Query 6 | 7 | interface FlickrService { 8 | 9 | @GET("?method=flickr.photos.search") 10 | suspend fun search( 11 | @Query("text") keyword: String, 12 | @Query("page") page: Int = 1, 13 | @Query("per_page") perPage: Int = 100 14 | ): FlickrResult 15 | 16 | @GET("?method=flickr.photos.getRecent") 17 | suspend fun getRecent( 18 | @Query("page") page: Int = 1, 19 | @Query("per_page") perPage: Int = 100 20 | ): FlickrResult 21 | 22 | 23 | } -------------------------------------------------------------------------------- /data/src/test/java/com/charlezz/data/FlickrPagingSourceTest.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data 2 | 3 | import androidx.paging.PagingSource 4 | import com.charlezz.data.fake.FakeFlickrService 5 | import com.charlezz.data.flickr.FlickrPagingSource 6 | import com.charlezz.data.flickr.FlickrPhoto 7 | import com.charlezz.data.flickr.FlickrPhotoMapper 8 | import com.charlezz.domain.Photo 9 | import kotlinx.coroutines.runBlocking 10 | import org.junit.Test 11 | import kotlin.random.Random 12 | import kotlin.test.assertEquals 13 | 14 | class FlickrPagingSourceTest { 15 | 16 | private val fakePhotos = listOf( 17 | createPhoto(), 18 | createPhoto(), 19 | createPhoto(), 20 | ) 21 | private val fakeService = FakeFlickrService().apply { 22 | fakePhotos.forEach{ photo -> addPhoto(photo)} 23 | } 24 | 25 | @Test 26 | fun pagingSourceTest() = runBlocking { 27 | val pagingSource = FlickrPagingSource(fakeService, "") 28 | assertEquals( 29 | expected = PagingSource.LoadResult.Page( 30 | data = listOf(FlickrPhotoMapper.toModel(fakePhotos[0]),FlickrPhotoMapper.toModel(fakePhotos[1])), 31 | prevKey = null, 32 | nextKey = null, 33 | ), 34 | actual = pagingSource.load( 35 | PagingSource.LoadParams.Refresh( 36 | key = null, 37 | loadSize = 2, 38 | placeholdersEnabled = false 39 | ) 40 | ) 41 | ) 42 | } 43 | 44 | private fun createPhoto(): FlickrPhoto { 45 | return FlickrPhoto( 46 | Random.nextLong(), 47 | "owner", 48 | "owner", 49 | "server", 50 | 0, 51 | "title", 52 | 0, 53 | 0, 54 | 0, 55 | 0 56 | ) 57 | 58 | } 59 | 60 | } -------------------------------------------------------------------------------- /data/src/test/java/com/charlezz/data/FlickrServiceTest.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data 2 | 3 | import com.charlezz.data.flickr.FlickrRetrofitModule 4 | import com.charlezz.data.flickr.FlickrService 5 | import kotlinx.coroutines.runBlocking 6 | import org.junit.Before 7 | import org.junit.Test 8 | 9 | class FlickrServiceTest { 10 | 11 | lateinit var flickrService: FlickrService 12 | 13 | @Before 14 | fun setup() { 15 | val intercepter = FlickrRetrofitModule.providesIntercepter() 16 | val okHttpClient = FlickrRetrofitModule.providesOkHttpClient(intercepter) 17 | val retrofit = FlickrRetrofitModule.providesRetrofit(okHttpClient) 18 | this.flickrService = retrofit.create(FlickrService::class.java) 19 | } 20 | 21 | @Test 22 | fun getRecentTest() = runBlocking{ 23 | val result = flickrService.getRecent() 24 | assert(result.stat == "ok") 25 | } 26 | 27 | @Test 28 | fun searchTest() = runBlocking{ 29 | val result = flickrService.search("tree") 30 | assert(result.stat == "ok") 31 | } 32 | 33 | } -------------------------------------------------------------------------------- /data/src/test/java/com/charlezz/data/fake/FakeFlickrService.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.data.fake 2 | 3 | import com.charlezz.data.flickr.FlickrPhoto 4 | import com.charlezz.data.flickr.FlickrPhotos 5 | import com.charlezz.data.flickr.FlickrResult 6 | import com.charlezz.data.flickr.FlickrService 7 | import kotlin.math.min 8 | 9 | class FakeFlickrService : FlickrService { 10 | 11 | private val result = ArrayList() 12 | 13 | fun addPhoto(flickrPhoto: FlickrPhoto) { 14 | result.add(flickrPhoto) 15 | } 16 | 17 | override suspend fun search(keyword: String, page: Int, perPage: Int): FlickrResult { 18 | return getPagedList(page, perPage) 19 | } 20 | 21 | override suspend fun getRecent(page: Int, perPage: Int): FlickrResult { 22 | return getPagedList(page, perPage) 23 | } 24 | 25 | private fun getPagedList(page: Int, perpage: Int): FlickrResult { 26 | val fromIndex = 100 * (page - 1) 27 | val toIndex = min(fromIndex + perpage, result.size) 28 | val photos = result.subList(fromIndex, toIndex) 29 | return FlickrResult( 30 | FlickrPhotos( 31 | page = page, 32 | pages = result.size / perpage, 33 | perpage = perpage, 34 | result.size, 35 | photos 36 | ), "ok" 37 | ) 38 | } 39 | 40 | } -------------------------------------------------------------------------------- /domain/.gitignore: -------------------------------------------------------------------------------- 1 | /build -------------------------------------------------------------------------------- /domain/build.gradle.kts: -------------------------------------------------------------------------------- 1 | plugins { 2 | id("java-library") 3 | id("kotlin") 4 | } 5 | 6 | java { 7 | sourceCompatibility = JavaVersion.VERSION_11 8 | targetCompatibility = JavaVersion.VERSION_11 9 | } 10 | 11 | dependencies { 12 | implementation(Dependencies.AndroidX.PAGING_COMMON) 13 | } -------------------------------------------------------------------------------- /domain/consumer-rules.pro: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AndroidDeepDive/ADD-Architecture-Pattern-Sample/20149a5eaca1aedffb5c63002ef9d39aed9b34f2/domain/consumer-rules.pro -------------------------------------------------------------------------------- /domain/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # You can control the set of applied configuration files using the 3 | # proguardFiles setting in build.gradle. 4 | # 5 | # For more details, see 6 | # http://developer.android.com/guide/developing/tools/proguard.html 7 | 8 | # If your project uses WebView with JS, uncomment the following 9 | # and specify the fully qualified class name to the JavaScript interface 10 | # class: 11 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 12 | # public *; 13 | #} 14 | 15 | # Uncomment this to preserve the line number information for 16 | # debugging stack traces. 17 | #-keepattributes SourceFile,LineNumberTable 18 | 19 | # If you keep the line number information, uncomment this to 20 | # hide the original source file name. 21 | #-renamesourcefileattribute SourceFile -------------------------------------------------------------------------------- /domain/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | -------------------------------------------------------------------------------- /domain/src/main/java/com/charlezz/domain/Photo.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.domain 2 | 3 | data class Photo( 4 | val title:String, 5 | val url:String 6 | ) -------------------------------------------------------------------------------- /domain/src/main/java/com/charlezz/domain/repository/PhotosRepository.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.domain.repository 2 | 3 | import androidx.paging.PagingData 4 | import com.charlezz.domain.Photo 5 | import kotlinx.coroutines.flow.Flow 6 | 7 | interface PhotosRepository { 8 | 9 | suspend fun search(keyword: String?): Flow> 10 | 11 | } -------------------------------------------------------------------------------- /domain/src/main/java/com/charlezz/domain/usecase/SearchUseCase.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.domain.usecase 2 | 3 | import androidx.paging.PagingData 4 | import com.charlezz.domain.Photo 5 | import com.charlezz.domain.repository.PhotosRepository 6 | import kotlinx.coroutines.flow.Flow 7 | 8 | class SearchUseCase(private val photosRepository: PhotosRepository) { 9 | suspend operator fun invoke(keyword:String?):Flow>{ 10 | return photosRepository.search(keyword) 11 | } 12 | } -------------------------------------------------------------------------------- /gradle.properties: -------------------------------------------------------------------------------- 1 | android.useAndroidX=true 2 | android.enableJetifier=true -------------------------------------------------------------------------------- /gradle.properties.kts: -------------------------------------------------------------------------------- 1 | # Project-wide Gradle settings. 2 | # IDE (e.g. Android Studio) users: 3 | # Gradle settings configured through the IDE *will override* 4 | # any settings specified in this file. 5 | # For more details on how to configure your build environment visit 6 | # http://www.gradle.org/docs/current/userguide/build_environment.html 7 | # Specifies the JVM arguments used for the daemon process. 8 | # The setting is particularly useful for tweaking memory settings. 9 | org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8 10 | # When configured, Gradle will run in incubating parallel mode. 11 | # This option should only be used with decoupled projects. More details, visit 12 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects 13 | # org.gradle.parallel=true 14 | # AndroidX package structure to make it clearer which packages are bundled with the 15 | # Android operating system, and which are packaged with your app"s APK 16 | # https://developer.android.com/topic/libraries/support-library/androidx-rn 17 | android.useAndroidX=true 18 | # Automatically convert third-party libraries to use AndroidX 19 | android.enableJetifier=true 20 | # Kotlin code style for this project: "official" or "obsolete": 21 | kotlin.code.style=official -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AndroidDeepDive/ADD-Architecture-Pattern-Sample/20149a5eaca1aedffb5c63002ef9d39aed9b34f2/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | #Wed Aug 25 18:28:58 KST 2021 2 | distributionBase=GRADLE_USER_HOME 3 | distributionUrl=https\://services.gradle.org/distributions/gradle-7.0.2-bin.zip 4 | distributionPath=wrapper/dists 5 | zipStorePath=wrapper/dists 6 | zipStoreBase=GRADLE_USER_HOME 7 | -------------------------------------------------------------------------------- /gradlew: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env sh 2 | 3 | # 4 | # Copyright 2015 the original author or authors. 5 | # 6 | # Licensed under the Apache License, Version 2.0 (the "License"); 7 | # you may not use this file except in compliance with the License. 8 | # You may obtain a copy of the License at 9 | # 10 | # https://www.apache.org/licenses/LICENSE-2.0 11 | # 12 | # Unless required by applicable law or agreed to in writing, software 13 | # distributed under the License is distributed on an "AS IS" BASIS, 14 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 15 | # See the License for the specific language governing permissions and 16 | # limitations under the License. 17 | # 18 | 19 | ############################################################################## 20 | ## 21 | ## Gradle start up script for UN*X 22 | ## 23 | ############################################################################## 24 | 25 | # Attempt to set APP_HOME 26 | # Resolve links: $0 may be a link 27 | PRG="$0" 28 | # Need this for relative symlinks. 29 | while [ -h "$PRG" ] ; do 30 | ls=`ls -ld "$PRG"` 31 | link=`expr "$ls" : '.*-> \(.*\)$'` 32 | if expr "$link" : '/.*' > /dev/null; then 33 | PRG="$link" 34 | else 35 | PRG=`dirname "$PRG"`"/$link" 36 | fi 37 | done 38 | SAVED="`pwd`" 39 | cd "`dirname \"$PRG\"`/" >/dev/null 40 | APP_HOME="`pwd -P`" 41 | cd "$SAVED" >/dev/null 42 | 43 | APP_NAME="Gradle" 44 | APP_BASE_NAME=`basename "$0"` 45 | 46 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 47 | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' 48 | 49 | # Use the maximum available, or set MAX_FD != -1 to use that value. 50 | MAX_FD="maximum" 51 | 52 | warn () { 53 | echo "$*" 54 | } 55 | 56 | die () { 57 | echo 58 | echo "$*" 59 | echo 60 | exit 1 61 | } 62 | 63 | # OS specific support (must be 'true' or 'false'). 64 | cygwin=false 65 | msys=false 66 | darwin=false 67 | nonstop=false 68 | case "`uname`" in 69 | CYGWIN* ) 70 | cygwin=true 71 | ;; 72 | Darwin* ) 73 | darwin=true 74 | ;; 75 | MINGW* ) 76 | msys=true 77 | ;; 78 | NONSTOP* ) 79 | nonstop=true 80 | ;; 81 | esac 82 | 83 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 84 | 85 | 86 | # Determine the Java command to use to start the JVM. 87 | if [ -n "$JAVA_HOME" ] ; then 88 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 89 | # IBM's JDK on AIX uses strange locations for the executables 90 | JAVACMD="$JAVA_HOME/jre/sh/java" 91 | else 92 | JAVACMD="$JAVA_HOME/bin/java" 93 | fi 94 | if [ ! -x "$JAVACMD" ] ; then 95 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 96 | 97 | Please set the JAVA_HOME variable in your environment to match the 98 | location of your Java installation." 99 | fi 100 | else 101 | JAVACMD="java" 102 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 103 | 104 | Please set the JAVA_HOME variable in your environment to match the 105 | location of your Java installation." 106 | fi 107 | 108 | # Increase the maximum file descriptors if we can. 109 | if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then 110 | MAX_FD_LIMIT=`ulimit -H -n` 111 | if [ $? -eq 0 ] ; then 112 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 113 | MAX_FD="$MAX_FD_LIMIT" 114 | fi 115 | ulimit -n $MAX_FD 116 | if [ $? -ne 0 ] ; then 117 | warn "Could not set maximum file descriptor limit: $MAX_FD" 118 | fi 119 | else 120 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 121 | fi 122 | fi 123 | 124 | # For Darwin, add options to specify how the application appears in the dock 125 | if $darwin; then 126 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 127 | fi 128 | 129 | # For Cygwin or MSYS, switch paths to Windows format before running java 130 | if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then 131 | APP_HOME=`cygpath --path --mixed "$APP_HOME"` 132 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 133 | 134 | JAVACMD=`cygpath --unix "$JAVACMD"` 135 | 136 | # We build the pattern for arguments to be converted via cygpath 137 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` 138 | SEP="" 139 | for dir in $ROOTDIRSRAW ; do 140 | ROOTDIRS="$ROOTDIRS$SEP$dir" 141 | SEP="|" 142 | done 143 | OURCYGPATTERN="(^($ROOTDIRS))" 144 | # Add a user-defined pattern to the cygpath arguments 145 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then 146 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" 147 | fi 148 | # Now convert the arguments - kludge to limit ourselves to /bin/sh 149 | i=0 150 | for arg in "$@" ; do 151 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` 152 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option 153 | 154 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition 155 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` 156 | else 157 | eval `echo args$i`="\"$arg\"" 158 | fi 159 | i=`expr $i + 1` 160 | done 161 | case $i in 162 | 0) set -- ;; 163 | 1) set -- "$args0" ;; 164 | 2) set -- "$args0" "$args1" ;; 165 | 3) set -- "$args0" "$args1" "$args2" ;; 166 | 4) set -- "$args0" "$args1" "$args2" "$args3" ;; 167 | 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; 168 | 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; 169 | 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; 170 | 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; 171 | 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; 172 | esac 173 | fi 174 | 175 | # Escape application args 176 | save () { 177 | for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done 178 | echo " " 179 | } 180 | APP_ARGS=`save "$@"` 181 | 182 | # Collect all arguments for the java command, following the shell quoting and substitution rules 183 | eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" 184 | 185 | exec "$JAVACMD" "$@" 186 | -------------------------------------------------------------------------------- /gradlew.bat: -------------------------------------------------------------------------------- 1 | @rem 2 | @rem Copyright 2015 the original author or authors. 3 | @rem 4 | @rem Licensed under the Apache License, Version 2.0 (the "License"); 5 | @rem you may not use this file except in compliance with the License. 6 | @rem You may obtain a copy of the License at 7 | @rem 8 | @rem https://www.apache.org/licenses/LICENSE-2.0 9 | @rem 10 | @rem Unless required by applicable law or agreed to in writing, software 11 | @rem distributed under the License is distributed on an "AS IS" BASIS, 12 | @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 | @rem See the License for the specific language governing permissions and 14 | @rem limitations under the License. 15 | @rem 16 | 17 | @if "%DEBUG%" == "" @echo off 18 | @rem ########################################################################## 19 | @rem 20 | @rem Gradle startup script for Windows 21 | @rem 22 | @rem ########################################################################## 23 | 24 | @rem Set local scope for the variables with windows NT shell 25 | if "%OS%"=="Windows_NT" setlocal 26 | 27 | set DIRNAME=%~dp0 28 | if "%DIRNAME%" == "" set DIRNAME=. 29 | set APP_BASE_NAME=%~n0 30 | set APP_HOME=%DIRNAME% 31 | 32 | @rem Resolve any "." and ".." in APP_HOME to make it shorter. 33 | for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi 34 | 35 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 36 | set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" 37 | 38 | @rem Find java.exe 39 | if defined JAVA_HOME goto findJavaFromJavaHome 40 | 41 | set JAVA_EXE=java.exe 42 | %JAVA_EXE% -version >NUL 2>&1 43 | if "%ERRORLEVEL%" == "0" goto execute 44 | 45 | echo. 46 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 47 | echo. 48 | echo Please set the JAVA_HOME variable in your environment to match the 49 | echo location of your Java installation. 50 | 51 | goto fail 52 | 53 | :findJavaFromJavaHome 54 | set JAVA_HOME=%JAVA_HOME:"=% 55 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 56 | 57 | if exist "%JAVA_EXE%" goto execute 58 | 59 | echo. 60 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 61 | echo. 62 | echo Please set the JAVA_HOME variable in your environment to match the 63 | echo location of your Java installation. 64 | 65 | goto fail 66 | 67 | :execute 68 | @rem Setup the command line 69 | 70 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 71 | 72 | 73 | @rem Execute Gradle 74 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* 75 | 76 | :end 77 | @rem End local scope for the variables with windows NT shell 78 | if "%ERRORLEVEL%"=="0" goto mainEnd 79 | 80 | :fail 81 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 82 | rem the _cmd.exe /c_ return code! 83 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 84 | exit /b 1 85 | 86 | :mainEnd 87 | if "%OS%"=="Windows_NT" endlocal 88 | 89 | :omega 90 | -------------------------------------------------------------------------------- /mvi/.gitignore: -------------------------------------------------------------------------------- 1 | /build -------------------------------------------------------------------------------- /mvi/build.gradle.kts: -------------------------------------------------------------------------------- 1 | import Dependencies.applyAndroidX 2 | import Dependencies.applyGlide 3 | import Dependencies.applyHilt 4 | import Dependencies.applyRetrofit2 5 | import Dependencies.applyTest 6 | 7 | plugins { 8 | id ("com.android.application") 9 | id ("kotlin-android") 10 | id ("kotlin-kapt") 11 | id ("dagger.hilt.android.plugin") 12 | } 13 | 14 | android { 15 | compileSdk = Dependencies.COMPILE_SDK 16 | buildFeatures { 17 | dataBinding = true 18 | } 19 | 20 | defaultConfig { 21 | applicationId = "com.charlezz.mvi" 22 | minSdk = Dependencies.MIN_SDK 23 | targetSdk = Dependencies.TARGET_SDK 24 | versionCode = 1 25 | versionName = "1.0" 26 | 27 | testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" 28 | } 29 | 30 | buildTypes { 31 | getByName("release") { 32 | isMinifyEnabled = false 33 | proguardFiles(getDefaultProguardFile("proguard-android-optimize.txt"), "proguard-rules.pro") 34 | } 35 | } 36 | compileOptions { 37 | sourceCompatibility = JavaVersion.VERSION_11 38 | targetCompatibility = JavaVersion.VERSION_11 39 | } 40 | } 41 | 42 | dependencies { 43 | 44 | implementation (project(":domain")) 45 | implementation (project(":data")) 46 | 47 | implementation(Dependencies.Google.MATERIAL) 48 | applyAndroidX() 49 | applyTest() 50 | applyRetrofit2() 51 | applyHilt() 52 | applyGlide() 53 | } 54 | kapt { 55 | correctErrorTypes = true 56 | } 57 | hilt { 58 | enableTransformForLocalTests = true 59 | } 60 | -------------------------------------------------------------------------------- /mvi/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # You can control the set of applied configuration files using the 3 | # proguardFiles setting in build.gradle. 4 | # 5 | # For more details, see 6 | # http://developer.android.com/guide/developing/tools/proguard.html 7 | 8 | # If your project uses WebView with JS, uncomment the following 9 | # and specify the fully qualified class name to the JavaScript interface 10 | # class: 11 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 12 | # public *; 13 | #} 14 | 15 | # Uncomment this to preserve the line number information for 16 | # debugging stack traces. 17 | #-keepattributes SourceFile,LineNumberTable 18 | 19 | # If you keep the line number information, uncomment this to 20 | # hide the original source file name. 21 | #-renamesourcefileattribute SourceFile -------------------------------------------------------------------------------- /mvi/src/androidTest/java/com/charlezz/mvi/ExampleInstrumentedTest.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi 2 | 3 | import androidx.test.platform.app.InstrumentationRegistry 4 | import androidx.test.ext.junit.runners.AndroidJUnit4 5 | 6 | import org.junit.Test 7 | import org.junit.runner.RunWith 8 | 9 | import org.junit.Assert.* 10 | 11 | /** 12 | * Instrumented test, which will execute on an Android device. 13 | * 14 | * See [testing documentation](http://d.android.com/tools/testing). 15 | */ 16 | @RunWith(AndroidJUnit4::class) 17 | class ExampleInstrumentedTest { 18 | @Test 19 | fun useAppContext() { 20 | // Context of the app under test. 21 | val appContext = InstrumentationRegistry.getInstrumentation().targetContext 22 | assertEquals("com.charlezz.architecture", appContext.packageName) 23 | } 24 | } -------------------------------------------------------------------------------- /mvi/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 6 | 7 | 15 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/App.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi 2 | 3 | import android.app.Application 4 | import dagger.hilt.android.HiltAndroidApp 5 | 6 | @HiltAndroidApp 7 | class App : Application() 8 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/di/AppModules.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.di 2 | 3 | import com.charlezz.data.flickr.FlickrPhotosRepository 4 | import com.charlezz.data.flickr.FlickrService 5 | import com.charlezz.domain.repository.PhotosRepository 6 | import com.charlezz.domain.usecase.SearchUseCase 7 | import dagger.Module 8 | import dagger.Provides 9 | import dagger.hilt.InstallIn 10 | import dagger.hilt.components.SingletonComponent 11 | import retrofit2.Retrofit 12 | 13 | @InstallIn(SingletonComponent::class) 14 | @Module 15 | class AppModule { 16 | 17 | @Provides 18 | fun provideFlickrService(retrofit:Retrofit): FlickrService{ 19 | return retrofit.create(FlickrService::class.java) 20 | } 21 | 22 | @Provides 23 | fun providePhotosRepository(flickrService: FlickrService):PhotosRepository{ 24 | return FlickrPhotosRepository(flickrService) as PhotosRepository 25 | } 26 | 27 | @Provides 28 | fun provideSearchUseCase(photosRepository: PhotosRepository):SearchUseCase{ 29 | return SearchUseCase(photosRepository) 30 | } 31 | 32 | } 33 | 34 | 35 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/BindableViewHolder.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui 2 | 3 | import androidx.databinding.ViewDataBinding 4 | import androidx.recyclerview.widget.RecyclerView 5 | 6 | class BindableViewHolder(val binding: T) : 7 | RecyclerView.ViewHolder(binding.root) { 8 | } 9 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/PhotoAction.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui 2 | 3 | import com.charlezz.mvi.ui.base.Action 4 | 5 | sealed class PhotoAction: Action { 6 | 7 | object Initialize: PhotoAction() 8 | 9 | data class SearchKeyword( 10 | val keyword: String 11 | ): PhotoAction() 12 | 13 | } 14 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/PhotoActivity.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui 2 | 3 | import android.content.res.Configuration 4 | import android.os.Bundle 5 | import androidx.databinding.DataBindingUtil 6 | import androidx.recyclerview.widget.GridLayoutManager 7 | import dagger.hilt.android.AndroidEntryPoint 8 | import android.util.TypedValue 9 | import androidx.activity.viewModels 10 | import androidx.core.view.isVisible 11 | import androidx.lifecycle.lifecycleScope 12 | import androidx.paging.LoadState 13 | import com.charlezz.mvi.R 14 | import com.charlezz.mvi.databinding.ActivityPhotoBinding 15 | import com.charlezz.mvi.ui.base.BaseActivity 16 | import kotlinx.coroutines.launch 17 | import javax.inject.Inject 18 | 19 | @AndroidEntryPoint 20 | class PhotoActivity : BaseActivity() { 21 | 22 | @Inject 23 | lateinit var assistedFactory: PhotoViewModel.PhotoViewModelFactory 24 | 25 | override val viewModel: PhotoViewModel by viewModels { 26 | PhotoViewModel.Factory(assistedFactory) 27 | } 28 | 29 | private val adapter = PhotoAdapter() 30 | 31 | private val layoutManager: GridLayoutManager by lazy { 32 | GridLayoutManager(this, calculateSpanCount()) 33 | } 34 | 35 | private val binding: ActivityPhotoBinding by lazy { 36 | DataBindingUtil.setContentView(this, R.layout.activity_photo) 37 | } 38 | 39 | override fun onCreate(savedInstanceState: Bundle?) { 40 | super.onCreate(savedInstanceState) 41 | initUI() 42 | initData() 43 | } 44 | 45 | private fun initUI() { 46 | binding.lifecycleOwner = this 47 | binding.recyclerView.adapter = adapter 48 | binding.recyclerView.layoutManager = layoutManager 49 | 50 | binding.search.setOnClickListener { 51 | viewModel.setAction( 52 | PhotoAction.SearchKeyword( 53 | binding.editText.text.toString() 54 | ) 55 | ) 56 | } 57 | 58 | adapter.addLoadStateListener { loadStates -> 59 | binding.progressBar.isVisible = loadStates.refresh is LoadState.Loading 60 | } 61 | } 62 | 63 | private fun initData() { 64 | viewModel.setAction(PhotoAction.Initialize) 65 | } 66 | 67 | override fun onConfigurationChanged(newConfig: Configuration) { 68 | super.onConfigurationChanged(newConfig) 69 | layoutManager.spanCount = calculateSpanCount() 70 | } 71 | 72 | private fun calculateSpanCount(): Int { 73 | return resources.displayMetrics.widthPixels / TypedValue.applyDimension( 74 | TypedValue.COMPLEX_UNIT_DIP, 75 | 105.0f, 76 | resources.displayMetrics 77 | ).toInt() 78 | } 79 | 80 | override fun render(state: PhotoState) = when(state) { 81 | is PhotoState.List -> handleListState(state) 82 | is PhotoState.Unitialized -> Unit 83 | } 84 | 85 | private fun handleListState(state: PhotoState.List) { 86 | lifecycleScope.launch { 87 | adapter.submitData(state.photoData) 88 | } 89 | } 90 | 91 | } 92 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/PhotoAdapter.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui 2 | 3 | import android.view.LayoutInflater 4 | import android.view.ViewGroup 5 | import androidx.databinding.DataBindingUtil 6 | import androidx.paging.PagingDataAdapter 7 | import androidx.recyclerview.widget.DiffUtil 8 | import com.charlezz.mvi.R 9 | import com.charlezz.mvi.BR 10 | 11 | class PhotoAdapter : PagingDataAdapter>(diffCallback) { 12 | 13 | companion object { 14 | private val diffCallback = object : DiffUtil.ItemCallback() { 15 | override fun areItemsTheSame( 16 | oldItem: PhotoUiModel, 17 | newItem: PhotoUiModel 18 | ): Boolean { 19 | return oldItem.getImageUrl() == newItem.getImageUrl() 20 | } 21 | 22 | override fun areContentsTheSame( 23 | oldItem: PhotoUiModel, 24 | newItem: PhotoUiModel 25 | ): Boolean { 26 | return true 27 | } 28 | } 29 | } 30 | 31 | override fun getItemViewType(position: Int): Int { 32 | return getItem(position)?.getLayout() ?: R.layout.view_photo 33 | } 34 | 35 | override fun onBindViewHolder(holder: BindableViewHolder<*>, position: Int) { 36 | getItem(position)?.let { 37 | holder.binding.setVariable(BR.model, it) 38 | holder.binding.executePendingBindings() 39 | } 40 | 41 | } 42 | 43 | override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): BindableViewHolder<*> { 44 | return BindableViewHolder( 45 | DataBindingUtil.inflate( 46 | LayoutInflater.from(parent.context), 47 | viewType, 48 | parent, 49 | false 50 | ) 51 | ) 52 | } 53 | 54 | 55 | } 56 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/PhotoState.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui 2 | 3 | import androidx.paging.PagingData 4 | import com.charlezz.mvi.ui.base.State 5 | 6 | sealed class PhotoState : State { 7 | 8 | object Unitialized : PhotoState() 9 | 10 | data class List( 11 | val photoData: PagingData 12 | ) : PhotoState() 13 | 14 | } 15 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/PhotoUiModel.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui 2 | 3 | import com.charlezz.domain.Photo 4 | import com.charlezz.mvi.R 5 | 6 | class PhotoUiModel(private val photo: Photo) { 7 | 8 | fun getLayout(): Int = R.layout.view_photo 9 | 10 | fun getImageUrl(): String { 11 | return photo.url 12 | } 13 | 14 | } 15 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/PhotoViewModel.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui 2 | 3 | import androidx.lifecycle.ViewModel 4 | import androidx.lifecycle.ViewModelProvider 5 | import androidx.lifecycle.viewModelScope 6 | import androidx.paging.cachedIn 7 | import androidx.paging.map 8 | import com.charlezz.domain.usecase.SearchUseCase 9 | import com.charlezz.mvi.ui.base.BaseViewModel 10 | import dagger.assisted.Assisted 11 | import dagger.assisted.AssistedFactory 12 | import dagger.assisted.AssistedInject 13 | import kotlinx.coroutines.flow.* 14 | import kotlinx.coroutines.launch 15 | 16 | class PhotoViewModel @AssistedInject constructor( 17 | @Assisted state: PhotoState, 18 | private val searchUseCase: SearchUseCase, 19 | ) : BaseViewModel(state) { 20 | 21 | override fun handleAction(action: PhotoAction) { 22 | when (action) { 23 | is PhotoAction.Initialize -> handleInitialize() 24 | is PhotoAction.SearchKeyword -> handleSearchKeyword(action) 25 | } 26 | } 27 | 28 | private fun handleInitialize() = viewModelScope.launch { 29 | searchUseCase("") 30 | .map { pagingData -> pagingData.map { PhotoUiModel(it) } } 31 | .cachedIn(viewModelScope) 32 | .collectLatest { photoData -> 33 | setState { 34 | PhotoState.List(photoData = photoData) 35 | } 36 | } 37 | } 38 | 39 | private fun handleSearchKeyword(action: PhotoAction.SearchKeyword) = viewModelScope.launch { 40 | searchUseCase(action.keyword) 41 | .map { pagingData -> pagingData.map { PhotoUiModel(it) } } 42 | .cachedIn(viewModelScope) 43 | .collectLatest { photoData -> 44 | setState { 45 | PhotoState.List(photoData = photoData) 46 | } 47 | } 48 | } 49 | 50 | @AssistedFactory 51 | interface PhotoViewModelFactory { 52 | 53 | fun create(state: PhotoState): PhotoViewModel 54 | 55 | } 56 | 57 | class Factory( 58 | private val assistedFactory: PhotoViewModelFactory, 59 | private val state: PhotoState = PhotoState.Unitialized 60 | ) : ViewModelProvider.Factory { 61 | override fun create(modelClass: Class): T { 62 | return if (modelClass.isAssignableFrom(PhotoViewModel::class.java)) { 63 | assistedFactory.create(state) as T 64 | } else { 65 | throw IllegalArgumentException() 66 | } 67 | } 68 | 69 | } 70 | 71 | } 72 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/base/Action.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui.base 2 | 3 | interface Action 4 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/base/BaseActivity.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui.base 2 | 3 | import android.os.Bundle 4 | import androidx.appcompat.app.AppCompatActivity 5 | import androidx.lifecycle.lifecycleScope 6 | import kotlinx.coroutines.flow.launchIn 7 | import kotlinx.coroutines.flow.onEach 8 | 9 | abstract class BaseActivity> : AppCompatActivity() { 10 | 11 | abstract val viewModel: VM 12 | 13 | override fun onCreate(savedInstanceState: Bundle?) { 14 | super.onCreate(savedInstanceState) 15 | observeState() 16 | } 17 | 18 | private fun observeState() = 19 | viewModel.uiStateFlow.onEach { state -> 20 | render(state) 21 | }.launchIn(lifecycleScope) 22 | 23 | abstract fun render(state: S) 24 | 25 | } 26 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/base/BaseViewModel.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui.base 2 | 3 | import androidx.lifecycle.ViewModel 4 | import androidx.lifecycle.viewModelScope 5 | import kotlinx.coroutines.flow.* 6 | import kotlinx.coroutines.launch 7 | 8 | abstract class BaseViewModel( 9 | state: S 10 | ) : ViewModel() { 11 | 12 | private val _uiStateFlow: MutableStateFlow = MutableStateFlow(state) 13 | 14 | private val currentUiState: S 15 | get() = _uiStateFlow.value 16 | 17 | val uiStateFlow: Flow = _uiStateFlow.asStateFlow() 18 | 19 | private val _actionFlow : MutableSharedFlow = MutableSharedFlow() 20 | val actionFlow = _actionFlow.asSharedFlow() 21 | 22 | init { subscribeAction() } 23 | 24 | fun setAction(action: A) = viewModelScope.launch { 25 | _actionFlow.emit(action) 26 | } 27 | 28 | private fun subscribeAction() = viewModelScope.launch { 29 | _actionFlow.collect { handleAction(it) } 30 | } 31 | 32 | abstract fun handleAction(action: A) 33 | 34 | protected fun setState(reduce: S.() -> S) = viewModelScope.launch { 35 | val newState = currentUiState.reduce() 36 | _uiStateFlow.value = newState 37 | } 38 | 39 | protected fun withState(state: (S) -> Unit) { 40 | return state(currentUiState) 41 | } 42 | 43 | } 44 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/ui/base/State.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.ui.base 2 | 3 | interface State 4 | -------------------------------------------------------------------------------- /mvi/src/main/java/com/charlezz/mvi/util/ImageViewBindingAdapter.kt: -------------------------------------------------------------------------------- 1 | package com.charlezz.mvi.util 2 | 3 | import android.widget.ImageView 4 | import androidx.databinding.BindingAdapter 5 | import com.bumptech.glide.Glide 6 | 7 | 8 | @BindingAdapter("imageUrl") 9 | fun setImageUrl(view: ImageView, url:String?){ 10 | Glide.with(view).load(url).into(view) 11 | } 12 | -------------------------------------------------------------------------------- /mvi/src/main/res/drawable-v24/ic_launcher_foreground.xml: -------------------------------------------------------------------------------- 1 | 7 | 8 | 9 | 15 | 18 | 21 | 22 | 23 | 24 | 30 | -------------------------------------------------------------------------------- /mvi/src/main/res/drawable/ic_launcher_background.xml: -------------------------------------------------------------------------------- 1 | 2 | 7 | 10 | 15 | 20 | 25 | 30 | 35 | 40 | 45 | 50 | 55 | 60 | 65 | 70 | 75 | 80 | 85 | 90 | 95 | 100 | 105 | 110 | 115 | 120 | 125 | 130 | 135 | 140 | 145 | 150 | 155 | 160 | 165 | 170 | 171 | -------------------------------------------------------------------------------- /mvi/src/main/res/layout/activity_photo.xml: -------------------------------------------------------------------------------- 1 | 2 | 5 | 6 | 7 | 8 | 11 | 12 | 21 | 22 | 27 | 28 | 29 | 30 |