├── .gitignore ├── README.md ├── app ├── .gitignore ├── build.gradle ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── com │ │ └── antonioleiva │ │ └── architectcoderslite │ │ └── ExampleInstrumentedTest.kt │ ├── main │ ├── AndroidManifest.xml │ ├── java │ │ └── com │ │ │ └── antonioleiva │ │ │ └── architectcoderslite │ │ │ ├── data │ │ │ ├── LoginRemoteDataSource.kt │ │ │ └── LoginRepository.kt │ │ │ ├── domain │ │ │ └── TryLoginUseCase.kt │ │ │ └── ui │ │ │ ├── MainActivity.kt │ │ │ ├── MainViewModel.kt │ │ │ └── NextActivity.kt │ └── res │ │ ├── drawable-v24 │ │ └── ic_launcher_foreground.xml │ │ ├── drawable │ │ └── ic_launcher_background.xml │ │ ├── layout │ │ ├── activity_main.xml │ │ └── activity_next.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 │ └── antonioleiva │ └── architectcoderslite │ └── data │ ├── LoginRemoteDataSourceImplTest.kt │ └── LoginRepositoryTest.kt ├── build.gradle ├── gradle.properties ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat └── settings.gradle /.gitignore: -------------------------------------------------------------------------------- 1 | # Generated files 2 | bin/ 3 | gen/ 4 | 5 | # Gradle files 6 | .gradle/ 7 | build/ 8 | 9 | # Local configuration file (sdk path, etc) 10 | local.properties 11 | 12 | # Intellij project files 13 | *.iws 14 | .idea/tasks.xml 15 | .idea 16 | *.iml 17 | 18 | # OS 19 | .DS_Store -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Architect Coders Lite 2 | 3 | Aprende arquitecturas Android de forma estructurada y con un método diseñado para asimilar los conocimientos de manera rápida y eficiente.Android 4 | 5 | Architect Coders Lite es una versión reducida y gratuita de mi formación Architect Coders. Podrás participar en alguna de las siguientes ediciones.Android 6 | 7 | Para no perdértela, apúntate a la lista de espera en: [architectcoders.com](https://architectcoders.com) 8 | 9 | ## License 10 | 11 | Copyright 2022 Antonio Leiva 12 | 13 | Licensed under the Apache License, Version 2.0 (the "License"); 14 | you may not use this file except in compliance with the License. 15 | You may obtain a copy of the License at 16 | 17 | http://www.apache.org/licenses/LICENSE-2.0 18 | 19 | Unless required by applicable law or agreed to in writing, software 20 | distributed under the License is distributed on an "AS IS" BASIS, 21 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 22 | See the License for the specific language governing permissions and 23 | limitations under the License. -------------------------------------------------------------------------------- /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | plugins { 2 | id 'com.android.application' 3 | id 'org.jetbrains.kotlin.android' 4 | } 5 | 6 | android { 7 | compileSdk 31 8 | 9 | defaultConfig { 10 | applicationId "com.antonioleiva.architectcoderslite" 11 | minSdk 23 12 | targetSdk 31 13 | versionCode 1 14 | versionName "1.0" 15 | 16 | testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" 17 | } 18 | 19 | buildTypes { 20 | release { 21 | minifyEnabled false 22 | proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' 23 | } 24 | } 25 | compileOptions { 26 | sourceCompatibility JavaVersion.VERSION_1_8 27 | targetCompatibility JavaVersion.VERSION_1_8 28 | } 29 | kotlinOptions { 30 | jvmTarget = '1.8' 31 | } 32 | buildFeatures { 33 | viewBinding true 34 | } 35 | } 36 | 37 | dependencies { 38 | 39 | implementation 'androidx.core:core-ktx:1.7.0' 40 | implementation 'androidx.appcompat:appcompat:1.4.0' 41 | implementation 'com.google.android.material:material:1.4.0' 42 | implementation 'androidx.constraintlayout:constraintlayout:2.1.2' 43 | implementation 'androidx.lifecycle:lifecycle-runtime-ktx:2.4.0' 44 | implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.4.0' 45 | implementation 'androidx.activity:activity-ktx:1.4.0' 46 | 47 | testImplementation 'junit:junit:4.13.2' 48 | androidTestImplementation 'androidx.test.ext:junit:1.1.3' 49 | androidTestImplementation 'androidx.test.espresso:espresso-core:3.4.0' 50 | } -------------------------------------------------------------------------------- /app/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 -------------------------------------------------------------------------------- /app/src/androidTest/java/com/antonioleiva/architectcoderslite/ExampleInstrumentedTest.kt: -------------------------------------------------------------------------------- 1 | package com.antonioleiva.architectcoderslite 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.antonioleiva.architectcoderslite", appContext.packageName) 23 | } 24 | } -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 12 | 15 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | -------------------------------------------------------------------------------- /app/src/main/java/com/antonioleiva/architectcoderslite/data/LoginRemoteDataSource.kt: -------------------------------------------------------------------------------- 1 | package com.antonioleiva.architectcoderslite.data 2 | 3 | import kotlinx.coroutines.delay 4 | 5 | interface LoginRemoteDataSource{ 6 | suspend fun tryLogin(user: String, pass: String): LoginResult 7 | } 8 | 9 | class LoginRemoteDataSourceImpl : LoginRemoteDataSource { 10 | 11 | override suspend fun tryLogin(user: String, pass: String): LoginResult { 12 | delay(2000) 13 | val userError = !user.contains('@') 14 | val passError = pass.length < 5 15 | 16 | return LoginResult(userError, passError) 17 | } 18 | } -------------------------------------------------------------------------------- /app/src/main/java/com/antonioleiva/architectcoderslite/data/LoginRepository.kt: -------------------------------------------------------------------------------- 1 | package com.antonioleiva.architectcoderslite.data 2 | 3 | class LoginRepository( 4 | private val remoteDataSource: LoginRemoteDataSource = LoginRemoteDataSourceImpl() 5 | ) { 6 | suspend fun tryLogin(user: String, pass: String): LoginResult = 7 | remoteDataSource.tryLogin(user, pass) 8 | } 9 | 10 | data class LoginResult(val userError: Boolean, val passError: Boolean) 11 | 12 | val LoginResult.success get() = !userError && !passError -------------------------------------------------------------------------------- /app/src/main/java/com/antonioleiva/architectcoderslite/domain/TryLoginUseCase.kt: -------------------------------------------------------------------------------- 1 | package com.antonioleiva.architectcoderslite.domain 2 | 3 | import com.antonioleiva.architectcoderslite.data.LoginRepository 4 | import com.antonioleiva.architectcoderslite.data.LoginResult 5 | 6 | class TryLoginUseCase( 7 | private val loginRepository: LoginRepository = LoginRepository() 8 | ) { 9 | 10 | suspend operator fun invoke(user: String, pass: String): LoginResult = 11 | loginRepository.tryLogin(user, pass) 12 | } -------------------------------------------------------------------------------- /app/src/main/java/com/antonioleiva/architectcoderslite/ui/MainActivity.kt: -------------------------------------------------------------------------------- 1 | package com.antonioleiva.architectcoderslite.ui 2 | 3 | import android.content.Intent 4 | import android.os.Bundle 5 | import android.view.View 6 | import androidx.activity.viewModels 7 | import androidx.appcompat.app.AppCompatActivity 8 | import com.antonioleiva.architectcoderslite.databinding.ActivityMainBinding 9 | 10 | class MainActivity : AppCompatActivity() { 11 | 12 | private lateinit var binding: ActivityMainBinding 13 | 14 | private val viewModel: MainViewModel by viewModels() 15 | 16 | override fun onCreate(savedInstanceState: Bundle?) { 17 | super.onCreate(savedInstanceState) 18 | binding = ActivityMainBinding.inflate(layoutInflater) 19 | setContentView(binding.root) 20 | 21 | binding.button.setOnClickListener { 22 | viewModel.onTryLogin(binding.user.text.toString(), binding.pass.text.toString()) 23 | } 24 | 25 | viewModel.uiState.observe(this) { state -> 26 | binding.user.error = state.userError?.let(::getString) 27 | binding.pass.error = state.passError?.let(::getString) 28 | binding.button.visibility = if (state.loggingIn) View.GONE else View.VISIBLE 29 | binding.progress.visibility = if (state.loggingIn) View.VISIBLE else View.GONE 30 | 31 | if (state.loggedIn) { 32 | startActivity(Intent(this, NextActivity::class.java)) 33 | finish() 34 | } 35 | } 36 | } 37 | } -------------------------------------------------------------------------------- /app/src/main/java/com/antonioleiva/architectcoderslite/ui/MainViewModel.kt: -------------------------------------------------------------------------------- 1 | package com.antonioleiva.architectcoderslite.ui 2 | 3 | import androidx.lifecycle.LiveData 4 | import androidx.lifecycle.MutableLiveData 5 | import androidx.lifecycle.ViewModel 6 | import androidx.lifecycle.viewModelScope 7 | import com.antonioleiva.architectcoderslite.R 8 | import com.antonioleiva.architectcoderslite.data.success 9 | import com.antonioleiva.architectcoderslite.domain.TryLoginUseCase 10 | import kotlinx.coroutines.launch 11 | 12 | class MainViewModel( 13 | private val tryLoginUseCase: TryLoginUseCase = TryLoginUseCase() 14 | ) : ViewModel() { 15 | 16 | private val _uiState = MutableLiveData(UiState()) 17 | val uiState: LiveData get() = _uiState 18 | 19 | fun onTryLogin(user: String, pass: String) { 20 | viewModelScope.launch { 21 | _uiState.value = UiState(loggingIn = true) 22 | val result = tryLoginUseCase(user, pass) 23 | _uiState.value = UiState( 24 | loggedIn = result.success, 25 | userError = if (result.userError) R.string.user_error else null, 26 | passError = if (result.passError) R.string.pass_error else null 27 | ) 28 | } 29 | } 30 | 31 | data class UiState( 32 | val loggingIn: Boolean = false, 33 | val loggedIn: Boolean = false, 34 | val userError: Int? = null, 35 | val passError: Int? = null 36 | ) 37 | } -------------------------------------------------------------------------------- /app/src/main/java/com/antonioleiva/architectcoderslite/ui/NextActivity.kt: -------------------------------------------------------------------------------- 1 | package com.antonioleiva.architectcoderslite.ui 2 | 3 | import androidx.appcompat.app.AppCompatActivity 4 | import android.os.Bundle 5 | import com.antonioleiva.architectcoderslite.R 6 | 7 | class NextActivity : AppCompatActivity() { 8 | override fun onCreate(savedInstanceState: Bundle?) { 9 | super.onCreate(savedInstanceState) 10 | setContentView(R.layout.activity_next) 11 | } 12 | } -------------------------------------------------------------------------------- /app/src/main/res/drawable-v24/ic_launcher_foreground.xml: -------------------------------------------------------------------------------- 1 | 7 | 8 | 9 | 15 | 18 | 21 | 22 | 23 | 24 | 30 | -------------------------------------------------------------------------------- /app/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 | -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 10 | 11 | 16 | 17 | 21 | 22 | 23 | 24 | 29 | 30 | 35 | 36 | 37 | 38 |