├── app ├── .gitignore ├── src │ ├── test │ │ ├── resources │ │ │ └── mockito-extensions │ │ │ │ └── org.mockito.plugins.MockMaker │ │ └── java │ │ │ └── com │ │ │ └── jshvarts │ │ │ └── shoppinglist │ │ │ ├── rx │ │ │ └── SchedulersFacadeUtils.java │ │ │ ├── common │ │ │ └── domain │ │ │ │ └── model │ │ │ │ ├── ItemByIdSpecificationTest.java │ │ │ │ ├── ItemsSpecificationTest.java │ │ │ │ └── ShoppingListDataHelperTest.java │ │ │ └── lobby │ │ │ └── fragments │ │ │ ├── CreateShoppingListUseCaseTest.java │ │ │ ├── UpdateShoppingListUseCaseTest.java │ │ │ ├── AddShoppingListItemUseCaseTest.java │ │ │ ├── LoadShoppingListUseCaseTest.java │ │ │ └── ShoppingListViewModelTest.java │ └── main │ │ ├── res │ │ ├── mipmap-hdpi │ │ │ ├── ic_launcher.png │ │ │ └── ic_launcher_round.png │ │ ├── mipmap-mdpi │ │ │ ├── ic_launcher.png │ │ │ └── ic_launcher_round.png │ │ ├── mipmap-xhdpi │ │ │ ├── ic_launcher.png │ │ │ └── ic_launcher_round.png │ │ ├── mipmap-xxhdpi │ │ │ ├── ic_launcher.png │ │ │ └── ic_launcher_round.png │ │ ├── mipmap-xxxhdpi │ │ │ ├── ic_launcher.png │ │ │ └── ic_launcher_round.png │ │ ├── values │ │ │ ├── integers.xml │ │ │ ├── colors.xml │ │ │ ├── strings.xml │ │ │ └── styles.xml │ │ ├── drawable │ │ │ └── ic_plus_sign.xml │ │ └── layout │ │ │ ├── shopping_list_item.xml │ │ │ ├── lobby_activity.xml │ │ │ ├── shopping_list_fragment.xml │ │ │ └── add_shopping_list_item_fragment.xml │ │ ├── java │ │ └── com │ │ │ └── jshvarts │ │ │ └── shoppinglist │ │ │ ├── common │ │ │ ├── domain │ │ │ │ └── model │ │ │ │ │ ├── DatabaseConstants.java │ │ │ │ │ ├── Specification.java │ │ │ │ │ ├── ItemByIdSpecification.java │ │ │ │ │ ├── ItemsSpecification.java │ │ │ │ │ ├── ShoppingListItemNameComparator.java │ │ │ │ │ ├── Repository.java │ │ │ │ │ ├── ShoppingListItem.java │ │ │ │ │ ├── ShoppingList.java │ │ │ │ │ ├── ShoppingListDataHelper.java │ │ │ │ │ ├── stub │ │ │ │ │ └── StubShoppingListRepository.java │ │ │ │ │ └── firebase │ │ │ │ │ └── FirebaseShoppingListRepository.java │ │ │ └── viewmodel │ │ │ │ └── SingleLiveEvent.java │ │ │ ├── lobby │ │ │ ├── fragments │ │ │ │ ├── UpdateShoppingListUseCase.java │ │ │ │ ├── CreateShoppingListUseCase.java │ │ │ │ ├── AddShoppingListItemModule.java │ │ │ │ ├── LoadShoppingListUseCase.java │ │ │ │ ├── AddShoppingListItemUseCase.java │ │ │ │ ├── AddShoppingListItemViewModelFactory.java │ │ │ │ ├── ShoppingListViewModelFactory.java │ │ │ │ ├── ViewShoppingListModule.java │ │ │ │ ├── DeleteCompletedItemsViewModelFactory.java │ │ │ │ ├── ShoppingListAdapter.java │ │ │ │ ├── AddShoppingListItemViewModel.java │ │ │ │ ├── AddShoppingListItemFragment.java │ │ │ │ ├── ShoppingListViewModel.java │ │ │ │ ├── DeleteCompletedItemsViewModel.java │ │ │ │ └── ViewShoppingListFragment.java │ │ │ └── LobbyActivity.java │ │ │ ├── di │ │ │ ├── AppComponent.java │ │ │ ├── BuildersModule.java │ │ │ └── AppModule.java │ │ │ ├── rx │ │ │ └── SchedulersFacade.java │ │ │ └── App.java │ │ └── AndroidManifest.xml ├── proguard-rules.pro ├── google-services.json └── build.gradle ├── settings.gradle ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── .gitignore ├── README.md ├── .travis.yml ├── gradle.properties ├── config └── jacoco.gradle ├── gradlew.bat └── gradlew /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /settings.gradle: -------------------------------------------------------------------------------- 1 | include ':app' 2 | -------------------------------------------------------------------------------- /app/src/test/resources/mockito-extensions/org.mockito.plugins.MockMaker: -------------------------------------------------------------------------------- 1 | mock-maker-inline 2 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | *.iml 2 | .gradle 3 | /local.properties 4 | .idea 5 | .DS_Store 6 | /build 7 | /captures 8 | .externalNativeBuild 9 | -------------------------------------------------------------------------------- /app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-hdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/app/src/main/res/mipmap-hdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-mdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/app/src/main/res/mipmap-mdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/jshvarts/ShoppingList/HEAD/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /app/src/main/res/values/integers.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 250 4 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/common/domain/model/DatabaseConstants.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.common.domain.model; 2 | 3 | public class DatabaseConstants { 4 | public static String DEFAULT_SHOPPING_LIST_ID = "-Ks4AYWRbtHXRd4dUJl2"; 5 | } 6 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/common/domain/model/Specification.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.common.domain.model; 2 | 3 | /** 4 | * Marker interface to be implemented by concrete Specifications. 5 | */ 6 | public interface Specification { 7 | } 8 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | #Sun Oct 29 19:58:25 EDT 2017 2 | distributionBase=GRADLE_USER_HOME 3 | distributionPath=wrapper/dists 4 | zipStoreBase=GRADLE_USER_HOME 5 | zipStorePath=wrapper/dists 6 | distributionUrl=https\://services.gradle.org/distributions/gradle-4.1-all.zip 7 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Shopping List App 2 | 3 | [![Build Status](https://travis-ci.org/jshvarts/ShoppingList.svg?branch=master)](https://travis-ci.org/jshvarts/ShoppingList) 4 | 5 | Shopping List app implemented using MVVM design pattern with Firebase, LiveData, RxJava 2, Dagger 2.11 using Clean Architecture and Repository Design pattern. 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/colors.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | #3F51B5 4 | #303F9F 5 | #FF4081 6 | 7 | #ffffffff 8 | 9 | 10 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/common/domain/model/ItemByIdSpecification.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.common.domain.model; 2 | 3 | public class ItemByIdSpecification implements Specification { 4 | 5 | private final String id; 6 | 7 | public ItemByIdSpecification(String id) { 8 | this.id = id; 9 | } 10 | 11 | public String getId() { 12 | return id; 13 | } 14 | } 15 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/common/domain/model/ItemsSpecification.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.common.domain.model; 2 | 3 | public class ItemsSpecification implements Specification { 4 | private final int maxCount; 5 | 6 | public ItemsSpecification(int maxCount) { 7 | this.maxCount = maxCount; 8 | } 9 | 10 | public int getMaxCount() { 11 | return maxCount; 12 | } 13 | } 14 | -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | language: android 2 | android: 3 | components: 4 | - tools 5 | - platform-tools 6 | - tools 7 | 8 | # The BuildTools version used by your project 9 | - build-tools-26.0.2 10 | 11 | # The SDK version used to compile your project 12 | - android-26 13 | 14 | # Additional components 15 | - extra-google-google_play_services 16 | - extra-google-m2repository 17 | - extra-android-m2repository 18 | - addon-google_apis-google-26 19 | 20 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/common/domain/model/ShoppingListItemNameComparator.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.common.domain.model; 2 | 3 | import java.util.Comparator; 4 | 5 | /** 6 | * Allows sorting items by name 7 | */ 8 | public class ShoppingListItemNameComparator implements Comparator { 9 | @Override 10 | public int compare(ShoppingListItem o1, ShoppingListItem o2) { 11 | return o1.getName().compareTo(o2.getName()); 12 | } 13 | } 14 | -------------------------------------------------------------------------------- /app/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | Shopping List 3 | item name 4 | Add 5 | Invalid item name. 6 | Unable to add shopping list item. 7 | Deleted completed items. 8 | 9 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/common/domain/model/Repository.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.common.domain.model; 2 | 3 | import java.util.List; 4 | 5 | import io.reactivex.Completable; 6 | import io.reactivex.Observable; 7 | import io.reactivex.Single; 8 | 9 | public interface Repository { 10 | Single> getItems(Specification specification); 11 | 12 | Observable getItem(Specification specification); 13 | 14 | Single add(T item); 15 | 16 | Single update(T item); 17 | 18 | Completable removeItem(Specification specification); 19 | } 20 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_plus_sign.xml: -------------------------------------------------------------------------------- 1 | 2 | 7 | 8 | 9 | 14 | 15 | -------------------------------------------------------------------------------- /app/src/main/res/layout/shopping_list_item.xml: -------------------------------------------------------------------------------- 1 | 2 | 8 | 9 | 15 | 16 | -------------------------------------------------------------------------------- /app/src/test/java/com/jshvarts/shoppinglist/rx/SchedulersFacadeUtils.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.rx; 2 | 3 | import io.reactivex.schedulers.TestScheduler; 4 | 5 | import static org.mockito.Mockito.when; 6 | 7 | public class SchedulersFacadeUtils { 8 | public static TestScheduler setupSchedulersFacade(SchedulersFacade schedulersFacade) { 9 | TestScheduler testScheduler = new TestScheduler(); 10 | when(schedulersFacade.computation()).thenReturn(testScheduler); 11 | when(schedulersFacade.ui()).thenReturn(testScheduler); 12 | when(schedulersFacade.io()).thenReturn(testScheduler); 13 | return testScheduler; 14 | } 15 | } 16 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/lobby/fragments/UpdateShoppingListUseCase.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.lobby.fragments; 2 | 3 | import com.jshvarts.shoppinglist.common.domain.model.ShoppingList; 4 | import com.jshvarts.shoppinglist.common.domain.model.firebase.FirebaseShoppingListRepository; 5 | 6 | import javax.inject.Inject; 7 | 8 | import io.reactivex.Single; 9 | 10 | class UpdateShoppingListUseCase { 11 | private final FirebaseShoppingListRepository repository; 12 | 13 | @Inject 14 | UpdateShoppingListUseCase(FirebaseShoppingListRepository repository) { 15 | this.repository = repository; 16 | } 17 | 18 | Single updateShoppingList(ShoppingList shoppingList) { 19 | return repository.update(shoppingList); 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/di/AppComponent.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.di; 2 | 3 | import com.jshvarts.shoppinglist.App; 4 | 5 | import javax.inject.Singleton; 6 | 7 | import dagger.BindsInstance; 8 | import dagger.Component; 9 | import dagger.android.support.AndroidSupportInjectionModule; 10 | 11 | @Singleton 12 | @Component(modules = { 13 | /* Use AndroidInjectionModule.class if you're not using support library */ 14 | AndroidSupportInjectionModule.class, 15 | AppModule.class, 16 | BuildersModule.class}) 17 | public interface AppComponent { 18 | @Component.Builder 19 | interface Builder { 20 | @BindsInstance 21 | Builder application(App application); 22 | AppComponent build(); 23 | } 24 | void inject(App app); 25 | } 26 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/lobby/fragments/CreateShoppingListUseCase.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.lobby.fragments; 2 | 3 | import com.jshvarts.shoppinglist.common.domain.model.ShoppingList; 4 | import com.jshvarts.shoppinglist.common.domain.model.firebase.FirebaseShoppingListRepository; 5 | 6 | import javax.inject.Inject; 7 | 8 | import io.reactivex.Single; 9 | 10 | class CreateShoppingListUseCase { 11 | private final FirebaseShoppingListRepository repository; 12 | 13 | @Inject 14 | CreateShoppingListUseCase(FirebaseShoppingListRepository repository) { 15 | this.repository = repository; 16 | } 17 | 18 | Single execute() { 19 | ShoppingList shoppingList = new ShoppingList(); 20 | return repository.add(shoppingList); 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /gradle.properties: -------------------------------------------------------------------------------- 1 | # Project-wide Gradle settings. 2 | 3 | # IDE (e.g. Android Studio) users: 4 | # Gradle settings configured through the IDE *will override* 5 | # any settings specified in this file. 6 | 7 | # For more details on how to configure your build environment visit 8 | # http://www.gradle.org/docs/current/userguide/build_environment.html 9 | 10 | # Specifies the JVM arguments used for the daemon process. 11 | # The setting is particularly useful for tweaking memory settings. 12 | org.gradle.jvmargs=-Xmx1536m 13 | 14 | # When configured, Gradle will run in incubating parallel mode. 15 | # This option should only be used with decoupled projects. More details, visit 16 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects 17 | # org.gradle.parallel=true 18 | 19 | android.enableAapt2=false -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/lobby/fragments/AddShoppingListItemModule.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.lobby.fragments; 2 | 3 | import com.jshvarts.shoppinglist.rx.SchedulersFacade; 4 | 5 | import dagger.Module; 6 | import dagger.Provides; 7 | 8 | /** 9 | * Define ShoppingListItemFragment-specific dependencies here. 10 | */ 11 | @Module 12 | public class AddShoppingListItemModule { 13 | 14 | @Provides 15 | AddShoppingListItemViewModelFactory provideAddShoppingListItemViewModelFactory( 16 | AddShoppingListItemUseCase addShoppingListItemUseCase, 17 | LoadShoppingListUseCase loadShoppingListUseCase, 18 | SchedulersFacade schedulersFacade) { 19 | return new AddShoppingListItemViewModelFactory(addShoppingListItemUseCase, loadShoppingListUseCase, schedulersFacade); 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /app/src/test/java/com/jshvarts/shoppinglist/common/domain/model/ItemByIdSpecificationTest.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.common.domain.model; 2 | 3 | import org.junit.Before; 4 | import org.junit.Test; 5 | 6 | import static org.hamcrest.MatcherAssert.assertThat; 7 | import static org.hamcrest.Matchers.is; 8 | 9 | /** 10 | * Unit tests for {@link ItemByIdSpecification}. 11 | */ 12 | public class ItemByIdSpecificationTest { 13 | 14 | private final static String TEST_ID = "id"; 15 | 16 | ItemByIdSpecification testSubject; 17 | 18 | @Before 19 | public void setUp() throws Exception { 20 | testSubject = new ItemByIdSpecification(TEST_ID); 21 | } 22 | 23 | @Test 24 | public void getId_returnsId() throws Exception { 25 | // WHEN 26 | String result = testSubject.getId(); 27 | 28 | // THEN 29 | assertThat(result, is(TEST_ID)); 30 | } 31 | } -------------------------------------------------------------------------------- /app/src/test/java/com/jshvarts/shoppinglist/common/domain/model/ItemsSpecificationTest.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.common.domain.model; 2 | 3 | import org.junit.Before; 4 | import org.junit.Test; 5 | 6 | import static org.hamcrest.MatcherAssert.assertThat; 7 | import static org.hamcrest.Matchers.is; 8 | 9 | /** 10 | * Unit tests for {@link ItemsSpecification}. 11 | */ 12 | public class ItemsSpecificationTest { 13 | 14 | private final static int TEST_MAX_COUNT = 1; 15 | 16 | ItemsSpecification testSubject; 17 | 18 | @Before 19 | public void setUp() throws Exception { 20 | testSubject = new ItemsSpecification(TEST_MAX_COUNT); 21 | } 22 | 23 | @Test 24 | public void getMaxCount_returnsMaxCount() throws Exception { 25 | // WHEN 26 | int result = testSubject.getMaxCount(); 27 | 28 | // THEN 29 | assertThat(result, is(TEST_MAX_COUNT)); 30 | } 31 | } -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/lobby/fragments/LoadShoppingListUseCase.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.lobby.fragments; 2 | 3 | import com.jshvarts.shoppinglist.common.domain.model.ItemByIdSpecification; 4 | import com.jshvarts.shoppinglist.common.domain.model.ShoppingList; 5 | import com.jshvarts.shoppinglist.common.domain.model.firebase.FirebaseShoppingListRepository; 6 | 7 | import javax.inject.Inject; 8 | 9 | import io.reactivex.Observable; 10 | 11 | class LoadShoppingListUseCase { 12 | private final FirebaseShoppingListRepository repository; 13 | 14 | @Inject 15 | LoadShoppingListUseCase(FirebaseShoppingListRepository repository) { 16 | this.repository = repository; 17 | } 18 | 19 | Observable loadShoppingList(String id) { 20 | ItemByIdSpecification byIdSpecification = new ItemByIdSpecification(id); 21 | return repository.getItem(byIdSpecification); 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/rx/SchedulersFacade.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.rx; 2 | 3 | import javax.inject.Inject; 4 | 5 | import io.reactivex.Scheduler; 6 | import io.reactivex.android.schedulers.AndroidSchedulers; 7 | import io.reactivex.schedulers.Schedulers; 8 | 9 | /** 10 | * Provides various threading schedulers. 11 | */ 12 | 13 | public class SchedulersFacade { 14 | 15 | @Inject 16 | public SchedulersFacade() { 17 | } 18 | 19 | /** 20 | * IO thread pool scheduler 21 | */ 22 | public Scheduler io() { 23 | return Schedulers.io(); 24 | } 25 | 26 | /** 27 | * Computation thread pool scheduler 28 | */ 29 | public Scheduler computation() { 30 | return Schedulers.computation(); 31 | } 32 | 33 | /** 34 | * Main Thread scheduler 35 | */ 36 | public Scheduler ui() { 37 | return AndroidSchedulers.mainThread(); 38 | } 39 | } -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 13 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | -------------------------------------------------------------------------------- /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/shvartsy/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 | 19 | # Uncomment this to preserve the line number information for 20 | # debugging stack traces. 21 | #-keepattributes SourceFile,LineNumberTable 22 | 23 | # If you keep the line number information, uncomment this to 24 | # hide the original source file name. 25 | #-renamesourcefileattribute SourceFile 26 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/lobby/fragments/AddShoppingListItemUseCase.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.lobby.fragments; 2 | 3 | import com.jshvarts.shoppinglist.common.domain.model.ShoppingList; 4 | import com.jshvarts.shoppinglist.common.domain.model.ShoppingListDataHelper; 5 | import com.jshvarts.shoppinglist.common.domain.model.firebase.FirebaseShoppingListRepository; 6 | 7 | import javax.inject.Inject; 8 | 9 | import io.reactivex.Single; 10 | 11 | class AddShoppingListItemUseCase { 12 | 13 | private final FirebaseShoppingListRepository repository; 14 | 15 | private final ShoppingListDataHelper shoppingListDataHelper; 16 | 17 | @Inject 18 | AddShoppingListItemUseCase(FirebaseShoppingListRepository repository, ShoppingListDataHelper shoppingListDataHelper) { 19 | this.repository = repository; 20 | this.shoppingListDataHelper = shoppingListDataHelper; 21 | } 22 | 23 | Single execute(ShoppingList shoppingList, String itemName) { 24 | shoppingListDataHelper.addItem(shoppingList, itemName); 25 | return repository.update(shoppingList); 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/di/BuildersModule.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.di; 2 | 3 | import com.jshvarts.shoppinglist.lobby.LobbyActivity; 4 | import com.jshvarts.shoppinglist.lobby.fragments.ViewShoppingListModule; 5 | import com.jshvarts.shoppinglist.lobby.fragments.AddShoppingListItemFragment; 6 | import com.jshvarts.shoppinglist.lobby.fragments.AddShoppingListItemModule; 7 | import com.jshvarts.shoppinglist.lobby.fragments.ViewShoppingListFragment; 8 | 9 | import dagger.Module; 10 | import dagger.android.ContributesAndroidInjector; 11 | 12 | /** 13 | * Binds all sub-components within the app. 14 | */ 15 | @Module 16 | public abstract class BuildersModule { 17 | 18 | @ContributesAndroidInjector 19 | abstract LobbyActivity bindLobbyActivity(); 20 | 21 | @ContributesAndroidInjector(modules = ViewShoppingListModule.class) 22 | abstract ViewShoppingListFragment bindViewShoppingListFragment(); 23 | 24 | @ContributesAndroidInjector(modules = AddShoppingListItemModule.class) 25 | abstract AddShoppingListItemFragment bindAddShoppingListItemFragment(); 26 | 27 | // Add bindings for other sub-components here 28 | } 29 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/common/domain/model/ShoppingListItem.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.common.domain.model; 2 | 3 | import com.google.firebase.database.Exclude; 4 | import com.google.firebase.database.ServerValue; 5 | 6 | import java.util.Map; 7 | 8 | public class ShoppingListItem { 9 | private String name; 10 | private boolean completed; 11 | private Long timestamp; 12 | 13 | public ShoppingListItem() { 14 | // required by Firebase 15 | } 16 | 17 | public ShoppingListItem(String name) { 18 | this.name = name; 19 | } 20 | 21 | public String getName() { 22 | return name; 23 | } 24 | 25 | public void setName(String name) { 26 | this.name = name; 27 | } 28 | 29 | public boolean getCompleted() { 30 | return completed; 31 | } 32 | 33 | public void setCompleted(boolean completed) { 34 | this.completed = completed; 35 | } 36 | 37 | public Map getTimestamp() { 38 | return ServerValue.TIMESTAMP; 39 | } 40 | 41 | @Exclude 42 | public Long getTimestampLong() { 43 | return timestamp; 44 | } 45 | } 46 | -------------------------------------------------------------------------------- /app/src/main/res/layout/lobby_activity.xml: -------------------------------------------------------------------------------- 1 | 2 | 9 | 10 | 13 | 14 | 19 | 20 | 21 | 22 | 26 | 27 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/di/AppModule.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.di; 2 | 3 | import android.app.Application; 4 | import android.content.Context; 5 | 6 | import com.jshvarts.shoppinglist.App; 7 | import com.jshvarts.shoppinglist.common.domain.model.ShoppingListDataHelper; 8 | import com.jshvarts.shoppinglist.common.domain.model.firebase.FirebaseShoppingListRepository; 9 | 10 | import javax.inject.Singleton; 11 | 12 | import dagger.Module; 13 | import dagger.Provides; 14 | 15 | /** 16 | * This is where you will inject application-wide dependencies. 17 | */ 18 | @Module 19 | public class AppModule { 20 | 21 | @Provides 22 | Context provideContext(App application) { 23 | return application.getApplicationContext(); 24 | } 25 | 26 | @Provides 27 | Application provideApplication(App application) { 28 | return application; 29 | } 30 | 31 | @Singleton 32 | @Provides 33 | FirebaseShoppingListRepository provideFirebaseShoppingListRepository() { 34 | return new FirebaseShoppingListRepository(); 35 | } 36 | 37 | @Singleton 38 | @Provides 39 | ShoppingListDataHelper provideShoppingListDataHelper() { 40 | return new ShoppingListDataHelper(); 41 | } 42 | } 43 | -------------------------------------------------------------------------------- /app/src/test/java/com/jshvarts/shoppinglist/lobby/fragments/CreateShoppingListUseCaseTest.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.lobby.fragments; 2 | 3 | import com.jshvarts.shoppinglist.common.domain.model.ShoppingList; 4 | import com.jshvarts.shoppinglist.common.domain.model.firebase.FirebaseShoppingListRepository; 5 | 6 | import org.junit.Before; 7 | import org.junit.Test; 8 | import org.mockito.ArgumentMatchers; 9 | import org.mockito.InjectMocks; 10 | import org.mockito.Mock; 11 | import org.mockito.MockitoAnnotations; 12 | 13 | import static org.mockito.Mockito.spy; 14 | import static org.mockito.Mockito.verify; 15 | 16 | /** 17 | * Unit tests for {@link CreateShoppingListUseCase}. 18 | */ 19 | public class CreateShoppingListUseCaseTest { 20 | @InjectMocks 21 | CreateShoppingListUseCase testSubject; 22 | 23 | @Mock 24 | FirebaseShoppingListRepository repository; 25 | 26 | @Before 27 | public void setUp() throws Exception { 28 | MockitoAnnotations.initMocks(this); 29 | } 30 | 31 | @Test 32 | public void execute_delegatesToRepository() throws Exception { 33 | // WHEN 34 | testSubject.execute(); 35 | 36 | // THEN 37 | verify(repository).add(ArgumentMatchers.any(ShoppingList.class)); 38 | } 39 | } -------------------------------------------------------------------------------- /app/src/test/java/com/jshvarts/shoppinglist/lobby/fragments/UpdateShoppingListUseCaseTest.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.lobby.fragments; 2 | 3 | import com.jshvarts.shoppinglist.common.domain.model.ShoppingList; 4 | import com.jshvarts.shoppinglist.common.domain.model.firebase.FirebaseShoppingListRepository; 5 | 6 | import org.junit.Before; 7 | import org.junit.Test; 8 | import org.mockito.InjectMocks; 9 | import org.mockito.Mock; 10 | import org.mockito.MockitoAnnotations; 11 | 12 | import static org.mockito.Mockito.mock; 13 | import static org.mockito.Mockito.verify; 14 | 15 | /** 16 | * Unit tests for {@link UpdateShoppingListUseCase}. 17 | */ 18 | public class UpdateShoppingListUseCaseTest { 19 | 20 | @InjectMocks 21 | UpdateShoppingListUseCase testSubject; 22 | 23 | @Mock 24 | FirebaseShoppingListRepository repository; 25 | 26 | @Before 27 | public void setUp() throws Exception { 28 | MockitoAnnotations.initMocks(this); 29 | } 30 | 31 | @Test 32 | public void updateShoppingList_delegatesToRepository() throws Exception { 33 | // GIVEN 34 | ShoppingList shoppingList = mock(ShoppingList.class); 35 | 36 | // WHEN 37 | testSubject.updateShoppingList(shoppingList); 38 | 39 | // THEN 40 | verify(repository).update(shoppingList); 41 | } 42 | } -------------------------------------------------------------------------------- /app/src/main/res/layout/shopping_list_fragment.xml: -------------------------------------------------------------------------------- 1 | 2 | 7 | 8 | 14 | 15 | 21 | 22 | 31 | 32 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/App.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist; 2 | 3 | import android.app.Activity; 4 | import android.app.Application; 5 | import android.content.Context; 6 | import android.support.multidex.MultiDex; 7 | 8 | import com.jshvarts.shoppinglist.di.DaggerAppComponent; 9 | 10 | import javax.inject.Inject; 11 | 12 | import dagger.android.AndroidInjector; 13 | import dagger.android.DispatchingAndroidInjector; 14 | import dagger.android.HasActivityInjector; 15 | import timber.log.Timber; 16 | 17 | public class App extends Application implements HasActivityInjector { 18 | @Inject 19 | DispatchingAndroidInjector dispatchingAndroidInjector; 20 | 21 | @Override 22 | public void onCreate() { 23 | super.onCreate(); 24 | 25 | if (BuildConfig.DEBUG) { 26 | Timber.plant(new Timber.DebugTree()); 27 | } 28 | 29 | DaggerAppComponent 30 | .builder() 31 | .application(this) 32 | .build() 33 | .inject(this); 34 | } 35 | 36 | @Override 37 | protected void attachBaseContext(Context base) { 38 | super.attachBaseContext(base); 39 | MultiDex.install(this); 40 | } 41 | 42 | @Override 43 | public AndroidInjector activityInjector() { 44 | return dispatchingAndroidInjector; 45 | } 46 | } 47 | -------------------------------------------------------------------------------- /app/src/main/java/com/jshvarts/shoppinglist/common/domain/model/ShoppingList.java: -------------------------------------------------------------------------------- 1 | package com.jshvarts.shoppinglist.common.domain.model; 2 | 3 | import com.google.firebase.database.Exclude; 4 | import com.google.firebase.database.IgnoreExtraProperties; 5 | import com.google.firebase.database.ServerValue; 6 | 7 | import java.util.ArrayList; 8 | import java.util.List; 9 | import java.util.Map; 10 | 11 | @IgnoreExtraProperties 12 | public class ShoppingList { 13 | 14 | private String id; 15 | // Only List is supported by Firebase. Set would be more is more applicable though 16 | private List items = new ArrayList<>(); 17 | private Long timestamp; 18 | 19 | public ShoppingList() { 20 | // required by Firebase 21 | } 22 | 23 | public ShoppingList(List items) { 24 | this.items = items; 25 | } 26 | 27 | @Exclude 28 | public String getId() { 29 | return id; 30 | } 31 | 32 | public void setId(String id) { 33 | this.id = id; 34 | } 35 | 36 | public List getItems() { 37 | return items; 38 | } 39 | 40 | public void setItems(List items) { 41 | this.items = items; 42 | } 43 | 44 | public Map getTimestamp() { 45 | return ServerValue.TIMESTAMP; 46 | } 47 | 48 | @Exclude 49 | public Long getTimestampLong() { 50 | return timestamp; 51 | } 52 | } 53 | -------------------------------------------------------------------------------- /app/src/main/res/layout/add_shopping_list_item_fragment.xml: -------------------------------------------------------------------------------- 1 | 2 | 9 | 10 | 17 | 18 | 24 | 25 |