├── .gitignore ├── README.md ├── app ├── .gitignore ├── build.gradle ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── anton │ │ └── averin │ │ └── pro │ │ └── tests │ │ └── kotlin │ │ └── sample │ │ └── ApplicationTest.java │ ├── main │ ├── AndroidManifest.xml │ ├── java │ │ └── anton │ │ │ └── averin │ │ │ └── pro │ │ │ └── tests │ │ │ └── kotlin │ │ │ └── sample │ │ │ ├── BaseApplication.kt │ │ │ ├── SampleApplication.kt │ │ │ ├── data │ │ │ ├── Repository.kt │ │ │ └── RepositoryImpl.kt │ │ │ ├── di │ │ │ ├── ApplicationComponent.kt │ │ │ ├── ApplicationInjectsTo.kt │ │ │ └── ApplicationModule.kt │ │ │ ├── mvp │ │ │ ├── presenters │ │ │ │ ├── BasePresenter.kt │ │ │ │ ├── MainPresenter.kt │ │ │ │ └── PresentedActivity.kt │ │ │ ├── usecases │ │ │ │ ├── SampleUseCase.kt │ │ │ │ ├── SampleUseCaseImpl.kt │ │ │ │ └── UseCase.kt │ │ │ └── views │ │ │ │ ├── BaseView.kt │ │ │ │ └── MainView.kt │ │ │ └── ui │ │ │ ├── BaseActivity.kt │ │ │ └── MainActivity.kt │ └── res │ │ ├── layout │ │ ├── activity_main.xml │ │ └── content_main.xml │ │ ├── menu │ │ └── menu_main.xml │ │ ├── mipmap-hdpi │ │ └── ic_launcher.png │ │ ├── mipmap-mdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xhdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xxhdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xxxhdpi │ │ └── ic_launcher.png │ │ ├── values-v21 │ │ └── styles.xml │ │ ├── values-w820dp │ │ └── dimens.xml │ │ └── values │ │ ├── colors.xml │ │ ├── dimens.xml │ │ ├── strings.xml │ │ └── styles.xml │ └── test │ └── java │ └── anton │ └── averin │ └── pro │ └── tests │ └── kotlin │ └── sample │ ├── ExampleUnitTest.java │ ├── MainPresenterTest.kt │ └── SampleUseCaseTest.kt ├── build.gradle ├── gradle.properties ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat └── settings.gradle /.gitignore: -------------------------------------------------------------------------------- 1 | *.iml 2 | .gradle 3 | /local.properties 4 | /.idea/workspace.xml 5 | /.idea/libraries 6 | .DS_Store 7 | /build 8 | /captures 9 | .idea 10 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # android_clean_kotlin 2 | A Clean architecture for Android on Kotlin: MVP, UseCase, Repository 3 | -------------------------------------------------------------------------------- /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.application' 2 | apply plugin: 'kotlin-android' 3 | 4 | android { 5 | compileSdkVersion 23 6 | buildToolsVersion "23.0.1" 7 | 8 | defaultConfig { 9 | applicationId "anton.averin.pro.tests.kotlin.sample" 10 | minSdkVersion 14 11 | targetSdkVersion 23 12 | versionCode 1 13 | versionName "1.0" 14 | } 15 | buildTypes { 16 | release { 17 | minifyEnabled false 18 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 19 | } 20 | } 21 | sourceSets { 22 | main.java.srcDirs += 'src/main/kotlin' 23 | } 24 | } 25 | 26 | dependencies { 27 | compile fileTree(dir: 'libs', include: ['*.jar']) 28 | testCompile 'junit:junit:4.12' 29 | compile 'com.android.support:appcompat-v7:23.1.1' 30 | compile 'com.android.support:design:23.1.1' 31 | compile "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version" 32 | compile 'com.jakewharton:kotterknife:0.1.0-SNAPSHOT' 33 | 34 | compile 'com.google.dagger:dagger:2.0' 35 | kapt 'com.google.dagger:dagger-compiler:2.0' 36 | provided 'org.glassfish:javax.annotation:10.0-b28' 37 | 38 | testCompile 'junit:junit:4.12' 39 | testCompile "org.mockito:mockito-core:1.10.19" 40 | testCompile "org.hamcrest:hamcrest-all:1.3" 41 | 42 | compile 'io.reactivex:rxkotlin:0.30.1' 43 | } 44 | 45 | kapt { 46 | generateStubs = true 47 | } 48 | 49 | buildscript { 50 | ext.kotlin_version = '1.0.0-beta-3595' 51 | repositories { 52 | mavenCentral() 53 | } 54 | dependencies { 55 | classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" 56 | classpath "org.jetbrains.kotlin:kotlin-android-extensions:$kotlin_version" 57 | } 58 | } 59 | repositories { 60 | mavenCentral() 61 | maven { url "https://oss.sonatype.org/content/repositories/snapshots" } 62 | } 63 | -------------------------------------------------------------------------------- /app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # By default, the flags in this file are appended to flags specified 3 | # in /Users/antonaverin/Library/Android/sdk/tools/proguard/proguard-android.txt 4 | # You can edit the include path and order by changing the proguardFiles 5 | # directive in build.gradle. 6 | # 7 | # For more details, see 8 | # http://developer.android.com/guide/developing/tools/proguard.html 9 | 10 | # Add any project specific keep options here: 11 | 12 | # If your project uses WebView with JS, uncomment the following 13 | # and specify the fully qualified class name to the JavaScript interface 14 | # class: 15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 16 | # public *; 17 | #} 18 | -------------------------------------------------------------------------------- /app/src/androidTest/java/anton/averin/pro/tests/kotlin/sample/ApplicationTest.java: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample; 2 | 3 | import android.app.Application; 4 | import android.test.ApplicationTestCase; 5 | 6 | /** 7 | * Testing Fundamentals 8 | */ 9 | public class ApplicationTest extends ApplicationTestCase { 10 | public ApplicationTest() { 11 | super(Application.class); 12 | } 13 | } -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 12 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/BaseApplication.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample 2 | 3 | import android.app.Application 4 | import anton.averin.pro.tests.kotlin.sample.di.ApplicationComponent 5 | import anton.averin.pro.tests.kotlin.sample.di.ApplicationModule 6 | import anton.averin.pro.tests.kotlin.sample.di.DaggerApplicationComponent 7 | 8 | public abstract class BaseApplication : Application() { 9 | 10 | protected fun initDaggerComponent(): ApplicationComponent { 11 | return DaggerApplicationComponent.builder().applicationModule(ApplicationModule(this)).build(); 12 | } 13 | } 14 | 15 | 16 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/SampleApplication.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample 2 | 3 | import anton.averin.pro.tests.kotlin.sample.di.ApplicationComponent 4 | 5 | public class SampleApplication : BaseApplication() { 6 | 7 | lateinit var graph: ApplicationComponent 8 | 9 | override fun onCreate() { 10 | super.onCreate() 11 | val graph = initDaggerComponent(); 12 | graph.inject(this); 13 | this.graph = graph; 14 | } 15 | 16 | } 17 | 18 | 19 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/data/Repository.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.data 2 | 3 | import rx.Observable 4 | 5 | interface Repository { 6 | fun doSampleRequest(): Observable 7 | } -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/data/RepositoryImpl.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.data 2 | 3 | import rx.Observable 4 | import javax.inject.Inject 5 | import javax.inject.Singleton 6 | 7 | @Singleton 8 | class RepositoryImpl @Inject constructor(): Repository { 9 | override fun doSampleRequest() : Observable { 10 | return Observable.just("Hello Reactive World!") 11 | } 12 | 13 | } -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/di/ApplicationComponent.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.di 2 | 3 | import dagger.Component 4 | import javax.inject.Singleton 5 | 6 | @Singleton 7 | @Component(modules = arrayOf(ApplicationModule::class)) 8 | public interface ApplicationComponent : ApplicationInjectsTo { 9 | 10 | } 11 | 12 | 13 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/di/ApplicationInjectsTo.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.di 2 | 3 | import anton.averin.pro.tests.kotlin.sample.SampleApplication 4 | import anton.averin.pro.tests.kotlin.sample.ui.MainActivity 5 | 6 | public interface ApplicationInjectsTo { 7 | fun inject(sampleApplication: SampleApplication) 8 | 9 | fun inject(mainActivity: MainActivity) 10 | } -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/di/ApplicationModule.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.di 2 | 3 | import anton.averin.pro.tests.kotlin.sample.BaseApplication 4 | import anton.averin.pro.tests.kotlin.sample.data.Repository 5 | import anton.averin.pro.tests.kotlin.sample.data.RepositoryImpl 6 | import anton.averin.pro.tests.kotlin.sample.mvp.usecases.SampleUseCase 7 | import anton.averin.pro.tests.kotlin.sample.mvp.usecases.SampleUseCaseImpl 8 | import dagger.Module 9 | import dagger.Provides 10 | import javax.inject.Singleton 11 | 12 | @Module 13 | public class ApplicationModule(private val application: BaseApplication) { 14 | 15 | @Singleton 16 | @Provides 17 | fun repository(repository: RepositoryImpl) : Repository { 18 | return repository 19 | } 20 | 21 | @Singleton 22 | @Provides 23 | fun sampleUsecase(useCase: SampleUseCaseImpl) : SampleUseCase { 24 | return useCase 25 | } 26 | 27 | } 28 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/mvp/presenters/BasePresenter.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.mvp.presenters 2 | 3 | import anton.averin.pro.tests.kotlin.sample.mvp.views.BaseView 4 | 5 | interface BasePresenter { 6 | val view: BaseView? 7 | 8 | open fun onCreate() { 9 | //override for implementation 10 | } 11 | 12 | open fun onPause() { 13 | //override for implementation 14 | } 15 | 16 | open fun onResume() { 17 | //override for implementation 18 | } 19 | 20 | open fun onDestroy() { 21 | //override for implementation 22 | } 23 | } 24 | 25 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/mvp/presenters/MainPresenter.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.mvp.presenters 2 | 3 | import android.util.Log 4 | import anton.averin.pro.tests.kotlin.sample.mvp.usecases.SampleUseCase 5 | import anton.averin.pro.tests.kotlin.sample.mvp.views.MainView 6 | import javax.inject.Inject 7 | import javax.inject.Singleton 8 | 9 | @Singleton 10 | class MainPresenter @Inject constructor(): BasePresenter { 11 | 12 | @Inject 13 | lateinit var useCase: SampleUseCase 14 | 15 | override var view: MainView? = null 16 | 17 | fun fabClicked() { 18 | useCase.getDataForCase().subscribe { 19 | Log.d("TEST", it) 20 | } 21 | } 22 | 23 | } 24 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/mvp/presenters/PresentedActivity.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.mvp.presenters 2 | 3 | import anton.averin.pro.tests.kotlin.sample.ui.BaseActivity 4 | 5 | open class PresentedActivity : BaseActivity() 6 | where T : BasePresenter { 7 | 8 | private var presenter: T? = null 9 | 10 | public fun initPresenter(presenter: T) { 11 | this.presenter = presenter 12 | presenter.onCreate() 13 | } 14 | 15 | override fun onResume() { 16 | super.onResume() 17 | presenter?.onResume() 18 | } 19 | 20 | override fun onPause() { 21 | super.onPause() 22 | presenter?.onPause() 23 | } 24 | 25 | override fun onDestroy() { 26 | super.onDestroy() 27 | presenter?.onDestroy() 28 | } 29 | 30 | } 31 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/mvp/usecases/SampleUseCase.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.mvp.usecases 2 | 3 | import rx.Observable 4 | 5 | interface SampleUseCase { 6 | fun getDataForCase(): Observable 7 | } -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/mvp/usecases/SampleUseCaseImpl.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.mvp.usecases 2 | 3 | import anton.averin.pro.tests.kotlin.sample.data.Repository 4 | import rx.Observable 5 | import javax.inject.Inject 6 | import javax.inject.Singleton 7 | 8 | @Singleton 9 | class SampleUseCaseImpl @Inject constructor(): UseCase(), SampleUseCase { 10 | 11 | @Inject 12 | lateinit var repository: Repository 13 | 14 | override fun getDataForCase() : Observable { 15 | return repository.doSampleRequest().map { 16 | it.plus(" !mapped!") 17 | } 18 | } 19 | 20 | } 21 | 22 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/mvp/usecases/UseCase.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.mvp.usecases 2 | 3 | abstract class UseCase { 4 | 5 | } 6 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/mvp/views/BaseView.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.mvp.views 2 | 3 | interface BaseView { 4 | } -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/mvp/views/MainView.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.mvp.views 2 | 3 | public interface MainView : BaseView { 4 | fun showMessage() 5 | 6 | } 7 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/ui/BaseActivity.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.ui 2 | 3 | import android.support.v7.app.AppCompatActivity 4 | 5 | open class BaseActivity : AppCompatActivity() { 6 | 7 | } 8 | 9 | 10 | -------------------------------------------------------------------------------- /app/src/main/java/anton/averin/pro/tests/kotlin/sample/ui/MainActivity.kt: -------------------------------------------------------------------------------- 1 | package anton.averin.pro.tests.kotlin.sample.ui 2 | 3 | import android.os.Bundle 4 | import android.support.design.widget.FloatingActionButton 5 | import android.support.design.widget.Snackbar 6 | import android.support.v7.widget.Toolbar 7 | import android.view.Menu 8 | import android.view.MenuItem 9 | import anton.averin.pro.tests.kotlin.sample.R 10 | import anton.averin.pro.tests.kotlin.sample.SampleApplication 11 | import anton.averin.pro.tests.kotlin.sample.mvp.presenters.MainPresenter 12 | import anton.averin.pro.tests.kotlin.sample.mvp.presenters.PresentedActivity 13 | import anton.averin.pro.tests.kotlin.sample.mvp.views.MainView 14 | import butterknife.bindView 15 | import javax.inject.Inject 16 | 17 | class MainActivity : PresentedActivity(), MainView { 18 | 19 | 20 | @Inject 21 | lateinit var presenter: MainPresenter 22 | 23 | val toolbar: Toolbar by bindView(R.id.toolbar) 24 | val fab: FloatingActionButton by bindView(R.id.fab) 25 | 26 | override fun onCreate(savedInstanceState: Bundle?) { 27 | super.onCreate(savedInstanceState) 28 | setContentView(R.layout.activity_main) 29 | 30 | (application as SampleApplication).graph.inject(this) 31 | presenter.view = this 32 | initPresenter(presenter) 33 | 34 | setSupportActionBar(toolbar) 35 | 36 | fab.setOnClickListener { view -> 37 | presenter.fabClicked(); 38 | } 39 | } 40 | 41 | override fun showMessage() { 42 | Snackbar.make(fab, "Replace with your own action", Snackbar.LENGTH_LONG).setAction("Action", null).show() 43 | } 44 | 45 | override fun onCreateOptionsMenu(menu: Menu): Boolean { 46 | // Inflate the menu; this adds items to the action bar if it is present. 47 | menuInflater.inflate(R.menu.menu_main, menu) 48 | return true 49 | } 50 | 51 | override fun onOptionsItemSelected(item: MenuItem): Boolean { 52 | // Handle action bar item clicks here. The action bar will 53 | // automatically handle clicks on the Home/Up button, so long 54 | // as you specify a parent activity in AndroidManifest.xml. 55 | val id = item.itemId 56 | 57 | //noinspection SimplifiableIfStatement 58 | if (id == R.id.action_settings) { 59 | return true 60 | } 61 | 62 | return super.onOptionsItemSelected(item) 63 | } 64 | } 65 | -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 9 | 10 | 14 | 15 | 21 | 22 | 23 | 24 | 25 | 26 | 33 | 34 | 35 | -------------------------------------------------------------------------------- /app/src/main/res/layout/content_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 14 | 15 | 19 | 20 | -------------------------------------------------------------------------------- /app/src/main/res/menu/menu_main.xml: -------------------------------------------------------------------------------- 1 | 5 | 10 | 11 | -------------------------------------------------------------------------------- /app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AAverin/android_clean_kotlin/f2e62c160bb6891ccdf8b4ca8bce6e8894ba0632/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AAverin/android_clean_kotlin/f2e62c160bb6891ccdf8b4ca8bce6e8894ba0632/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AAverin/android_clean_kotlin/f2e62c160bb6891ccdf8b4ca8bce6e8894ba0632/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AAverin/android_clean_kotlin/f2e62c160bb6891ccdf8b4ca8bce6e8894ba0632/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AAverin/android_clean_kotlin/f2e62c160bb6891ccdf8b4ca8bce6e8894ba0632/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/values-v21/styles.xml: -------------------------------------------------------------------------------- 1 | > 2 | 3 | 9 | 10 | -------------------------------------------------------------------------------- /app/src/main/res/values-w820dp/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 5 | 64dp 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/colors.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | #3F51B5 4 | #303F9F 5 | #FF4081 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 16dp 4 | 16dp 5 | 16dp 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | Mediteo Kotlin Sample 3 | Settings 4 | 5 | -------------------------------------------------------------------------------- /app/src/main/res/values/styles.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 10 | 11 | 15 | 16 |