From fa79d3eac2695c79d61b0da3f73a153dca528d23 Mon Sep 17 00:00:00 2001 From: Lucas Lima Date: Thu, 6 Feb 2020 22:50:47 -0300 Subject: [PATCH] First open source release --- .gitignore | 13 + app/.gitignore | 1 + app/build.gradle | 62 +++ app/fabric.properties | 1 + app/gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 54708 bytes app/gradle/wrapper/gradle-wrapper.properties | 6 + app/gradlew | 172 ++++++ app/gradlew.bat | 84 +++ app/proguard-rules.pro | 43 ++ app/src/main/AndroidManifest.xml | 124 +++++ .../dev/lucasnlm/antimine/GameActivity.kt | 504 ++++++++++++++++++ .../dev/lucasnlm/antimine/MainApplication.kt | 27 + .../dev/lucasnlm/antimine/TvGameActivity.kt | 318 +++++++++++ .../lucasnlm/antimine/about/AboutActivity.kt | 47 ++ .../dev/lucasnlm/antimine/about/Constants.kt | 6 + .../lucasnlm/antimine/about/TextActivity.kt | 97 ++++ .../about/thirds/ThirdPartiesActivity.kt | 78 +++ .../antimine/about/thirds/data/ThirdParty.kt | 8 + .../about/thirds/view/ThirdPartyAdapter.kt | 37 ++ .../about/thirds/view/ThirdPartyItemHolder.kt | 11 + .../lucasnlm/antimine/di/ActivityModule.kt | 18 + .../dev/lucasnlm/antimine/di/AppComponent.kt | 35 ++ .../dev/lucasnlm/antimine/di/AppModule.kt | 14 + .../lucasnlm/antimine/di/FragmentModule.kt | 18 + .../level/view/CustomLevelDialogFragment.kt | 88 +++ .../antimine/level/view/LevelFragment.kt | 126 +++++ .../preferences/PreferencesActivity.kt | 29 + .../antimine/splash/SplashActivity.kt | 24 + app/src/main/res/drawable/mine_white.xml | 10 + app/src/main/res/layout/activity_about.xml | 59 ++ app/src/main/res/layout/activity_game.xml | 104 ++++ app/src/main/res/layout/activity_text.xml | 39 ++ .../main/res/layout/activity_third_party.xml | 31 ++ app/src/main/res/layout/activity_tv_game.xml | 74 +++ .../main/res/layout/dialog_custom_game.xml | 102 ++++ app/src/main/res/layout/view_third_party.xml | 21 + app/src/main/res/menu/top_menu_over.xml | 13 + app/src/main/res/values/strings.xml | 4 + app/src/main/res/xml/backup_rules.xml | 5 + app/src/main/res/xml/shortcuts.xml | 52 ++ build.gradle | 40 ++ common/.gitignore | 1 + common/build.gradle | 68 +++ common/gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 54708 bytes .../gradle/wrapper/gradle-wrapper.properties | 6 + common/gradlew | 172 ++++++ common/gradlew.bat | 84 +++ common/proguard-rules.pro | 47 ++ common/src/main/AndroidManifest.xml | 9 + .../antimine/common/level/GameModeFactory.kt | 41 ++ .../antimine/common/level/LevelFacade.kt | 264 +++++++++ .../common/level/data/AmbientSettings.kt | 6 + .../antimine/common/level/data/Area.kt | 14 + .../common/level/data/DifficultyPreset.kt | 9 + .../antimine/common/level/data/GameEvent.kt | 13 + .../antimine/common/level/data/GameStats.kt | 7 + .../antimine/common/level/data/GameStatus.kt | 7 + .../antimine/common/level/data/LevelSetup.kt | 8 + .../antimine/common/level/data/Mark.kt | 11 + .../common/level/database/AppDataBase.kt | 24 + .../database/converters/FieldConverter.kt | 25 + .../converters/LevelSetupConverter.kt | 24 + .../converters/SaveStatusConverter.kt | 19 + .../common/level/database/dao/SaveDao.kt | 28 + .../common/level/database/data/Save.kt | 36 ++ .../common/level/database/data/SaveStatus.kt | 11 + .../common/level/di/LevelComponent.kt | 18 + .../antimine/common/level/di/LevelModule.kt | 73 +++ .../level/repository/DimensionRepository.kt | 38 ++ .../level/repository/DrawableRepository.kt | 44 ++ .../level/repository/SavesRepository.kt | 22 + .../antimine/common/level/utils/Clock.kt | 39 ++ .../level/utils/HapticFeedbackInteractor.kt | 44 ++ .../antimine/common/level/view/AreaAdapter.kt | 109 ++++ .../common/level/view/AreaPaintSettings.kt | 11 + .../antimine/common/level/view/AreaView.kt | 273 ++++++++++ .../common/level/view/FieldViewHolder.kt | 46 ++ .../view/UnlockedHorizontalScrollView.kt | 31 ++ .../common/level/viewmodel/GameViewModel.kt | 257 +++++++++ .../level/viewmodel/GameViewModelFactory.kt | 38 ++ .../analytics/AmplitudeAnalyticsManager.kt | 27 + .../core/analytics/AnalyticsManager.kt | 8 + .../core/analytics/DebugAnalyticsManager.kt | 18 + .../lucasnlm/antimine/core/analytics/Event.kt | 82 +++ .../antimine/core/di/CommonComponent.kt | 18 + .../lucasnlm/antimine/core/di/CommonModule.kt | 38 ++ .../core/preferences/PreferencesInteractor.kt | 42 ++ .../core/preferences/PreferencesRepository.kt | 49 ++ .../antimine/core/scope/ActivityScope.kt | 7 + .../antimine/core/scope/FragmentScope.kt | 7 + .../antimine/core/utils/DarkModeUtils.kt | 12 + .../main/res/drawable-night/mine_exploded.xml | 14 + .../src/main/res/drawable-night/red_flag.xml | 10 + common/src/main/res/drawable-night/replay.xml | 9 + common/src/main/res/drawable-night/splash.xml | 13 + .../main/res/drawable-night/splash_image.png | Bin 0 -> 6291 bytes common/src/main/res/drawable/checked.xml | 9 + common/src/main/res/drawable/close.xml | 9 + common/src/main/res/drawable/done.xml | 9 + common/src/main/res/drawable/flag.xml | 10 + common/src/main/res/drawable/main_logo.xml | 18 + common/src/main/res/drawable/mine.xml | 14 + .../src/main/res/drawable/mine_exploded.xml | 14 + common/src/main/res/drawable/mine_icon.xml | 10 + common/src/main/res/drawable/mine_low.xml | 10 + common/src/main/res/drawable/question.xml | 5 + common/src/main/res/drawable/red_flag.xml | 10 + common/src/main/res/drawable/replay.xml | 9 + common/src/main/res/drawable/splash.xml | 13 + common/src/main/res/drawable/splash_image.png | Bin 0 -> 9316 bytes common/src/main/res/drawable/timer.xml | 10 + common/src/main/res/drawable/title.xml | 17 + common/src/main/res/drawable/unchecked.xml | 9 + .../main/res/layout-watch/fragment_level.xml | 34 ++ common/src/main/res/layout/fragment_level.xml | 32 ++ .../res/layout/view_accessibility_field.xml | 14 + common/src/main/res/layout/view_field.xml | 14 + .../res/mipmap-anydpi-v26/ic_launcher.xml | 5 + .../mipmap-anydpi-v26/ic_launcher_round.xml | 5 + .../src/main/res/mipmap-hdpi/ic_launcher.png | Bin 0 -> 4603 bytes .../mipmap-hdpi/ic_launcher_foreground.png | Bin 0 -> 1277 bytes .../res/mipmap-hdpi/ic_launcher_round.png | Bin 0 -> 4395 bytes .../src/main/res/mipmap-ldpi/ic_launcher.png | Bin 0 -> 2188 bytes .../res/mipmap-ldpi/ic_launcher_round.png | Bin 0 -> 2130 bytes .../src/main/res/mipmap-mdpi/ic_launcher.png | Bin 0 -> 2966 bytes .../mipmap-mdpi/ic_launcher_foreground.png | Bin 0 -> 765 bytes .../res/mipmap-mdpi/ic_launcher_round.png | Bin 0 -> 2851 bytes .../src/main/res/mipmap-xhdpi/ic_launcher.png | Bin 0 -> 6561 bytes .../mipmap-xhdpi/ic_launcher_foreground.png | Bin 0 -> 1575 bytes .../res/mipmap-xhdpi/ic_launcher_round.png | Bin 0 -> 6149 bytes .../main/res/mipmap-xxhdpi/ic_launcher.png | Bin 0 -> 10428 bytes .../mipmap-xxhdpi/ic_launcher_foreground.png | Bin 0 -> 3047 bytes .../res/mipmap-xxhdpi/ic_launcher_round.png | Bin 0 -> 10169 bytes .../main/res/mipmap-xxxhdpi/ic_launcher.png | Bin 0 -> 14618 bytes .../mipmap-xxxhdpi/ic_launcher_foreground.png | Bin 0 -> 3701 bytes .../res/mipmap-xxxhdpi/ic_launcher_round.png | Bin 0 -> 14768 bytes common/src/main/res/raw/android_sdk.txt | 142 +++++ common/src/main/res/raw/apache2.txt | 202 +++++++ .../src/main/res/raw/mine_explosion_sound.ogg | Bin 0 -> 14878 bytes common/src/main/res/raw/mockito.txt | 9 + common/src/main/res/raw/sounds.txt | 6 + common/src/main/res/values-es/strings.xml | 82 +++ common/src/main/res/values-night/colors.xml | 32 ++ common/src/main/res/values-pt/strings.xml | 82 +++ common/src/main/res/values-v21/themes.xml | 21 + common/src/main/res/values-w600dp/dimens.xml | 4 + common/src/main/res/values-zh/strings.xml | 82 +++ common/src/main/res/values/colors.xml | 32 ++ common/src/main/res/values/dimens.xml | 11 + common/src/main/res/values/keys.xml | 5 + common/src/main/res/values/strings.xml | 82 +++ common/src/main/res/values/themes.xml | 24 + common/src/main/res/xml/nav_menu.xml | 62 +++ common/src/main/res/xml/preferences.xml | 48 ++ .../antimine/common/level/LevelFacadeTest.kt | 412 ++++++++++++++ .../common/level/LevelSetupFactoryTest.kt | 88 +++ gradle.properties | 25 + gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 53637 bytes gradle/wrapper/gradle-wrapper.properties | 6 + gradlew | 160 ++++++ gradlew.bat | 90 ++++ settings.gradle | 1 + wear/.gitignore | 1 + wear/build.gradle | 52 ++ wear/proguard-rules.pro | 43 ++ wear/src/main/AndroidManifest.xml | 53 ++ .../lucasnlm/antimine/wear/MainApplication.kt | 14 + .../antimine/wear/WatchGameActivity.kt | 204 +++++++ .../antimine/wear/WatchLevelFragment.kt | 118 ++++ .../antimine/wear/di/ActivityModule.kt | 13 + .../lucasnlm/antimine/wear/di/AppComponent.kt | 35 ++ .../lucasnlm/antimine/wear/di/AppModule.kt | 14 + .../antimine/wear/di/FragmentModule.kt | 13 + .../main/res/drawable/round_background.xml | 5 + .../res/drawable/round_background_low.xml | 7 + wear/src/main/res/layout/activity_level.xml | 49 ++ wear/src/main/res/values-es/strings.xml | 6 + wear/src/main/res/values-pt/strings.xml | 6 + wear/src/main/res/values-zh-rCN/strings.xml | 6 + wear/src/main/res/values/strings.xml | 6 + wear/src/main/res/values/themes.xml | 12 + 181 files changed, 7521 insertions(+) create mode 100644 .gitignore create mode 100644 app/.gitignore create mode 100644 app/build.gradle create mode 100644 app/fabric.properties create mode 100644 app/gradle/wrapper/gradle-wrapper.jar create mode 100644 app/gradle/wrapper/gradle-wrapper.properties create mode 100644 app/gradlew create mode 100644 app/gradlew.bat create mode 100644 app/proguard-rules.pro create mode 100644 app/src/main/AndroidManifest.xml create mode 100644 app/src/main/java/dev/lucasnlm/antimine/GameActivity.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/MainApplication.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/TvGameActivity.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/about/AboutActivity.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/about/Constants.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/about/TextActivity.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/about/thirds/ThirdPartiesActivity.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/about/thirds/data/ThirdParty.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/about/thirds/view/ThirdPartyAdapter.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/about/thirds/view/ThirdPartyItemHolder.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/di/ActivityModule.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/di/AppComponent.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/di/AppModule.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/di/FragmentModule.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/level/view/CustomLevelDialogFragment.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/level/view/LevelFragment.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/preferences/PreferencesActivity.kt create mode 100644 app/src/main/java/dev/lucasnlm/antimine/splash/SplashActivity.kt create mode 100644 app/src/main/res/drawable/mine_white.xml create mode 100644 app/src/main/res/layout/activity_about.xml create mode 100644 app/src/main/res/layout/activity_game.xml create mode 100644 app/src/main/res/layout/activity_text.xml create mode 100644 app/src/main/res/layout/activity_third_party.xml create mode 100644 app/src/main/res/layout/activity_tv_game.xml create mode 100644 app/src/main/res/layout/dialog_custom_game.xml create mode 100644 app/src/main/res/layout/view_third_party.xml create mode 100644 app/src/main/res/menu/top_menu_over.xml create mode 100644 app/src/main/res/values/strings.xml create mode 100644 app/src/main/res/xml/backup_rules.xml create mode 100644 app/src/main/res/xml/shortcuts.xml create mode 100644 build.gradle create mode 100644 common/.gitignore create mode 100644 common/build.gradle create mode 100644 common/gradle/wrapper/gradle-wrapper.jar create mode 100644 common/gradle/wrapper/gradle-wrapper.properties create mode 100644 common/gradlew create mode 100644 common/gradlew.bat create mode 100644 common/proguard-rules.pro create mode 100644 common/src/main/AndroidManifest.xml create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/GameModeFactory.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/LevelFacade.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/data/AmbientSettings.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/data/Area.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/data/DifficultyPreset.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameEvent.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameStats.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameStatus.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/data/LevelSetup.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/data/Mark.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/database/AppDataBase.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/FieldConverter.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/LevelSetupConverter.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/SaveStatusConverter.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/database/dao/SaveDao.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/database/data/Save.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/database/data/SaveStatus.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/di/LevelComponent.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/di/LevelModule.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/repository/DimensionRepository.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/repository/DrawableRepository.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/repository/SavesRepository.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/utils/Clock.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/utils/HapticFeedbackInteractor.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaAdapter.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaPaintSettings.kt create mode 100755 common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaView.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/view/FieldViewHolder.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/view/UnlockedHorizontalScrollView.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/viewmodel/GameViewModel.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/common/level/viewmodel/GameViewModelFactory.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/analytics/AmplitudeAnalyticsManager.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/analytics/AnalyticsManager.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/analytics/DebugAnalyticsManager.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/analytics/Event.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/di/CommonComponent.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/di/CommonModule.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/preferences/PreferencesInteractor.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/preferences/PreferencesRepository.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/scope/ActivityScope.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/scope/FragmentScope.kt create mode 100644 common/src/main/java/dev/lucasnlm/antimine/core/utils/DarkModeUtils.kt create mode 100644 common/src/main/res/drawable-night/mine_exploded.xml create mode 100644 common/src/main/res/drawable-night/red_flag.xml create mode 100644 common/src/main/res/drawable-night/replay.xml create mode 100644 common/src/main/res/drawable-night/splash.xml create mode 100644 common/src/main/res/drawable-night/splash_image.png create mode 100644 common/src/main/res/drawable/checked.xml create mode 100644 common/src/main/res/drawable/close.xml create mode 100644 common/src/main/res/drawable/done.xml create mode 100755 common/src/main/res/drawable/flag.xml create mode 100644 common/src/main/res/drawable/main_logo.xml create mode 100755 common/src/main/res/drawable/mine.xml create mode 100644 common/src/main/res/drawable/mine_exploded.xml create mode 100755 common/src/main/res/drawable/mine_icon.xml create mode 100644 common/src/main/res/drawable/mine_low.xml create mode 100644 common/src/main/res/drawable/question.xml create mode 100644 common/src/main/res/drawable/red_flag.xml create mode 100644 common/src/main/res/drawable/replay.xml create mode 100644 common/src/main/res/drawable/splash.xml create mode 100644 common/src/main/res/drawable/splash_image.png create mode 100755 common/src/main/res/drawable/timer.xml create mode 100644 common/src/main/res/drawable/title.xml create mode 100644 common/src/main/res/drawable/unchecked.xml create mode 100644 common/src/main/res/layout-watch/fragment_level.xml create mode 100644 common/src/main/res/layout/fragment_level.xml create mode 100644 common/src/main/res/layout/view_accessibility_field.xml create mode 100644 common/src/main/res/layout/view_field.xml create mode 100644 common/src/main/res/mipmap-anydpi-v26/ic_launcher.xml create mode 100644 common/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml create mode 100644 common/src/main/res/mipmap-hdpi/ic_launcher.png create mode 100644 common/src/main/res/mipmap-hdpi/ic_launcher_foreground.png create mode 100644 common/src/main/res/mipmap-hdpi/ic_launcher_round.png create mode 100644 common/src/main/res/mipmap-ldpi/ic_launcher.png create mode 100644 common/src/main/res/mipmap-ldpi/ic_launcher_round.png create mode 100644 common/src/main/res/mipmap-mdpi/ic_launcher.png create mode 100644 common/src/main/res/mipmap-mdpi/ic_launcher_foreground.png create mode 100644 common/src/main/res/mipmap-mdpi/ic_launcher_round.png create mode 100644 common/src/main/res/mipmap-xhdpi/ic_launcher.png create mode 100644 common/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png create mode 100644 common/src/main/res/mipmap-xhdpi/ic_launcher_round.png create mode 100644 common/src/main/res/mipmap-xxhdpi/ic_launcher.png create mode 100644 common/src/main/res/mipmap-xxhdpi/ic_launcher_foreground.png create mode 100644 common/src/main/res/mipmap-xxhdpi/ic_launcher_round.png create mode 100644 common/src/main/res/mipmap-xxxhdpi/ic_launcher.png create mode 100644 common/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png create mode 100644 common/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png create mode 100644 common/src/main/res/raw/android_sdk.txt create mode 100644 common/src/main/res/raw/apache2.txt create mode 100644 common/src/main/res/raw/mine_explosion_sound.ogg create mode 100644 common/src/main/res/raw/mockito.txt create mode 100644 common/src/main/res/raw/sounds.txt create mode 100644 common/src/main/res/values-es/strings.xml create mode 100644 common/src/main/res/values-night/colors.xml create mode 100644 common/src/main/res/values-pt/strings.xml create mode 100644 common/src/main/res/values-v21/themes.xml create mode 100644 common/src/main/res/values-w600dp/dimens.xml create mode 100644 common/src/main/res/values-zh/strings.xml create mode 100644 common/src/main/res/values/colors.xml create mode 100644 common/src/main/res/values/dimens.xml create mode 100644 common/src/main/res/values/keys.xml create mode 100644 common/src/main/res/values/strings.xml create mode 100644 common/src/main/res/values/themes.xml create mode 100644 common/src/main/res/xml/nav_menu.xml create mode 100644 common/src/main/res/xml/preferences.xml create mode 100644 common/src/test/java/dev/lucasnlm/antimine/common/level/LevelFacadeTest.kt create mode 100644 common/src/test/java/dev/lucasnlm/antimine/common/level/LevelSetupFactoryTest.kt create mode 100644 gradle.properties create mode 100644 gradle/wrapper/gradle-wrapper.jar create mode 100644 gradle/wrapper/gradle-wrapper.properties create mode 100755 gradlew create mode 100644 gradlew.bat create mode 100644 settings.gradle create mode 100644 wear/.gitignore create mode 100644 wear/build.gradle create mode 100644 wear/proguard-rules.pro create mode 100644 wear/src/main/AndroidManifest.xml create mode 100644 wear/src/main/java/dev/lucasnlm/antimine/wear/MainApplication.kt create mode 100644 wear/src/main/java/dev/lucasnlm/antimine/wear/WatchGameActivity.kt create mode 100644 wear/src/main/java/dev/lucasnlm/antimine/wear/WatchLevelFragment.kt create mode 100644 wear/src/main/java/dev/lucasnlm/antimine/wear/di/ActivityModule.kt create mode 100644 wear/src/main/java/dev/lucasnlm/antimine/wear/di/AppComponent.kt create mode 100644 wear/src/main/java/dev/lucasnlm/antimine/wear/di/AppModule.kt create mode 100644 wear/src/main/java/dev/lucasnlm/antimine/wear/di/FragmentModule.kt create mode 100644 wear/src/main/res/drawable/round_background.xml create mode 100644 wear/src/main/res/drawable/round_background_low.xml create mode 100644 wear/src/main/res/layout/activity_level.xml create mode 100644 wear/src/main/res/values-es/strings.xml create mode 100644 wear/src/main/res/values-pt/strings.xml create mode 100644 wear/src/main/res/values-zh-rCN/strings.xml create mode 100644 wear/src/main/res/values/strings.xml create mode 100644 wear/src/main/res/values/themes.xml diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..25e345c4 --- /dev/null +++ b/.gitignore @@ -0,0 +1,13 @@ +*.iml +.gradle +/local.properties +/.idea/workspace.xml +/.idea/libraries +.DS_Store +/build +/captures +.idea/ +app/local.properties +*.apk +app/release/ +app/standalone/ diff --git a/app/.gitignore b/app/.gitignore new file mode 100644 index 00000000..796b96d1 --- /dev/null +++ b/app/.gitignore @@ -0,0 +1 @@ +/build diff --git a/app/build.gradle b/app/build.gradle new file mode 100644 index 00000000..918681ce --- /dev/null +++ b/app/build.gradle @@ -0,0 +1,62 @@ +apply plugin: 'com.android.application' +apply plugin: 'kotlin-android' +apply plugin: 'kotlin-android-extensions' +apply plugin: 'io.fabric' +apply plugin: 'kotlin-kapt' + +android { + compileSdkVersion 29 + + defaultConfig { + applicationId 'dev.lucasnlm.antimine' + minSdkVersion 16 + targetSdkVersion 29 + versionCode 40 + versionName '6.0' + + resConfigs 'en', 'pt', 'es', 'zh' + } + + buildTypes { + debug { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' + } + + release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' + } + } + + kapt { + generateStubs true + } +} + +dependencies { + implementation fileTree(dir: 'libs', include: ['*.jar']) + implementation project(':common') + + // Android + implementation 'com.google.android.material:material:1.0.0' + implementation 'com.google.android.play:core:1.6.4' + implementation 'androidx.appcompat:appcompat:1.1.0' + implementation 'androidx.preference:preference:1.1.0' + implementation 'androidx.recyclerview:recyclerview:1.1.0' + implementation 'androidx.constraintlayout:constraintlayout:1.1.3' + + // Dagger + def dagger_version='2.21' + api "com.google.dagger:dagger-android:$dagger_version" + api "com.google.dagger:dagger-android-support:$dagger_version" + kapt "com.google.dagger:dagger-android-processor:$dagger_version" + kapt "com.google.dagger:dagger-compiler:$dagger_version" + + implementation('com.crashlytics.sdk.android:crashlytics:2.6.8@aar') { + transitive = true + } + + implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.0' + implementation 'org.jetbrains.kotlin:kotlin-stdlib-jdk7:1.3.50' +} diff --git a/app/fabric.properties b/app/fabric.properties new file mode 100644 index 00000000..23b563a9 --- /dev/null +++ b/app/fabric.properties @@ -0,0 +1 @@ +apiKey=18aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa diff --git a/app/gradle/wrapper/gradle-wrapper.jar b/app/gradle/wrapper/gradle-wrapper.jar new file mode 100644 index 0000000000000000000000000000000000000000..7a3265ee94c0ab25cf079ac8ccdf87f41d455d42 GIT binary patch literal 54708 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

_x$+l{{cH8$W#CT literal 0 HcmV?d00001 diff --git a/app/gradle/wrapper/gradle-wrapper.properties b/app/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 00000000..541abfe5 --- /dev/null +++ b/app/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,6 @@ +#Mon May 21 22:33:16 BRT 2018 +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +zipStoreBase=GRADLE_USER_HOME +zipStorePath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-4.4-all.zip diff --git a/app/gradlew b/app/gradlew new file mode 100644 index 00000000..cccdd3d5 --- /dev/null +++ b/app/gradlew @@ -0,0 +1,172 @@ +#!/usr/bin/env sh + +############################################################################## +## +## Gradle start up script for UN*X +## +############################################################################## + +# Attempt to set APP_HOME +# Resolve links: $0 may be a link +PRG="$0" +# Need this for relative symlinks. +while [ -h "$PRG" ] ; do + ls=`ls -ld "$PRG"` + link=`expr "$ls" : '.*-> \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG=`dirname "$PRG"`"/$link" + fi +done +SAVED="`pwd`" +cd "`dirname \"$PRG\"`/" >/dev/null +APP_HOME="`pwd -P`" +cd "$SAVED" >/dev/null + +APP_NAME="Gradle" +APP_BASE_NAME=`basename "$0"` + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS="" + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD="maximum" + +warn () { + echo "$*" +} + +die () { + echo + echo "$*" + echo + exit 1 +} + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +nonstop=false +case "`uname`" in + CYGWIN* ) + cygwin=true + ;; + Darwin* ) + darwin=true + ;; + MINGW* ) + msys=true + ;; + NONSTOP* ) + nonstop=true + ;; +esac + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD="java" + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." +fi + +# Increase the maximum file descriptors if we can. +if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then + MAX_FD_LIMIT=`ulimit -H -n` + if [ $? -eq 0 ] ; then + if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then + MAX_FD="$MAX_FD_LIMIT" + fi + ulimit -n $MAX_FD + if [ $? -ne 0 ] ; then + warn "Could not set maximum file descriptor limit: $MAX_FD" + fi + else + warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" + fi +fi + +# For Darwin, add options to specify how the application appears in the dock +if $darwin; then + GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" +fi + +# For Cygwin, switch paths to Windows format before running java +if $cygwin ; then + APP_HOME=`cygpath --path --mixed "$APP_HOME"` + CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` + JAVACMD=`cygpath --unix "$JAVACMD"` + + # We build the pattern for arguments to be converted via cygpath + ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` + SEP="" + for dir in $ROOTDIRSRAW ; do + ROOTDIRS="$ROOTDIRS$SEP$dir" + SEP="|" + done + OURCYGPATTERN="(^($ROOTDIRS))" + # Add a user-defined pattern to the cygpath arguments + if [ "$GRADLE_CYGPATTERN" != "" ] ; then + OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" + fi + # Now convert the arguments - kludge to limit ourselves to /bin/sh + i=0 + for arg in "$@" ; do + CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` + CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option + + if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition + eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` + else + eval `echo args$i`="\"$arg\"" + fi + i=$((i+1)) + done + case $i in + (0) set -- ;; + (1) set -- "$args0" ;; + (2) set -- "$args0" "$args1" ;; + (3) set -- "$args0" "$args1" "$args2" ;; + (4) set -- "$args0" "$args1" "$args2" "$args3" ;; + (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; + (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; + (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; + (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; + (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; + esac +fi + +# Escape application args +save () { + for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done + echo " " +} +APP_ARGS=$(save "$@") + +# Collect all arguments for the java command, following the shell quoting and substitution rules +eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" + +# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong +if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then + cd "$(dirname "$0")" +fi + +exec "$JAVACMD" "$@" diff --git a/app/gradlew.bat b/app/gradlew.bat new file mode 100644 index 00000000..e95643d6 --- /dev/null +++ b/app/gradlew.bat @@ -0,0 +1,84 @@ +@if "%DEBUG%" == "" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +set DIRNAME=%~dp0 +if "%DIRNAME%" == "" set DIRNAME=. +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS= + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if "%ERRORLEVEL%" == "0" goto init + +echo. +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto init + +echo. +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:init +@rem Get command-line arguments, handling Windows variants + +if not "%OS%" == "Windows_NT" goto win9xME_args + +:win9xME_args +@rem Slurp the command line arguments. +set CMD_LINE_ARGS= +set _SKIP=2 + +:win9xME_args_slurp +if "x%~1" == "x" goto execute + +set CMD_LINE_ARGS=%* + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% + +:end +@rem End local scope for the variables with windows NT shell +if "%ERRORLEVEL%"=="0" goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 +exit /b 1 + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/app/proguard-rules.pro b/app/proguard-rules.pro new file mode 100644 index 00000000..19a93f16 --- /dev/null +++ b/app/proguard-rules.pro @@ -0,0 +1,43 @@ +# Project specific ProGuard rules + +# Crashlytics 2.+ +-keep class com.crashlytics.** { *; } +-keep class com.crashlytics.android.** +-keepattributes SourceFile, LineNumberTable, *Annotation* + +# For Fabric to properly de-obfuscate your crash reports, you need to remove this line from your ProGuard config: +-printmapping mapping.txt + +# support design +-dontwarn android.support.design.** +-keep class android.support.design.** { *; } +-keep interface android.support.design.** { *; } +-keep public class android.support.design.R$* { *; } + +# support v7 +-keep public class android.support.v7.widget.** { *; } +-keep public class android.support.v7.internal.widget.** { *; } +-keep public class android.support.v7.internal.view.menu.** { *; } + +-keep public class * extends android.support.v4.view.ActionProvider { + public (android.content.Context); +} + +# support v4 +-keep class android.support.v4.** { *; } +-keep interface android.support.v4.** { *; } + +# build config +-keep class com.example.BuildConfig { *; } + +# firebase +-keep class com.firebase.** { *; } + +# google gms +-keep class com.google.android.gms.** { *; } +-dontwarn com.google.android.gms.** + +# recycler view +-keep public class * extends android.support.v7.widget.RecyclerView$LayoutManager { + public (...); +} \ No newline at end of file diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml new file mode 100644 index 00000000..920b6850 --- /dev/null +++ b/app/src/main/AndroidManifest.xml @@ -0,0 +1,124 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/java/dev/lucasnlm/antimine/GameActivity.kt b/app/src/main/java/dev/lucasnlm/antimine/GameActivity.kt new file mode 100644 index 00000000..b1386ccc --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/GameActivity.kt @@ -0,0 +1,504 @@ +package dev.lucasnlm.antimine + +import android.content.ActivityNotFoundException +import android.content.Intent +import android.content.IntentSender +import android.net.Uri +import android.os.Build +import android.os.Bundle +import android.os.Handler +import android.text.format.DateUtils +import android.util.Log +import android.view.Menu +import android.view.MenuItem +import android.view.View +import androidx.appcompat.app.ActionBarDrawerToggle +import androidx.appcompat.app.AlertDialog +import androidx.core.content.ContextCompat +import androidx.core.os.HandlerCompat.postDelayed +import androidx.core.view.GravityCompat +import androidx.drawerlayout.widget.DrawerLayout +import androidx.fragment.app.FragmentTransaction +import androidx.lifecycle.Observer +import androidx.lifecycle.ViewModelProviders +import androidx.preference.PreferenceManager +import com.google.android.play.core.appupdate.AppUpdateManagerFactory +import com.google.android.play.core.install.model.AppUpdateType +import com.google.android.play.core.install.model.UpdateAvailability +import dev.lucasnlm.antimine.about.AboutActivity +import dev.lucasnlm.antimine.preferences.PreferencesActivity +import dagger.android.support.DaggerAppCompatActivity +import dev.lucasnlm.antimine.common.level.data.DifficultyPreset +import dev.lucasnlm.antimine.common.level.data.GameEvent +import dev.lucasnlm.antimine.common.level.data.GameStatus + +import dev.lucasnlm.antimine.common.level.viewmodel.GameViewModel +import dev.lucasnlm.antimine.common.level.viewmodel.GameViewModelFactory +import dev.lucasnlm.antimine.core.analytics.AnalyticsManager +import dev.lucasnlm.antimine.core.analytics.Event +import dev.lucasnlm.antimine.core.preferences.IPreferencesRepository +import dev.lucasnlm.antimine.core.utils.isDarkModeEnabled +import dev.lucasnlm.antimine.level.view.CustomLevelDialogFragment +import dev.lucasnlm.antimine.level.view.LevelFragment +import kotlinx.android.synthetic.main.activity_game.* +import kotlinx.coroutines.GlobalScope +import kotlinx.coroutines.launch +import javax.inject.Inject + +class GameActivity : DaggerAppCompatActivity() { + + @Inject + lateinit var viewModelFactory: GameViewModelFactory + + @Inject + lateinit var preferencesRepository: IPreferencesRepository + + @Inject + lateinit var analyticsManager: AnalyticsManager + + private lateinit var viewModel: GameViewModel + + private var gameStatus: GameStatus = GameStatus.PreGame + private var keepConfirmingNewGame = true + private val usingLargeArea by lazy { preferencesRepository.useLargeAreas() } + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_game) + viewModel = ViewModelProviders.of(this, viewModelFactory).get(GameViewModel::class.java) + bindViewModel() + + PreferenceManager.setDefaultValues(this, R.xml.preferences, false) + + bindToolbarAndDrawer() + + loadGameFragment() + + if (Build.VERSION.SDK_INT >= 21) { + checkUpdate() + } + + checkUseCount() + } + + private fun bindViewModel() = viewModel.apply { + eventObserver.observe(this@GameActivity, Observer { + onGameEvent(it) + }) + elapsedTimeSeconds.observe(this@GameActivity, Observer { + timer.apply { + visibility = if (it == 0L) View.GONE else View.VISIBLE + text = DateUtils.formatElapsedTime(it) + } + }) + mineCount.observe(this@GameActivity, Observer { + minesCount.apply { + visibility = View.VISIBLE + text = it.toString() + } + }) + difficulty.observe(this@GameActivity, Observer { + onChangeDifficulty(it) + }) + } + + override fun onBackPressed() { + when { + drawer.isDrawerOpen(GravityCompat.START) -> { + drawer.closeDrawer(GravityCompat.START) + viewModel.resumeGame() + } + gameStatus == GameStatus.Running -> showQuitConfirmation { + super.onBackPressed() + } + else -> super.onBackPressed() + } + } + + override fun onResume() { + super.onResume() + if (gameStatus == GameStatus.Running) { + viewModel.resumeGame() + analyticsManager.sentEvent(Event.Resume()) + } + + restartIfNeed() + } + + override fun onPause() { + super.onPause() + + if (gameStatus == GameStatus.Running) { + viewModel.pauseGame() + } + + analyticsManager.sentEvent(Event.Quit()) + } + + override fun onCreateOptionsMenu(menu: Menu): Boolean = + when (gameStatus) { + GameStatus.Over, GameStatus.Running -> { + menuInflater.inflate(R.menu.top_menu_over, menu) + true + } + else -> true + } + + override fun onOptionsItemSelected(item: MenuItem): Boolean { + return if (item.itemId == R.id.reset) { + + val confirmResign = gameStatus == GameStatus.Running + analyticsManager.sentEvent(Event.TapGameReset(confirmResign)) + + if (confirmResign) { + newGameConfirmation { + GlobalScope.launch { + viewModel.startNewGame() + } + } + } else { + GlobalScope.launch { + viewModel.startNewGame() + } + } + true + } else { + super.onOptionsItemSelected(item) + } + } + + private fun bindToolbarAndDrawer() { + setSupportActionBar(toolbar) + toolbar.title = "" + + supportActionBar?.apply { + title = "" + setDisplayHomeAsUpEnabled(true) + setHomeButtonEnabled(true) + } + + drawer.apply { + addDrawerListener( + ActionBarDrawerToggle(this@GameActivity, drawer, toolbar, R.string.open_menu, R.string.close_menu).apply { + if (!isDarkModeEnabled(applicationContext)) { + drawerArrowDrawable.color = ContextCompat.getColor(applicationContext, R.color.primary) + } + + syncState() + } + ) + + addDrawerListener(object : DrawerLayout.DrawerListener { + override fun onDrawerSlide(drawerView: View, slideOffset: Float) { + // Empty + } + + override fun onDrawerOpened(drawerView: View) { + if (gameStatus != GameStatus.Over) { + viewModel.pauseGame() + } + analyticsManager.sentEvent(Event.OpenDrawer()) + } + + override fun onDrawerClosed(drawerView: View) { + if (gameStatus != GameStatus.Over) { + viewModel.resumeGame() + } + analyticsManager.sentEvent(Event.CloseDrawer()) + } + + override fun onDrawerStateChanged(newState: Int) { + // Empty + } + }) + } + + navigationView.setNavigationItemSelectedListener { item -> + var handled = true + + when (item.itemId) { + R.id.standard -> changeDifficulty(DifficultyPreset.Standard) + R.id.beginner -> changeDifficulty(DifficultyPreset.Beginner) + R.id.intermediate -> changeDifficulty(DifficultyPreset.Intermediate) + R.id.expert -> changeDifficulty(DifficultyPreset.Expert) + R.id.custom -> showCustomLevelDialog() + R.id.about -> showAbout() + R.id.settings -> showSettings() + R.id.rate -> openRateUsLink("Drawer") + else -> handled = false + } + + if (handled) { + drawer.closeDrawer(GravityCompat.START) + } + + handled + } + + if (preferencesRepository.getBoolean(PREFERENCE_FIRST_USE, false)) { + drawer.openDrawer(GravityCompat.START) + preferencesRepository.putBoolean(PREFERENCE_FIRST_USE, true) + } + } + + private fun checkUseCount() { + val current = preferencesRepository.getInt(PREFERENCE_USE_COUNT, 0) + val shouldRequestRating = preferencesRepository.getBoolean(PREFERENCE_REQUEST_RATING, true) + + if (current >= 4 && shouldRequestRating) { + analyticsManager.sentEvent(Event.ShowRatingRequest(current)) + showRequestRating() + } + + preferencesRepository.putInt(PREFERENCE_USE_COUNT, current + 1) + } + + private fun onChangeDifficulty(difficulty: DifficultyPreset) { + navigationView.menu.apply { + arrayOf( + DifficultyPreset.Standard to findItem(R.id.standard), + DifficultyPreset.Beginner to findItem(R.id.beginner), + DifficultyPreset.Intermediate to findItem(R.id.intermediate), + DifficultyPreset.Expert to findItem(R.id.expert), + DifficultyPreset.Custom to findItem(R.id.custom) + ).map { + it.second to (if (it.first == difficulty) R.drawable.checked else R.drawable.unchecked) + }.forEach { (menuItem, icon) -> + menuItem.setIcon(icon) + } + } + } + + private fun loadGameFragment() { + val fragmentManager = supportFragmentManager + + fragmentManager.popBackStack() + + fragmentManager.findFragmentById(R.id.levelContainer)?.let { it -> + fragmentManager.beginTransaction().apply { + remove(it) + commitAllowingStateLoss() + } + } + + fragmentManager.beginTransaction().apply { + replace(R.id.levelContainer, LevelFragment()) + setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN) + commitAllowingStateLoss() + } + } + + private fun showRequestRating() { + if (getString(R.string.rating_message).isNotEmpty()) { + + AlertDialog.Builder(this) + .setMessage(R.string.rating_message) + .setPositiveButton(R.string.rating_button) { _, _ -> + openRateUsLink("Dialog") + } + .setNegativeButton(R.string.rating_button_no) { _, _ -> + preferencesRepository.putBoolean(PREFERENCE_REQUEST_RATING, false) + } + .show() + } + } + + private fun newGameConfirmation(action: () -> Unit) { + AlertDialog.Builder(this, R.style.MyDialog).apply { + setTitle(R.string.start_over) + setMessage(R.string.retry_sure) + setPositiveButton(R.string.resume) { _, _ -> action() } + setNegativeButton(R.string.cancel, null) + show() + } + } + + private fun showQuitConfirmation(action: () -> Unit) { + AlertDialog.Builder(this, R.style.MyDialog) + .setTitle(R.string.are_you_sure) + .setMessage(R.string.sure_quit_desc) + .setPositiveButton(R.string.quit) { _, _ -> action() } + .setNegativeButton(R.string.cancel, null) + .show() + } + + private fun showCustomLevelDialog() { + CustomLevelDialogFragment().apply { + show(supportFragmentManager, "custom_level_fragment") + } + } + + private fun showAbout() { + analyticsManager.sentEvent(Event.OpenAbout()) + Intent(this, AboutActivity::class.java).apply { + startActivity(this) + } + } + + private fun showSettings() { + analyticsManager.sentEvent(Event.OpenSettings()) + Intent(this, PreferencesActivity::class.java).apply { + startActivity(this) + } + } + + private fun showVictory() { + AlertDialog.Builder(this, R.style.MyDialog).apply { + setTitle(R.string.you_won) + setMessage(R.string.all_mines_disabled) + setCancelable(false) + setPositiveButton(R.string.new_game) { _, _ -> + GlobalScope.launch { + viewModel.startNewGame() + } + } + setNegativeButton(R.string.cancel, null) + show() + } + } + + private fun waitAndShowConfirmNewGame() { + if (keepConfirmingNewGame) { + postDelayed(Handler(), { + if (this.gameStatus == GameStatus.Over && !isFinishing) { + AlertDialog.Builder(this, R.style.MyDialog).apply { + setTitle(R.string.new_game) + setMessage(R.string.new_game_request) + setPositiveButton(R.string.yes) { _, _ -> + GlobalScope.launch { + viewModel.startNewGame() + } + } + setNegativeButton(R.string.cancel, null) + }.show() + + keepConfirmingNewGame = false + } + }, null, DateUtils.SECOND_IN_MILLIS) + } + } + + private fun waitAndShowGameOverConfirmNewGame() { + postDelayed(Handler(), { + if (this.gameStatus == GameStatus.Over && !isFinishing) { + AlertDialog.Builder(this, R.style.MyDialog).apply { + setTitle(R.string.you_lost) + setMessage(R.string.new_game_request) + setPositiveButton(R.string.yes) { _, _ -> + GlobalScope.launch { + viewModel.startNewGame() + } + } + setNegativeButton(R.string.cancel, null) + }.show() + } + }, null, DateUtils.SECOND_IN_MILLIS) + } + + private fun changeDifficulty(newDifficulty: DifficultyPreset) { + if (gameStatus == GameStatus.PreGame) { + GlobalScope.launch { + viewModel.startNewGame(newDifficulty) + } + } else { + newGameConfirmation { + GlobalScope.launch { + viewModel.startNewGame(newDifficulty) + } + } + } + } + + private fun onGameEvent(event: GameEvent) { + when (event) { + GameEvent.ResumeGame -> { + invalidateOptionsMenu() + } + GameEvent.StartNewGame -> { + gameStatus = GameStatus.PreGame + invalidateOptionsMenu() + } + GameEvent.Resume, GameEvent.Running -> { + gameStatus = GameStatus.Running + viewModel.runClock() + invalidateOptionsMenu() + } + GameEvent.Victory -> { + gameStatus = GameStatus.Over + viewModel.stopClock() + viewModel.revealAllEmptyAreas() + viewModel.victory() + invalidateOptionsMenu() + showVictory() + } + GameEvent.GameOver -> { + gameStatus = GameStatus.Over + invalidateOptionsMenu() + viewModel.stopClock() + viewModel.gameOver() + + waitAndShowGameOverConfirmNewGame() + } + GameEvent.ResumeVictory, GameEvent.ResumeGameOver -> { + gameStatus = GameStatus.Over + invalidateOptionsMenu() + viewModel.stopClock() + + waitAndShowConfirmNewGame() + } + else -> { + + } + } + } + + /** + * Call Google API to request update. + */ + private fun checkUpdate() { + val appUpdateManager = AppUpdateManagerFactory.create(this) + val appUpdateInfoTask = appUpdateManager.appUpdateInfo + + appUpdateInfoTask.addOnSuccessListener { info -> + if (info.updateAvailability() == UpdateAvailability.UPDATE_AVAILABLE + && info.isUpdateTypeAllowed(AppUpdateType.FLEXIBLE)) { + try { + appUpdateManager.startUpdateFlowForResult( + info, AppUpdateType.FLEXIBLE, this, 1) + } catch (e: IntentSender.SendIntentException) { + Log.e(TAG, "Fail to request update.") + } + } + } + + } + + /** + * If user change any accessibility preference, the game will restart the activity to + * apply these changes. + */ + private fun restartIfNeed() { + if (usingLargeArea != preferencesRepository.useLargeAreas()) { + finish() + Intent(this, GameActivity::class.java).run { startActivity(this) } + } + } + + private fun openRateUsLink(from: String) { + try { + startActivity(Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=$packageName"))) + } catch (e: ActivityNotFoundException) { + startActivity(Intent(Intent.ACTION_VIEW, Uri.parse("https://play.google.com/store/apps/details?id=$packageName"))) + } + + analyticsManager.sentEvent(Event.TapRatingRequest(from)) + preferencesRepository.putBoolean(PREFERENCE_REQUEST_RATING, false) + } + + companion object { + const val TAG = "GameActivity" + const val PREFERENCE_FIRST_USE = "preference_first_use" + const val PREFERENCE_USE_COUNT = "preference_use_count" + const val PREFERENCE_REQUEST_RATING = "preference_request_rating" + } +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/MainApplication.kt b/app/src/main/java/dev/lucasnlm/antimine/MainApplication.kt new file mode 100644 index 00000000..7097a1e3 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/MainApplication.kt @@ -0,0 +1,27 @@ +package dev.lucasnlm.antimine + +import dagger.android.AndroidInjector +import dagger.android.support.DaggerApplication +import dev.lucasnlm.antimine.core.analytics.AnalyticsManager +import dev.lucasnlm.antimine.core.analytics.Event +import dev.lucasnlm.antimine.di.AppModule +import dev.lucasnlm.antimine.di.DaggerAppComponent +import javax.inject.Inject + +class MainApplication : DaggerApplication() { + + @Inject + lateinit var analyticsManager: AnalyticsManager + + override fun applicationInjector(): AndroidInjector = + DaggerAppComponent.builder() + .application(this) + .appModule(AppModule(this)) + .build() + + override fun onCreate() { + super.onCreate() + analyticsManager.setup(applicationContext, mapOf()) + analyticsManager.sentEvent(Event.Open()) + } +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/TvGameActivity.kt b/app/src/main/java/dev/lucasnlm/antimine/TvGameActivity.kt new file mode 100644 index 00000000..3948c170 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/TvGameActivity.kt @@ -0,0 +1,318 @@ +package dev.lucasnlm.antimine + +import android.content.Intent +import android.content.IntentSender +import android.os.Build +import android.os.Bundle +import android.os.Handler +import android.text.format.DateUtils +import android.util.Log +import android.view.Menu +import android.view.MenuItem +import android.view.View +import androidx.appcompat.app.AlertDialog +import androidx.core.os.HandlerCompat +import androidx.fragment.app.FragmentTransaction +import androidx.lifecycle.Observer +import androidx.lifecycle.ViewModelProviders +import androidx.preference.PreferenceManager +import com.google.android.play.core.appupdate.AppUpdateManagerFactory +import com.google.android.play.core.install.model.AppUpdateType +import com.google.android.play.core.install.model.UpdateAvailability +import dagger.android.support.DaggerAppCompatActivity +import dev.lucasnlm.antimine.about.AboutActivity +import dev.lucasnlm.antimine.common.level.data.DifficultyPreset +import dev.lucasnlm.antimine.common.level.data.GameEvent +import dev.lucasnlm.antimine.common.level.data.GameStatus +import dev.lucasnlm.antimine.common.level.viewmodel.GameViewModel +import dev.lucasnlm.antimine.common.level.viewmodel.GameViewModelFactory +import dev.lucasnlm.antimine.core.preferences.IPreferencesRepository +import dev.lucasnlm.antimine.level.view.CustomLevelDialogFragment +import dev.lucasnlm.antimine.level.view.LevelFragment +import dev.lucasnlm.antimine.preferences.PreferencesActivity +import kotlinx.android.synthetic.main.activity_tv_game.* +import kotlinx.coroutines.GlobalScope +import kotlinx.coroutines.launch +import javax.inject.Inject + +class TvGameActivity : DaggerAppCompatActivity() { + + @Inject + lateinit var viewModelFactory: GameViewModelFactory + + @Inject + lateinit var preferencesRepository: IPreferencesRepository + + private lateinit var viewModel: GameViewModel + + private var gameStatus: GameStatus = GameStatus.PreGame + + private var keepConfirmingNewGame = true + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_tv_game) + viewModel = ViewModelProviders.of(this, viewModelFactory).get(GameViewModel::class.java) + bindViewModel() + + PreferenceManager.setDefaultValues(this, R.xml.preferences, false) + + loadGameFragment() + + if (Build.VERSION.SDK_INT >= 21) { + checkUpdate() + } + } + + private fun bindViewModel() = viewModel.apply { + eventObserver.observe(this@TvGameActivity, Observer { + onGameEvent(it) + }) + elapsedTimeSeconds.observe(this@TvGameActivity, Observer { + timer.apply { + visibility = if (it == 0L) View.GONE else View.VISIBLE + text = DateUtils.formatElapsedTime(it) + } + }) + mineCount.observe(this@TvGameActivity, Observer { + minesCount.apply { + visibility = View.VISIBLE + text = it.toString() + } + }) + difficulty.observe(this@TvGameActivity, Observer { + //onChangeDifficulty(it) + }) + } + + override fun onResume() { + super.onResume() + if (gameStatus == GameStatus.Running) { + viewModel.resumeGame() + } + } + + override fun onPause() { + super.onPause() + + if (gameStatus == GameStatus.Running) { + viewModel.pauseGame() + } + } + + override fun onCreateOptionsMenu(menu: Menu): Boolean = + when (gameStatus) { + GameStatus.Over, GameStatus.Running -> { + menuInflater.inflate(R.menu.top_menu_over, menu) + true + } + else -> true + } + + override fun onOptionsItemSelected(item: MenuItem): Boolean { + return if (item.itemId == R.id.reset) { + if (gameStatus == GameStatus.Running) { + newGameConfirmation { + GlobalScope.launch { + viewModel.startNewGame() + } + } + } else { + GlobalScope.launch { + viewModel.startNewGame() + } + } + true + } else { + super.onOptionsItemSelected(item) + } + } + + private fun loadGameFragment() { + val fragmentManager = supportFragmentManager + + fragmentManager.popBackStack() + + fragmentManager.findFragmentById(R.id.levelContainer)?.let { it -> + fragmentManager.beginTransaction().apply { + remove(it) + commitAllowingStateLoss() + } + } + + fragmentManager.beginTransaction().apply { + replace(R.id.levelContainer, LevelFragment()) + setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN) + commitAllowingStateLoss() + } + } + + private fun newGameConfirmation(action: () -> Unit) { + AlertDialog.Builder(this, R.style.MyDialog).apply { + setTitle(R.string.start_over) + setMessage(R.string.retry_sure) + setPositiveButton(R.string.resume) { _, _ -> action() } + setNegativeButton(R.string.cancel, null) + show() + } + } + + private fun showQuitConfirmation(action: () -> Unit) { + AlertDialog.Builder(this, R.style.MyDialog) + .setTitle(R.string.are_you_sure) + .setMessage(R.string.sure_quit_desc) + .setPositiveButton(R.string.quit) { _, _ -> action() } + .setNegativeButton(R.string.cancel, null) + .show() + } + + private fun showCustomLevelDialog() { + CustomLevelDialogFragment().apply { + show(supportFragmentManager, "custom_level_fragment") + } + } + + private fun showAbout() { + Intent(this, AboutActivity::class.java).apply { + startActivity(this) + } + } + + private fun showSettings() { + Intent(this, PreferencesActivity::class.java).apply { + startActivity(this) + } + } + + private fun showVictory() { + AlertDialog.Builder(this, R.style.MyDialog).apply { + setTitle(R.string.you_won) + setMessage(R.string.all_mines_disabled) + setCancelable(false) + setPositiveButton(R.string.new_game) { _, _ -> + GlobalScope.launch { + viewModel.startNewGame() + } + } + setNegativeButton(R.string.cancel, null) + show() + } + } + + private fun waitAndShowConfirmNewGame() { + if (keepConfirmingNewGame) { + HandlerCompat.postDelayed(Handler(), { + if (this.gameStatus == GameStatus.Over && !isFinishing) { + AlertDialog.Builder(this, R.style.MyDialog).apply { + setTitle(R.string.new_game) + setMessage(R.string.new_game_request) + setPositiveButton(R.string.yes) { _, _ -> + GlobalScope.launch { + viewModel.startNewGame() + } + } + setNegativeButton(R.string.cancel, null) + }.show() + + keepConfirmingNewGame = false + } + }, null, DateUtils.SECOND_IN_MILLIS) + } + } + + private fun waitAndShowGameOverConfirmNewGame() { + HandlerCompat.postDelayed(Handler(), { + if (this.gameStatus == GameStatus.Over && !isFinishing) { + AlertDialog.Builder(this, R.style.MyDialog).apply { + setTitle(R.string.you_lost) + setMessage(R.string.new_game_request) + setPositiveButton(R.string.yes) { _, _ -> + GlobalScope.launch { + viewModel.startNewGame() + } + } + setNegativeButton(R.string.cancel, null) + }.show() + } + }, null, DateUtils.SECOND_IN_MILLIS) + } + + private fun changeDifficulty(newDifficulty: DifficultyPreset) { + if (gameStatus == GameStatus.PreGame) { + GlobalScope.launch { + viewModel.startNewGame(newDifficulty) + } + } else { + newGameConfirmation { + GlobalScope.launch { + viewModel.startNewGame(newDifficulty) + } + } + } + } + + private fun onGameEvent(event: GameEvent) { + when (event) { + GameEvent.ResumeGame -> { + invalidateOptionsMenu() + } + GameEvent.StartNewGame -> { + gameStatus = GameStatus.PreGame + invalidateOptionsMenu() + } + GameEvent.Resume, GameEvent.Running -> { + gameStatus = GameStatus.Running + viewModel.runClock() + invalidateOptionsMenu() + } + GameEvent.Victory -> { + gameStatus = GameStatus.Over + viewModel.stopClock() + viewModel.revealAllEmptyAreas() + invalidateOptionsMenu() + showVictory() + } + GameEvent.GameOver -> { + gameStatus = GameStatus.Over + invalidateOptionsMenu() + viewModel.stopClock() + viewModel.gameOver() + + waitAndShowGameOverConfirmNewGame() + } + GameEvent.ResumeVictory, GameEvent.ResumeGameOver -> { + gameStatus = GameStatus.Over + invalidateOptionsMenu() + viewModel.stopClock() + + waitAndShowConfirmNewGame() + } + else -> { + + } + } + } + + private fun checkUpdate() { + val appUpdateManager = AppUpdateManagerFactory.create(this) + val appUpdateInfoTask = appUpdateManager.appUpdateInfo + + appUpdateInfoTask.addOnSuccessListener { info -> + if (info.updateAvailability() == UpdateAvailability.UPDATE_AVAILABLE + && info.isUpdateTypeAllowed(AppUpdateType.FLEXIBLE)) { + try { + appUpdateManager.startUpdateFlowForResult( + info, AppUpdateType.FLEXIBLE, this, 1) + } catch (e: IntentSender.SendIntentException) { + Log.e(TAG, "Fail to request update.") + } + } + } + + } + + companion object { + const val TAG = "GameActivity" + const val PREFERENCE_FIRST_USE = "preference_first_use" + } +} \ No newline at end of file diff --git a/app/src/main/java/dev/lucasnlm/antimine/about/AboutActivity.kt b/app/src/main/java/dev/lucasnlm/antimine/about/AboutActivity.kt new file mode 100644 index 00000000..a983e4b8 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/about/AboutActivity.kt @@ -0,0 +1,47 @@ +package dev.lucasnlm.antimine.about + +import android.content.Intent +import android.os.Bundle +import androidx.appcompat.app.AppCompatActivity +import android.view.MenuItem +import dev.lucasnlm.antimine.BuildConfig +import dev.lucasnlm.antimine.R + +import dev.lucasnlm.antimine.about.thirds.ThirdPartiesActivity +import kotlinx.android.synthetic.main.activity_about.* + +class AboutActivity : AppCompatActivity() { + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_about) + bindToolbar() + + version.text = getString( + R.string.version_s, + getString(R.string.app_name), BuildConfig.VERSION_NAME + ) + + thirdsParties.setOnClickListener { openThirdParties() } + } + + override fun onOptionsItemSelected(item: MenuItem): Boolean = + when (item.itemId) { + android.R.id.home -> { + onBackPressed() + true + } + else -> super.onOptionsItemSelected(item) + } + + private fun bindToolbar() { + supportActionBar?.let { actionBar -> + actionBar.setTitle(R.string.about) + actionBar.setDisplayHomeAsUpEnabled(true) + actionBar.setHomeButtonEnabled(true) + } + } + + private fun openThirdParties() { + startActivity(Intent(this, ThirdPartiesActivity::class.java)) + } +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/about/Constants.kt b/app/src/main/java/dev/lucasnlm/antimine/about/Constants.kt new file mode 100644 index 00000000..d3dad9a5 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/about/Constants.kt @@ -0,0 +1,6 @@ +package dev.lucasnlm.antimine.about + +object Constants { + const val TEXT_TITLE = "third_title" + const val TEXT_PATH = "third_path" +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/about/TextActivity.kt b/app/src/main/java/dev/lucasnlm/antimine/about/TextActivity.kt new file mode 100644 index 00000000..11bf0a7e --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/about/TextActivity.kt @@ -0,0 +1,97 @@ +package dev.lucasnlm.antimine.about + +import android.os.Bundle +import androidx.appcompat.app.AppCompatActivity +import android.util.Log +import android.view.MenuItem +import android.view.View +import dev.lucasnlm.antimine.R + +import kotlinx.android.synthetic.main.activity_text.* +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.GlobalScope +import kotlinx.coroutines.launch +import kotlinx.coroutines.withContext + +import java.io.ByteArrayOutputStream +import java.io.IOException +import java.io.InputStream + +class TextActivity : AppCompatActivity() { + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_text) + bindToolbar() + + progressBar.isIndeterminate = true + + GlobalScope.launch { + withContext(Dispatchers.Main) { + progressBar.visibility = View.VISIBLE + } + + withContext(Dispatchers.IO) { + val rawPath = intent.getIntExtra(Constants.TEXT_PATH, -1) + var result: String? = null + + if (rawPath > 0) { + resources.openRawResource(rawPath).use { inputStream -> + + result = readTextFile(inputStream) + } + } + + withContext(Dispatchers.Main) { + textView.text = result + progressBar.visibility = View.GONE + } + } + } + } + + private fun readTextFile(inputStream: InputStream): String { + var result = "" + ByteArrayOutputStream().use { outputStream -> + val buf = ByteArray(4096) + var len: Int + try { + while (true) { + len = inputStream.read(buf) + if (len != -1) { + outputStream.write(buf, 0, len) + } else { + break + } + } + } catch (e: IOException) { + Log.e(TAG, "Fail to read file.", e) + } + result = outputStream.toString() + } + return result + } + + override fun onOptionsItemSelected(item: MenuItem): Boolean { + var handled = false + + if (item.itemId == android.R.id.home) { + onBackPressed() + handled = true + } + + return handled || super.onOptionsItemSelected(item) + } + + private fun bindToolbar() { + supportActionBar?.apply { + title = intent.getStringExtra(Constants.TEXT_TITLE) + setDisplayHomeAsUpEnabled(true) + setHomeButtonEnabled(true) + } + } + + companion object { + private const val TAG = "TextActivity" + } +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/about/thirds/ThirdPartiesActivity.kt b/app/src/main/java/dev/lucasnlm/antimine/about/thirds/ThirdPartiesActivity.kt new file mode 100644 index 00000000..ff348710 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/about/thirds/ThirdPartiesActivity.kt @@ -0,0 +1,78 @@ +package dev.lucasnlm.antimine.about.thirds + +import android.os.Bundle +import androidx.appcompat.app.AppCompatActivity +import android.view.MenuItem +import androidx.recyclerview.widget.DividerItemDecoration +import androidx.recyclerview.widget.LinearLayoutManager + +import dev.lucasnlm.antimine.R +import dev.lucasnlm.antimine.about.thirds.data.ThirdParty +import dev.lucasnlm.antimine.about.thirds.view.ThirdPartyAdapter +import kotlinx.android.synthetic.main.activity_third_party.* + +class ThirdPartiesActivity : AppCompatActivity() { + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_third_party) + bindToolbar() + + licenses.apply { + setHasFixedSize(true) + addItemDecoration( + DividerItemDecoration(context, DividerItemDecoration.VERTICAL) + ) + layoutManager = LinearLayoutManager(context) + } + + loadLicenses() + } + + private fun loadLicenses() { + licenses.adapter = ThirdPartyAdapter( + listOf( + ThirdParty( + "Android SDK License", + R.raw.android_sdk + ), + ThirdParty( + "Material Design Icons", + R.raw.apache2 + ), + ThirdParty( + "Dagger", + R.raw.apache2 + ), + ThirdParty( + "Moshi", + R.raw.apache2 + ), + ThirdParty( + "Mockito", + R.raw.mockito + ), + ThirdParty( + "Sounds", + R.raw.sounds + ) + ) + ) + } + + override fun onOptionsItemSelected(item: MenuItem): Boolean = + when (item.itemId) { + android.R.id.home -> { + onBackPressed() + true + } + else -> super.onOptionsItemSelected(item) + } + + private fun bindToolbar() { + supportActionBar?.apply { + setTitle(R.string.licenses) + setDisplayHomeAsUpEnabled(true) + setHomeButtonEnabled(true) + } + } +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/about/thirds/data/ThirdParty.kt b/app/src/main/java/dev/lucasnlm/antimine/about/thirds/data/ThirdParty.kt new file mode 100644 index 00000000..9cd03818 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/about/thirds/data/ThirdParty.kt @@ -0,0 +1,8 @@ +package dev.lucasnlm.antimine.about.thirds.data + +import androidx.annotation.RawRes + +internal data class ThirdParty( + val name: String, + @RawRes val license: Int +) diff --git a/app/src/main/java/dev/lucasnlm/antimine/about/thirds/view/ThirdPartyAdapter.kt b/app/src/main/java/dev/lucasnlm/antimine/about/thirds/view/ThirdPartyAdapter.kt new file mode 100644 index 00000000..3bb7fd77 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/about/thirds/view/ThirdPartyAdapter.kt @@ -0,0 +1,37 @@ +package dev.lucasnlm.antimine.about.thirds.view + +import android.content.Intent +import androidx.recyclerview.widget.RecyclerView +import android.view.LayoutInflater +import android.view.ViewGroup +import dev.lucasnlm.antimine.R + +import dev.lucasnlm.antimine.about.Constants +import dev.lucasnlm.antimine.about.TextActivity +import dev.lucasnlm.antimine.about.thirds.data.ThirdParty + +internal class ThirdPartyAdapter( + private val thirdParties: List +) : RecyclerView.Adapter() { + + override fun getItemCount(): Int = thirdParties.size + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ThirdPartyItemHolder { + val view = LayoutInflater.from(parent.context) + .inflate(R.layout.view_third_party, parent, false) + return ThirdPartyItemHolder(view) + } + + override fun onBindViewHolder(holder: ThirdPartyItemHolder, position: Int) { + val thirdParty = thirdParties[position] + holder.title.text = thirdParty.name + holder.itemView.setOnClickListener { view -> + val intent = Intent(view.context, TextActivity::class.java).apply { + putExtra(Constants.TEXT_TITLE, thirdParty.name) + putExtra(Constants.TEXT_PATH, thirdParty.license) + } + + view.context.startActivity(intent) + } + } +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/about/thirds/view/ThirdPartyItemHolder.kt b/app/src/main/java/dev/lucasnlm/antimine/about/thirds/view/ThirdPartyItemHolder.kt new file mode 100644 index 00000000..47769e2b --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/about/thirds/view/ThirdPartyItemHolder.kt @@ -0,0 +1,11 @@ +package dev.lucasnlm.antimine.about.thirds.view + +import androidx.recyclerview.widget.RecyclerView +import android.view.View +import android.widget.TextView + +import dev.lucasnlm.antimine.R + +internal class ThirdPartyItemHolder(view: View) : RecyclerView.ViewHolder(view) { + val title: TextView = view.findViewById(R.id.third_name) +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/di/ActivityModule.kt b/app/src/main/java/dev/lucasnlm/antimine/di/ActivityModule.kt new file mode 100644 index 00000000..5eb4c98a --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/di/ActivityModule.kt @@ -0,0 +1,18 @@ +package dev.lucasnlm.antimine.di + +import dev.lucasnlm.antimine.GameActivity +import dagger.Module +import dagger.android.ContributesAndroidInjector +import dev.lucasnlm.antimine.TvGameActivity +import dev.lucasnlm.antimine.core.scope.ActivityScope + +@Module +interface ActivityModule { + @ActivityScope + @ContributesAndroidInjector + fun contributeGameActivityInjector(): GameActivity + + @ActivityScope + @ContributesAndroidInjector + fun contributeTvGameActivityInjector(): TvGameActivity +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/di/AppComponent.kt b/app/src/main/java/dev/lucasnlm/antimine/di/AppComponent.kt new file mode 100644 index 00000000..cceaaeaa --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/di/AppComponent.kt @@ -0,0 +1,35 @@ +package dev.lucasnlm.antimine.di + +import android.app.Application +import dagger.BindsInstance +import dagger.Component +import dagger.android.AndroidInjector +import dagger.android.support.AndroidSupportInjectionModule +import dev.lucasnlm.antimine.MainApplication +import dev.lucasnlm.antimine.common.level.di.LevelModule +import dev.lucasnlm.antimine.core.di.CommonModule +import javax.inject.Singleton + +@Component( + modules = [ + AndroidSupportInjectionModule::class, + AppModule::class, + LevelModule::class, + ActivityModule::class, + FragmentModule::class, + CommonModule::class + ] +) +@Singleton +interface AppComponent : AndroidInjector { + + @Component.Builder + interface Builder { + @BindsInstance + fun application(application: Application): Builder + + fun appModule(module: AppModule): Builder + + fun build(): AppComponent + } +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/di/AppModule.kt b/app/src/main/java/dev/lucasnlm/antimine/di/AppModule.kt new file mode 100644 index 00000000..4c46d22e --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/di/AppModule.kt @@ -0,0 +1,14 @@ +package dev.lucasnlm.antimine.di + +import android.app.Application +import android.content.Context +import dagger.Module +import dagger.Provides + +@Module +class AppModule( + private val application: Application +) { + @Provides + fun provideContext(): Context = application.applicationContext +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/di/FragmentModule.kt b/app/src/main/java/dev/lucasnlm/antimine/di/FragmentModule.kt new file mode 100644 index 00000000..73d48312 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/di/FragmentModule.kt @@ -0,0 +1,18 @@ +package dev.lucasnlm.antimine.di + +import dev.lucasnlm.antimine.level.view.CustomLevelDialogFragment +import dagger.Module +import dagger.android.ContributesAndroidInjector +import dev.lucasnlm.antimine.core.scope.ActivityScope +import dev.lucasnlm.antimine.level.view.LevelFragment + +@Module +interface FragmentModule { + @ActivityScope + @ContributesAndroidInjector + fun contributeLevelFragmentInjector(): LevelFragment + + @ActivityScope + @ContributesAndroidInjector + fun contributeCustomLevelDialogFragmentInjector(): CustomLevelDialogFragment +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/level/view/CustomLevelDialogFragment.kt b/app/src/main/java/dev/lucasnlm/antimine/level/view/CustomLevelDialogFragment.kt new file mode 100644 index 00000000..61833f85 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/level/view/CustomLevelDialogFragment.kt @@ -0,0 +1,88 @@ +package dev.lucasnlm.antimine.level.view + +import android.app.Dialog +import android.os.Bundle +import androidx.appcompat.app.AlertDialog +import android.widget.TextView +import androidx.lifecycle.ViewModelProviders + +import dagger.android.support.DaggerAppCompatDialogFragment +import dev.lucasnlm.antimine.R +import dev.lucasnlm.antimine.common.level.data.DifficultyPreset +import dev.lucasnlm.antimine.common.level.data.LevelSetup +import dev.lucasnlm.antimine.common.level.viewmodel.GameViewModel +import dev.lucasnlm.antimine.common.level.viewmodel.GameViewModelFactory +import dev.lucasnlm.antimine.core.preferences.IPreferencesRepository +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.GlobalScope +import kotlinx.coroutines.launch +import javax.inject.Inject + +class CustomLevelDialogFragment : DaggerAppCompatDialogFragment() { + @Inject + lateinit var viewModelFactory: GameViewModelFactory + + @Inject + lateinit var preferencesRepository: IPreferencesRepository + + private lateinit var viewModel: GameViewModel + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + + activity?.let { + viewModel = ViewModelProviders.of(it, viewModelFactory).get(GameViewModel::class.java) + } + } + + private fun filterInput(target: String, min: Int): Int { + var result = min + + try { + result = Integer.valueOf(target) + } catch (e: NumberFormatException) { + result = min + } finally { + result = result.coerceAtLeast(min) + } + + return result + } + + override fun onCreateDialog(savedInstanceState: Bundle?): Dialog { + return AlertDialog.Builder(context!!, R.style.MyDialog).apply { + setTitle(R.string.new_game) + setView(R.layout.dialog_custom_game) + setNegativeButton(R.string.cancel, null) + setPositiveButton(R.string.start) { _, _ -> + val mapWidth: TextView? = dialog?.findViewById(R.id.map_width) + val mapHeight: TextView? = dialog?.findViewById(R.id.map_height) + val mapMines: TextView? = dialog?.findViewById(R.id.map_mines) + + var width = filterInput(mapWidth?.text.toString(), MIN_WIDTH) + var height = filterInput(mapHeight?.text.toString(), MIN_HEIGHT) + var mines = filterInput(mapMines?.text.toString(), MIN_MINES) + + if (width * height - 1 < mines) { + mines = width * height - 1 + } + + width = width.coerceAtMost(50) + height = height.coerceAtMost(50) + mines = mines.coerceAtLeast(1) + + preferencesRepository.updateCustomGameMode(LevelSetup(width, height, mines)) + + GlobalScope.launch(Dispatchers.IO) { + viewModel.startNewGame(DifficultyPreset.Custom) + } + } + }.create() + } + + companion object { + const val MIN_WIDTH = 5 + const val MIN_HEIGHT = 5 + const val MIN_MINES = 3 + } +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/level/view/LevelFragment.kt b/app/src/main/java/dev/lucasnlm/antimine/level/view/LevelFragment.kt new file mode 100644 index 00000000..2c1c0df9 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/level/view/LevelFragment.kt @@ -0,0 +1,126 @@ +package dev.lucasnlm.antimine.level.view + +import android.os.Bundle +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import androidx.lifecycle.Observer +import androidx.lifecycle.ViewModelProviders +import androidx.recyclerview.widget.GridLayoutManager +import androidx.recyclerview.widget.RecyclerView +import dev.lucasnlm.antimine.common.R +import dev.lucasnlm.antimine.common.level.view.UnlockedHorizontalScrollView +import dagger.android.support.DaggerFragment +import dev.lucasnlm.antimine.common.level.data.DifficultyPreset +import dev.lucasnlm.antimine.common.level.view.AreaAdapter +import dev.lucasnlm.antimine.common.level.viewmodel.GameViewModel +import dev.lucasnlm.antimine.common.level.viewmodel.GameViewModelFactory +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.GlobalScope +import kotlinx.coroutines.launch +import kotlinx.coroutines.withContext +import javax.inject.Inject + +open class LevelFragment : DaggerFragment() { + @Inject + lateinit var viewModelFactory: GameViewModelFactory + + private lateinit var viewModel: GameViewModel + private lateinit var recyclerGrid: RecyclerView + private lateinit var bidirectionalScroll: UnlockedHorizontalScrollView + private lateinit var areaAdapter: AreaAdapter + + override fun onCreateView( + inflater: LayoutInflater, + container: ViewGroup?, + savedInstanceState: Bundle? + ): View? = + inflater.inflate(R.layout.fragment_level, container, false) + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + + activity?.let { + viewModel = ViewModelProviders.of(it, viewModelFactory).get(GameViewModel::class.java) + areaAdapter = AreaAdapter(it.applicationContext, viewModel) + } + } + + override fun onPause() { + super.onPause() + + GlobalScope.launch { + viewModel.saveGame() + } + } + + override fun onViewCreated(view: View, savedInstanceState: Bundle?) { + super.onViewCreated(view, savedInstanceState) + + recyclerGrid = view.findViewById(R.id.recyclerGrid) + recyclerGrid.apply { + setHasFixedSize(true) + isNestedScrollingEnabled = false + adapter = areaAdapter + alpha = 0.0f + } + + bidirectionalScroll = view.findViewById(R.id.bidirectionalScroll) + bidirectionalScroll.setTarget(recyclerGrid) + + GlobalScope.launch { + val levelSetup = viewModel.onCreate(handleNewGameDeeplink()) + + val width = levelSetup.width + + withContext(Dispatchers.Main) { + recyclerGrid.layoutManager = + GridLayoutManager(activity, width, RecyclerView.VERTICAL, false) + + view.post { + recyclerGrid.scrollBy(0, recyclerGrid.height / 2) + bidirectionalScroll.scrollBy(recyclerGrid.width / 4, 0) + recyclerGrid.animate().apply { + alpha(1.0f) + duration = 1000 + }.start() + } + } + } + + viewModel.run { + field.observe(viewLifecycleOwner, Observer { + areaAdapter.bindField(it) + }) + levelSetup.observe(viewLifecycleOwner, Observer { + recyclerGrid.layoutManager = + GridLayoutManager(activity, it.width, RecyclerView.VERTICAL, false) + }) + fieldRefresh.observe(viewLifecycleOwner, Observer { + areaAdapter.notifyItemChanged(it) + }) + } + } + + private fun handleNewGameDeeplink(): DifficultyPreset? { + var result: DifficultyPreset? = null + + activity?.intent?.data?.let { uri -> + if (uri.scheme == DEFAULT_SCHEME) { + result = when (uri.schemeSpecificPart.removePrefix("//new-game/")) { + "beginner" -> DifficultyPreset.Beginner + "intermediate" -> DifficultyPreset.Intermediate + "expert" -> DifficultyPreset.Expert + "standard" -> DifficultyPreset.Standard + else -> null + } + } + } + + return result + } + + companion object { + const val DEFAULT_SCHEME = "antimine" + } +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/preferences/PreferencesActivity.kt b/app/src/main/java/dev/lucasnlm/antimine/preferences/PreferencesActivity.kt new file mode 100644 index 00000000..6ec4e812 --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/preferences/PreferencesActivity.kt @@ -0,0 +1,29 @@ +package dev.lucasnlm.antimine.preferences + +import android.os.Bundle +import androidx.appcompat.app.AppCompatActivity +import androidx.preference.PreferenceFragmentCompat +import androidx.preference.PreferenceManager + +import dev.lucasnlm.antimine.R + +class PreferencesActivity : AppCompatActivity() { + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + PreferenceManager.setDefaultValues(this, R.xml.preferences, false) + + // Load the preferences from an XML resource + supportFragmentManager + .beginTransaction() + .replace(android.R.id.content, PrefsFragment()) + .commitAllowingStateLoss() + } + + class PrefsFragment : PreferenceFragmentCompat() { + + override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) { + // Load the preferences from an XML resource + addPreferencesFromResource(R.xml.preferences) + } + } +} diff --git a/app/src/main/java/dev/lucasnlm/antimine/splash/SplashActivity.kt b/app/src/main/java/dev/lucasnlm/antimine/splash/SplashActivity.kt new file mode 100644 index 00000000..bd56b95c --- /dev/null +++ b/app/src/main/java/dev/lucasnlm/antimine/splash/SplashActivity.kt @@ -0,0 +1,24 @@ +package dev.lucasnlm.antimine.splash + +import android.app.UiModeManager +import android.content.Intent +import android.content.res.Configuration +import android.os.Bundle +import androidx.appcompat.app.AppCompatActivity +import dev.lucasnlm.antimine.GameActivity +import dev.lucasnlm.antimine.TvGameActivity + +class SplashActivity : AppCompatActivity() { + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + + val uiModeManager: UiModeManager = getSystemService(UI_MODE_SERVICE) as UiModeManager + if (uiModeManager.currentModeType == Configuration.UI_MODE_TYPE_TELEVISION) { + Intent(this, TvGameActivity::class.java).run { startActivity(this) } + } else { + Intent(this, GameActivity::class.java).run { startActivity(this) } + } + + finish() + } +} diff --git a/app/src/main/res/drawable/mine_white.xml b/app/src/main/res/drawable/mine_white.xml new file mode 100644 index 00000000..134dcbe0 --- /dev/null +++ b/app/src/main/res/drawable/mine_white.xml @@ -0,0 +1,10 @@ + + + diff --git a/app/src/main/res/layout/activity_about.xml b/app/src/main/res/layout/activity_about.xml new file mode 100644 index 00000000..95783eab --- /dev/null +++ b/app/src/main/res/layout/activity_about.xml @@ -0,0 +1,59 @@ + + + + + + + + + + diff --git a/app/src/main/res/layout/activity_game.xml b/app/src/main/res/layout/activity_game.xml new file mode 100644 index 00000000..0efdb71c --- /dev/null +++ b/app/src/main/res/layout/activity_game.xml @@ -0,0 +1,104 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/layout/activity_text.xml b/app/src/main/res/layout/activity_text.xml new file mode 100644 index 00000000..fc15d49c --- /dev/null +++ b/app/src/main/res/layout/activity_text.xml @@ -0,0 +1,39 @@ + + + + + + + + + + + + + + diff --git a/app/src/main/res/layout/activity_third_party.xml b/app/src/main/res/layout/activity_third_party.xml new file mode 100644 index 00000000..24502a09 --- /dev/null +++ b/app/src/main/res/layout/activity_third_party.xml @@ -0,0 +1,31 @@ + + + + + + + + diff --git a/app/src/main/res/layout/activity_tv_game.xml b/app/src/main/res/layout/activity_tv_game.xml new file mode 100644 index 00000000..0a28d7a6 --- /dev/null +++ b/app/src/main/res/layout/activity_tv_game.xml @@ -0,0 +1,74 @@ + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/layout/dialog_custom_game.xml b/app/src/main/res/layout/dialog_custom_game.xml new file mode 100644 index 00000000..471c2c05 --- /dev/null +++ b/app/src/main/res/layout/dialog_custom_game.xml @@ -0,0 +1,102 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/layout/view_third_party.xml b/app/src/main/res/layout/view_third_party.xml new file mode 100644 index 00000000..70e1c29d --- /dev/null +++ b/app/src/main/res/layout/view_third_party.xml @@ -0,0 +1,21 @@ + + + + + + diff --git a/app/src/main/res/menu/top_menu_over.xml b/app/src/main/res/menu/top_menu_over.xml new file mode 100644 index 00000000..139eef97 --- /dev/null +++ b/app/src/main/res/menu/top_menu_over.xml @@ -0,0 +1,13 @@ + +

+ + + diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml new file mode 100644 index 00000000..31e31f84 --- /dev/null +++ b/app/src/main/res/values/strings.xml @@ -0,0 +1,4 @@ + + + 00:00 + \ No newline at end of file diff --git a/app/src/main/res/xml/backup_rules.xml b/app/src/main/res/xml/backup_rules.xml new file mode 100644 index 00000000..ff836a11 --- /dev/null +++ b/app/src/main/res/xml/backup_rules.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/app/src/main/res/xml/shortcuts.xml b/app/src/main/res/xml/shortcuts.xml new file mode 100644 index 00000000..b1c70720 --- /dev/null +++ b/app/src/main/res/xml/shortcuts.xml @@ -0,0 +1,52 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/build.gradle b/build.gradle new file mode 100644 index 00000000..4b5d57ba --- /dev/null +++ b/build.gradle @@ -0,0 +1,40 @@ +buildscript { + ext.kotlin_version = '1.3.50' + + repositories { + mavenCentral() + jcenter() + google() + + maven { + url 'https://maven.fabric.io/public' + name 'Fabric' + } + } + + dependencies { + classpath 'com.android.tools.build:gradle:3.5.3' + + // noinspection GradleDynamicVersion + classpath 'io.fabric.tools:gradle:1.+' + classpath 'org.jetbrains.kotlin:kotlin-gradle-plugin:1.3.61' + + // NOTE: Do not place your application dependencies here; they belong + // in the individual module build.gradle files + classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" } +} + +allprojects { + repositories { + google() + jcenter() + + maven { + url 'https://jitpack.io' + } + } +} + +task clean(type: Delete) { + delete rootProject.buildDir +} diff --git a/common/.gitignore b/common/.gitignore new file mode 100644 index 00000000..796b96d1 --- /dev/null +++ b/common/.gitignore @@ -0,0 +1 @@ +/build diff --git a/common/build.gradle b/common/build.gradle new file mode 100644 index 00000000..057e92a3 --- /dev/null +++ b/common/build.gradle @@ -0,0 +1,68 @@ +apply plugin: 'com.android.library' +apply plugin: 'kotlin-android' +apply plugin: 'kotlin-android-extensions' +apply plugin: 'kotlin-kapt' + +android { + compileSdkVersion 29 + + defaultConfig { + minSdkVersion 14 + targetSdkVersion 29 + versionCode 19 + versionName '3.0' + } + + kapt { + generateStubs true + } +} + +dependencies { + implementation fileTree(dir: 'libs', include: ['*.jar']) + + // AndroidX + implementation 'androidx.appcompat:appcompat:1.1.0' + implementation 'androidx.preference:preference:1.1.0' + implementation 'androidx.recyclerview:recyclerview:1.1.0' + implementation 'androidx.constraintlayout:constraintlayout:1.1.3' + + // Lifecycle + api 'android.arch.lifecycle:extensions:1.1.1' + implementation "android.arch.lifecycle:viewmodel:1.1.1" + + // Amplitude + implementation 'com.amplitude:android-sdk:2.23.2' + + // Dagger + def dagger_version='2.21' + api "com.google.dagger:dagger-android:$dagger_version" + api "com.google.dagger:dagger-android-support:$dagger_version" + kapt "com.google.dagger:dagger-android-processor:$dagger_version" + kapt "com.google.dagger:dagger-compiler:$dagger_version" + + // Room + def room_version = '2.2.3' + api "androidx.room:room-runtime:$room_version" + api "androidx.room:room-ktx:$room_version" + kapt "androidx.room:room-compiler:$room_version" + testImplementation "androidx.room:room-testing:$room_version" + + // Moshi - Json + def moshi_version = '1.9.1' + api "com.squareup.moshi:moshi:$moshi_version" + api "com.squareup.moshi:moshi-kotlin:$moshi_version" + kapt "com.squareup.moshi:moshi-kotlin-codegen:$moshi_version" + + // Coroutines + def coroutines_version = '1.3.0' + implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutines_version" + implementation "org.jetbrains.kotlinx:kotlinx-coroutines-android:$coroutines_version" + + implementation 'org.jetbrains.kotlin:kotlin-stdlib-jdk7:1.3.50' + + testImplementation 'junit:junit:4.12' + testImplementation 'org.mockito:mockito-core:2.24.0' + testImplementation 'com.nhaarman.mockitokotlin2:mockito-kotlin:2.1.0' + +} diff --git a/common/gradle/wrapper/gradle-wrapper.jar b/common/gradle/wrapper/gradle-wrapper.jar new file mode 100644 index 0000000000000000000000000000000000000000..7a3265ee94c0ab25cf079ac8ccdf87f41d455d42 GIT binary patch literal 54708 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

_x$+l{{cH8$W#CT literal 0 HcmV?d00001 diff --git a/common/gradle/wrapper/gradle-wrapper.properties b/common/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 00000000..905919ca --- /dev/null +++ b/common/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,6 @@ +#Mon May 21 23:30:42 BRT 2018 +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +zipStoreBase=GRADLE_USER_HOME +zipStorePath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-4.4-all.zip diff --git a/common/gradlew b/common/gradlew new file mode 100644 index 00000000..cccdd3d5 --- /dev/null +++ b/common/gradlew @@ -0,0 +1,172 @@ +#!/usr/bin/env sh + +############################################################################## +## +## Gradle start up script for UN*X +## +############################################################################## + +# Attempt to set APP_HOME +# Resolve links: $0 may be a link +PRG="$0" +# Need this for relative symlinks. +while [ -h "$PRG" ] ; do + ls=`ls -ld "$PRG"` + link=`expr "$ls" : '.*-> \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG=`dirname "$PRG"`"/$link" + fi +done +SAVED="`pwd`" +cd "`dirname \"$PRG\"`/" >/dev/null +APP_HOME="`pwd -P`" +cd "$SAVED" >/dev/null + +APP_NAME="Gradle" +APP_BASE_NAME=`basename "$0"` + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS="" + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD="maximum" + +warn () { + echo "$*" +} + +die () { + echo + echo "$*" + echo + exit 1 +} + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +nonstop=false +case "`uname`" in + CYGWIN* ) + cygwin=true + ;; + Darwin* ) + darwin=true + ;; + MINGW* ) + msys=true + ;; + NONSTOP* ) + nonstop=true + ;; +esac + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD="java" + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." +fi + +# Increase the maximum file descriptors if we can. +if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then + MAX_FD_LIMIT=`ulimit -H -n` + if [ $? -eq 0 ] ; then + if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then + MAX_FD="$MAX_FD_LIMIT" + fi + ulimit -n $MAX_FD + if [ $? -ne 0 ] ; then + warn "Could not set maximum file descriptor limit: $MAX_FD" + fi + else + warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" + fi +fi + +# For Darwin, add options to specify how the application appears in the dock +if $darwin; then + GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" +fi + +# For Cygwin, switch paths to Windows format before running java +if $cygwin ; then + APP_HOME=`cygpath --path --mixed "$APP_HOME"` + CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` + JAVACMD=`cygpath --unix "$JAVACMD"` + + # We build the pattern for arguments to be converted via cygpath + ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` + SEP="" + for dir in $ROOTDIRSRAW ; do + ROOTDIRS="$ROOTDIRS$SEP$dir" + SEP="|" + done + OURCYGPATTERN="(^($ROOTDIRS))" + # Add a user-defined pattern to the cygpath arguments + if [ "$GRADLE_CYGPATTERN" != "" ] ; then + OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" + fi + # Now convert the arguments - kludge to limit ourselves to /bin/sh + i=0 + for arg in "$@" ; do + CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` + CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option + + if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition + eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` + else + eval `echo args$i`="\"$arg\"" + fi + i=$((i+1)) + done + case $i in + (0) set -- ;; + (1) set -- "$args0" ;; + (2) set -- "$args0" "$args1" ;; + (3) set -- "$args0" "$args1" "$args2" ;; + (4) set -- "$args0" "$args1" "$args2" "$args3" ;; + (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; + (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; + (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; + (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; + (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; + esac +fi + +# Escape application args +save () { + for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done + echo " " +} +APP_ARGS=$(save "$@") + +# Collect all arguments for the java command, following the shell quoting and substitution rules +eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" + +# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong +if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then + cd "$(dirname "$0")" +fi + +exec "$JAVACMD" "$@" diff --git a/common/gradlew.bat b/common/gradlew.bat new file mode 100644 index 00000000..e95643d6 --- /dev/null +++ b/common/gradlew.bat @@ -0,0 +1,84 @@ +@if "%DEBUG%" == "" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +set DIRNAME=%~dp0 +if "%DIRNAME%" == "" set DIRNAME=. +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS= + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if "%ERRORLEVEL%" == "0" goto init + +echo. +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto init + +echo. +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:init +@rem Get command-line arguments, handling Windows variants + +if not "%OS%" == "Windows_NT" goto win9xME_args + +:win9xME_args +@rem Slurp the command line arguments. +set CMD_LINE_ARGS= +set _SKIP=2 + +:win9xME_args_slurp +if "x%~1" == "x" goto execute + +set CMD_LINE_ARGS=%* + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% + +:end +@rem End local scope for the variables with windows NT shell +if "%ERRORLEVEL%"=="0" goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 +exit /b 1 + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/common/proguard-rules.pro b/common/proguard-rules.pro new file mode 100644 index 00000000..550dae22 --- /dev/null +++ b/common/proguard-rules.pro @@ -0,0 +1,47 @@ +# Project specific ProGuard rules + +# Crashlytics 2.+ +-keep class com.crashlytics.** { *; } +-keep class com.crashlytics.android.** +-keepattributes SourceFile, LineNumberTable, *Annotation* + +# For Fabric to properly de-obfuscate your crash reports, you need to remove this line from your ProGuard config: +-printmapping mapping.txt + +# support design +-dontwarn android.support.design.** +-keep class android.support.design.** { *; } +-keep interface android.support.design.** { *; } +-keep public class android.support.design.R$* { *; } + +# support v7 +-keep public class android.support.v7.widget.** { *; } +-keep public class android.support.v7.internal.widget.** { *; } +-keep public class android.support.v7.internal.view.menu.** { *; } + +-keep public class * extends android.support.v4.view.ActionProvider { + public (android.content.Context); +} + +# support v4 +-keep class android.support.v4.** { *; } +-keep interface android.support.v4.** { *; } + +# build config +-keep class com.example.BuildConfig { *; } + +# firebase +-keep class com.firebase.** { *; } + +# google gms +-keep class com.google.android.gms.** { *; } +-dontwarn com.google.android.gms.** + +# amplitude +-keep class com.google.android.gms.ads.** { *; } +-dontwarn okio.** + +# recycler view +-keep public class * extends android.support.v7.widget.RecyclerView$LayoutManager { + public (...); +} \ No newline at end of file diff --git a/common/src/main/AndroidManifest.xml b/common/src/main/AndroidManifest.xml new file mode 100644 index 00000000..7086a2af --- /dev/null +++ b/common/src/main/AndroidManifest.xml @@ -0,0 +1,9 @@ + + + + + + + + diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/GameModeFactory.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/GameModeFactory.kt new file mode 100644 index 00000000..72d05a82 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/GameModeFactory.kt @@ -0,0 +1,41 @@ +package dev.lucasnlm.antimine.common.level + +import dev.lucasnlm.antimine.common.level.data.DifficultyPreset +import dev.lucasnlm.antimine.common.level.data.LevelSetup +import dev.lucasnlm.antimine.common.level.repository.IDimensionRepository +import dev.lucasnlm.antimine.core.preferences.IPreferencesRepository + +object GameModeFactory { + fun fromDifficultyPreset( + difficulty: DifficultyPreset, + dimensionRepository: IDimensionRepository, + preferencesRepository: IPreferencesRepository + ): LevelSetup = + when (difficulty) { + DifficultyPreset.Standard -> calculateStandardMode(dimensionRepository) + DifficultyPreset.Beginner -> LevelSetup(9, 9, 10, difficulty) + DifficultyPreset.Intermediate -> LevelSetup(16, 16, 40, difficulty) + DifficultyPreset.Expert -> LevelSetup(24, 24, 99, difficulty) + DifficultyPreset.Custom -> preferencesRepository.customGameMode() + } + + private fun calculateStandardMode( + dimensionRepository: IDimensionRepository + ): LevelSetup { + val fieldSize = dimensionRepository.areaSize() + + val display = dimensionRepository.displaySize() + val width = display.widthPixels + val height = display.heightPixels + + val finalWidth = ((width / fieldSize).toInt() - 1).coerceAtLeast(6) + val finalHeight = ((height / fieldSize).toInt() - 3).coerceAtLeast(9) + + return LevelSetup( + finalWidth, + finalHeight, + (finalWidth * finalHeight * 0.2).toInt(), + DifficultyPreset.Standard + ) + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/LevelFacade.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/LevelFacade.kt new file mode 100644 index 00000000..7277b5e5 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/LevelFacade.kt @@ -0,0 +1,264 @@ +package dev.lucasnlm.antimine.common.level + +import dev.lucasnlm.antimine.common.level.data.LevelSetup +import dev.lucasnlm.antimine.common.level.data.Area +import dev.lucasnlm.antimine.common.level.data.GameStats +import dev.lucasnlm.antimine.common.level.data.Mark +import dev.lucasnlm.antimine.common.level.database.data.Save +import dev.lucasnlm.antimine.common.level.database.data.SaveStatus +import java.util.* +import kotlin.math.floor + +class LevelFacade { + private val levelSetup: LevelSetup + private val randomGenerator: Random + private var saveId = 0 + private val startTime = System.currentTimeMillis() + + var hasMines = false + private set + + var seed = 0L + private set + + lateinit var field: Sequence + private set + + private var mines: Sequence = sequenceOf() + + constructor(gameId: Int, levelSetup: LevelSetup, seed: Long = randomSeed()) { + this.saveId = gameId + this.levelSetup = levelSetup + this.randomGenerator = Random().apply { setSeed(seed) } + this.seed = seed + createEmptyField() + } + + constructor(save: Save) { + this.saveId = save.uid + this.levelSetup = save.levelSetup + this.randomGenerator = Random().apply { setSeed(save.seed) } + this.field = save.field.asSequence() + this.hasMines = this.field.firstOrNull { it.hasMine }?.hasMine ?: false + this.mines = this.field.filter { it.hasMine }.asSequence() + } + + private fun createEmptyField() { + val width = levelSetup.width + val height = levelSetup.height + val fieldSize = width * height + this.field = (0 until fieldSize).map { index -> + val yPosition = floor((index / width).toDouble()).toInt() + val xPosition = (index % width) + Area(index, xPosition, yPosition) + }.asSequence() + } + + private fun getArea(id: Int) = field.first { it.id == id } + + fun switchMarkAt(index: Int): Boolean { + val changed: Boolean + getArea(index).apply { + changed = isCovered + if (isCovered) { + mark = when (mark) { + Mark.None -> Mark.Flag + Mark.Flag -> Mark.Question + Mark.Question -> Mark.None + } + } + } + return changed + } + + fun removeMark(index: Int) { + getArea(index).apply { + mark = Mark.None + } + } + + fun hasCoverOn(index: Int): Boolean = getArea(index).isCovered + + fun hasMarkOn(index: Int): Boolean = getArea(index).mark != Mark.None + + fun hasMineOn(index: Int): Boolean = getArea(index).hasMine + + fun plantMinesExcept(index: Int, includeSafeArea: Boolean = false) { + plantRandomMines(index, includeSafeArea) + putMinesTips() + hasMines = true + } + + private fun plantRandomMines(ignoreIndex: Int, includeSafeArea: Boolean) { + getArea(ignoreIndex).run { + safeZone = true + + if (includeSafeArea) { + findNeighbors().forEach { + it.safeZone = true + } + } + } + + field.filterNot { it.safeZone } + .toSet() + .shuffled(randomGenerator) + .take(levelSetup.mines) + .forEach { it.hasMine = true } + mines = field.filter { it.hasMine }.asSequence() + } + + private fun putMinesTips() { + field.forEach { + it.minesAround = if (it.hasMine) 0 else it.findNeighbors().filter { neighbor -> + neighbor.hasMine + }.count() + } + } + + /** + * Run "Flood Fill algorithm" to open all empty neighbors of a target area. + */ + fun openField(target: Area): Boolean { + val result: Boolean = target.isCovered + + if (target.isCovered) { + target.isCovered = false + target.mark = Mark.None + + if (target.minesAround == 0 && !target.hasMine) { + target.findNeighbors().forEach { openField(it) } + } + + if (target.hasMine) { + target.mistake = true + } + } + + return result + } + + fun turnOffAllHighlighted() { + field.forEach { + it.highlighted = false + } + } + + private fun toggleHighlight(target: Area) { + target.highlighted = !target.highlighted + target.findNeighbors() + .filter { it.mark == Mark.None && it.isCovered } + .forEach { + it.highlighted = !it.highlighted + } + } + + fun clickArea(index: Int): Boolean = getArea(index).let { + when { + it.isCovered -> { + openField(getArea(index)) + } + it.minesAround != 0 -> { + toggleHighlight(it) + true + } + else -> { + false + } + } + } + + fun openNeighbors(index: Int): List = + getArea(index) + .findNeighbors() + .filter { + it.mark == Mark.None + } + .map{ + openField(it) + it.id + } + + fun runFlagAssistant() { + mines.filter { it.mark == Mark.None }.forEach { field -> + val neighbors = field.findNeighbors() + val neighborsCount = neighbors.count() + val revealedNeighborsCount = neighbors.filter { neighbor -> + !neighbor.isCovered || (neighbor.hasMine && neighbor.mark == Mark.Flag) + }.count() + + field.mark = if (revealedNeighborsCount == neighborsCount) Mark.Flag else Mark.None + } + } + + fun getStats() = GameStats( + mines.filter { !it.mistake && it.mark == Mark.Flag }.count(), + mines.count(), + field.count() + ) + + fun showAllMines() { + mines.filter { it.mark != Mark.Flag }.forEach { it.isCovered = false } + } + + fun showWrongFlags() { + field.filter { it.mark != Mark.None && !it.hasMine }.forEach { it.mistake = true } + } + + fun revealAllEmptyAreas() { + field.filter { !it.hasMine }.forEach { it.isCovered = false } + } + + fun hasAnyMineExploded(): Boolean = mines.firstOrNull { it.mistake } != null + + fun hasFlaggedAllMines(): Boolean = rightFlags() == levelSetup.mines + + fun hasIsolatedAllMines() = + mines.map { + val neighbors = it.findNeighbors() + val neighborsCount = neighbors.count() + val isolatedNeighborsCount = neighbors.filter { neighbor -> + !neighbor.isCovered || neighbor.hasMine + }.count() + neighborsCount == isolatedNeighborsCount + }.filterNot { it }.count() == 0 + + private fun rightFlags() = mines.count { it.mark == Mark.Flag } + + fun checkVictory(): Boolean = + hasFlaggedAllMines() && hasIsolatedAllMines() && !hasAnyMineExploded() + + fun remainingMines(): Int { + val flagsCount = field.count { it.mark == Mark.Flag } + val minesCount = mines.count() + return (minesCount - flagsCount).coerceAtLeast(0) + } + + private fun Area.findNeighbors() = arrayOf( + getNeighbor(1, 0), + getNeighbor(1, 1), + getNeighbor(0, 1), + getNeighbor(-1, 1), + getNeighbor(-1, 0), + getNeighbor(-1, -1), + getNeighbor(0, -1), + getNeighbor(1, -1) + ).filterNotNull() + + private fun Area.getNeighbor(x: Int, y: Int) = field.firstOrNull { + (it.posX == this.posX + x) && (it.posY == this.posY + y) + } + + fun getSaveState(): Save { + val saveStatus: SaveStatus = when { + checkVictory() -> SaveStatus.VICTORY + hasAnyMineExploded() -> SaveStatus.DEFEAT + else -> SaveStatus.ON_GOING + } + return Save(saveId, seed, startTime, 0L, levelSetup, saveStatus, field.toList()) + } + + companion object { + fun randomSeed(): Long = Random().nextLong() + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/data/AmbientSettings.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/AmbientSettings.kt new file mode 100644 index 00000000..b06d2a17 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/AmbientSettings.kt @@ -0,0 +1,6 @@ +package dev.lucasnlm.antimine.common.level.data + +data class AmbientSettings( + val isAmbientMode: Boolean = false, + val isLowBitAmbient: Boolean = false +) diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/data/Area.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/Area.kt new file mode 100644 index 00000000..8fe1f7b9 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/Area.kt @@ -0,0 +1,14 @@ +package dev.lucasnlm.antimine.common.level.data + +data class Area( + val id: Int, + val posX: Int, + val posY: Int, + var minesAround: Int = 0, + var safeZone: Boolean = false, + var hasMine: Boolean = false, + var mistake: Boolean = false, + var isCovered: Boolean = true, + var mark: Mark = Mark.None, + var highlighted: Boolean = false +) diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/data/DifficultyPreset.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/DifficultyPreset.kt new file mode 100644 index 00000000..d05801c8 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/DifficultyPreset.kt @@ -0,0 +1,9 @@ +package dev.lucasnlm.antimine.common.level.data + +enum class DifficultyPreset(val text: String) { + Standard("STANDARD"), + Beginner("BEGINNER"), + Intermediate("INTERMEDIATE"), + Expert("EXPERT"), + Custom("CUSTOM") +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameEvent.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameEvent.kt new file mode 100644 index 00000000..1df3ae37 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameEvent.kt @@ -0,0 +1,13 @@ +package dev.lucasnlm.antimine.common.level.data + +enum class GameEvent { + StartNewGame, + ResumeGame, + ResumeGameOver, + ResumeVictory, + Pause, + Resume, + Running, + Victory, + GameOver +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameStats.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameStats.kt new file mode 100644 index 00000000..b5182a9b --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameStats.kt @@ -0,0 +1,7 @@ +package dev.lucasnlm.antimine.common.level.data + +data class GameStats( + val rightMines: Int, + val totalMines: Int, + val totalArea: Int +) diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameStatus.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameStatus.kt new file mode 100644 index 00000000..8c163ef5 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/GameStatus.kt @@ -0,0 +1,7 @@ +package dev.lucasnlm.antimine.common.level.data + +enum class GameStatus { + PreGame, + Running, + Over +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/data/LevelSetup.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/LevelSetup.kt new file mode 100644 index 00000000..73bc0dc2 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/LevelSetup.kt @@ -0,0 +1,8 @@ +package dev.lucasnlm.antimine.common.level.data + +data class LevelSetup( + val width: Int, + val height: Int, + val mines: Int, + val preset: DifficultyPreset = DifficultyPreset.Custom +) diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/data/Mark.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/Mark.kt new file mode 100644 index 00000000..51eea9ec --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/data/Mark.kt @@ -0,0 +1,11 @@ +package dev.lucasnlm.antimine.common.level.data + +enum class Mark { + None, + Flag, + Question +} + +fun Mark.isFlag(): Boolean = this == Mark.Flag + +fun Mark.isQuestion(): Boolean = this == Mark.Question \ No newline at end of file diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/database/AppDataBase.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/AppDataBase.kt new file mode 100644 index 00000000..070767a1 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/AppDataBase.kt @@ -0,0 +1,24 @@ +package dev.lucasnlm.antimine.common.level.database + +import androidx.room.Database +import androidx.room.RoomDatabase +import androidx.room.TypeConverters +import dev.lucasnlm.antimine.common.level.database.converters.FieldConverter +import dev.lucasnlm.antimine.common.level.database.converters.LevelSetupConverter +import dev.lucasnlm.antimine.common.level.database.converters.SaveStatusConverter +import dev.lucasnlm.antimine.common.level.database.dao.SaveDao +import dev.lucasnlm.antimine.common.level.database.data.Save + +@Database( + entities = [ + Save::class + ], version = 1, exportSchema = false +) +@TypeConverters( + FieldConverter::class, + SaveStatusConverter::class, + LevelSetupConverter::class +) +abstract class AppDataBase : RoomDatabase() { + abstract fun userDao(): SaveDao +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/FieldConverter.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/FieldConverter.kt new file mode 100644 index 00000000..b7b94867 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/FieldConverter.kt @@ -0,0 +1,25 @@ +package dev.lucasnlm.antimine.common.level.database.converters + +import androidx.room.TypeConverter +import com.squareup.moshi.JsonAdapter +import com.squareup.moshi.Moshi +import com.squareup.moshi.Types +import com.squareup.moshi.kotlin.reflect.KotlinJsonAdapterFactory +import dev.lucasnlm.antimine.common.level.data.Area +import java.lang.reflect.Type + +class FieldConverter { + private val moshi: Moshi = Moshi.Builder().add(KotlinJsonAdapterFactory()).build() + private val jsonAdapter: JsonAdapter> + + init { + val type: Type = Types.newParameterizedType(List::class.java, Area::class.java) + this.jsonAdapter = moshi.adapter(type) + } + + @TypeConverter + fun toAreaList(jsonInput: String): List = jsonAdapter.fromJson(jsonInput) ?: listOf() + + @TypeConverter + fun toJsonString(field: List): String = jsonAdapter.toJson(field) +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/LevelSetupConverter.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/LevelSetupConverter.kt new file mode 100644 index 00000000..39ef8e30 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/LevelSetupConverter.kt @@ -0,0 +1,24 @@ +package dev.lucasnlm.antimine.common.level.database.converters + +import androidx.room.TypeConverter +import com.squareup.moshi.JsonAdapter +import com.squareup.moshi.Moshi +import com.squareup.moshi.kotlin.reflect.KotlinJsonAdapterFactory +import dev.lucasnlm.antimine.common.level.data.DifficultyPreset +import dev.lucasnlm.antimine.common.level.data.LevelSetup + +class LevelSetupConverter { + private val moshi: Moshi = Moshi.Builder().add(KotlinJsonAdapterFactory()).build() + private val jsonAdapter: JsonAdapter + + init { + this.jsonAdapter = moshi.adapter(LevelSetup::class.java) + } + + @TypeConverter + fun toLevelSetup(jsonInput: String): LevelSetup = + jsonAdapter.fromJson(jsonInput) ?: LevelSetup(9, 9, 10, DifficultyPreset.Beginner) + + @TypeConverter + fun toJsonString(field: LevelSetup): String = jsonAdapter.toJson(field) +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/SaveStatusConverter.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/SaveStatusConverter.kt new file mode 100644 index 00000000..f847b023 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/converters/SaveStatusConverter.kt @@ -0,0 +1,19 @@ +package dev.lucasnlm.antimine.common.level.database.converters + +import androidx.room.TypeConverter +import dev.lucasnlm.antimine.common.level.database.data.SaveStatus + +class SaveStatusConverter { + + @TypeConverter + fun toSaveStatus(status: Int): SaveStatus = + when (status) { + 0 -> SaveStatus.ON_GOING + 1 -> SaveStatus.VICTORY + 2 -> SaveStatus.DEFEAT + else -> throw IllegalArgumentException("Could not recognize SaveStatus") + } + + @TypeConverter + fun toInteger(status: SaveStatus): Int = status.code +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/database/dao/SaveDao.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/dao/SaveDao.kt new file mode 100644 index 00000000..7743a8ba --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/dao/SaveDao.kt @@ -0,0 +1,28 @@ +package dev.lucasnlm.antimine.common.level.database.dao + +import androidx.room.* +import dev.lucasnlm.antimine.common.level.database.data.Save + +@Dao +interface SaveDao { + @Query("SELECT * FROM save") + fun getAll(): List + + @Query("SELECT * FROM save WHERE uid IN (:gameIds)") + fun loadAllByIds(gameIds: IntArray): List + + @Query("SELECT * FROM save WHERE uid = :gameId LIMIT 1") + fun loadById(gameId: Int): Save + + @Query("SELECT * FROM save ORDER BY uid DESC LIMIT 1") + fun loadCurrent(): Save? + + @Query("SELECT count(uid) FROM save") + fun getSaveCounts(): Int + + @Insert(onConflict = OnConflictStrategy.REPLACE) + fun insertAll(vararg saves: Save): Array + + @Delete + fun delete(save: Save) +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/database/data/Save.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/data/Save.kt new file mode 100644 index 00000000..01ffcdb6 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/data/Save.kt @@ -0,0 +1,36 @@ +package dev.lucasnlm.antimine.common.level.database.data + +import androidx.room.* +import dev.lucasnlm.antimine.common.level.data.Area +import dev.lucasnlm.antimine.common.level.data.LevelSetup +import dev.lucasnlm.antimine.common.level.database.converters.FieldConverter +import dev.lucasnlm.antimine.common.level.database.converters.SaveStatusConverter + +@Entity +data class Save( + @PrimaryKey(autoGenerate = true) + val uid: Int, + + @ColumnInfo(name = "seed") + val seed: Long, + + @ColumnInfo(name = "date") + val startDate: Long, + + @ColumnInfo(name = "duration") + val duration: Long, + + @ColumnInfo(name = "width") + val levelSetup: LevelSetup, + + @TypeConverters(SaveStatusConverter::class) + @ColumnInfo(name = "status") + val status: SaveStatus, + + @TypeConverters(FieldConverter::class) + @ColumnInfo(name = "field") + val field: List +) + + + diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/database/data/SaveStatus.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/data/SaveStatus.kt new file mode 100644 index 00000000..d8af3be6 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/database/data/SaveStatus.kt @@ -0,0 +1,11 @@ +package dev.lucasnlm.antimine.common.level.database.data + +import androidx.room.TypeConverters +import dev.lucasnlm.antimine.common.level.database.converters.SaveStatusConverter + +@TypeConverters(SaveStatusConverter::class) +enum class SaveStatus constructor(val code: Int) { + ON_GOING(0), + VICTORY(1), + DEFEAT(2) +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/di/LevelComponent.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/di/LevelComponent.kt new file mode 100644 index 00000000..2f336859 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/di/LevelComponent.kt @@ -0,0 +1,18 @@ +package dev.lucasnlm.antimine.common.level.di + +import dagger.Component +import dagger.android.support.AndroidSupportInjectionModule + +@Component( + modules = [ + AndroidSupportInjectionModule::class, + LevelModule::class + ] +) +abstract class LevelComponent { + @Component.Builder + interface Builder { + fun levelModule(levelModule: LevelModule): Builder + fun build(): LevelComponent + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/di/LevelModule.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/di/LevelModule.kt new file mode 100644 index 00000000..9c40baa0 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/di/LevelModule.kt @@ -0,0 +1,73 @@ +package dev.lucasnlm.antimine.common.level.di + +import android.app.Application +import android.content.Context +import androidx.lifecycle.MutableLiveData +import androidx.room.Room +import dagger.Module +import dagger.Provides +import dev.lucasnlm.antimine.common.level.data.GameEvent +import dev.lucasnlm.antimine.common.level.database.AppDataBase +import dev.lucasnlm.antimine.common.level.database.dao.SaveDao +import dev.lucasnlm.antimine.common.level.repository.DimensionRepository +import dev.lucasnlm.antimine.common.level.repository.IDimensionRepository +import dev.lucasnlm.antimine.common.level.repository.ISavesRepository +import dev.lucasnlm.antimine.common.level.repository.SavesRepository +import dev.lucasnlm.antimine.common.level.utils.Clock +import dev.lucasnlm.antimine.common.level.utils.HapticFeedbackInteractor +import dev.lucasnlm.antimine.common.level.utils.IHapticFeedbackInteractor +import dev.lucasnlm.antimine.common.level.viewmodel.GameViewModelFactory +import dev.lucasnlm.antimine.core.analytics.AnalyticsManager +import dev.lucasnlm.antimine.core.preferences.IPreferencesRepository + +@Module +class LevelModule { + @Provides + fun provideGameEventObserver(): MutableLiveData = MutableLiveData() + + @Provides + fun provideClock(): Clock = Clock() + + @Provides + fun provideGameViewModelFactory( + application: Application, + gameEventObserver: MutableLiveData, + savesRepository: ISavesRepository, + dimensionRepository: IDimensionRepository, + preferencesRepository: IPreferencesRepository, + hapticFeedbackInteractor: IHapticFeedbackInteractor, + analyticsManager: AnalyticsManager, + clock: Clock + ) = GameViewModelFactory( + application, gameEventObserver, savesRepository, + dimensionRepository, preferencesRepository, hapticFeedbackInteractor, analyticsManager, clock + ) + + @Provides + fun provideDimensionRepository( + context: Context, + preferencesRepository: IPreferencesRepository + ): IDimensionRepository = + DimensionRepository(context, preferencesRepository) + + @Provides + fun provideDataBase(application: Application): AppDataBase = + Room.databaseBuilder(application, AppDataBase::class.java, DATA_BASE_NAME).build() + + @Provides + fun provideSaveDao(appDataBase: AppDataBase): SaveDao = appDataBase.userDao() + + @Provides + fun provideSavesRepository(saveDao: SaveDao): ISavesRepository = SavesRepository(saveDao) + + @Provides + fun provideHapticFeedbackInteractor( + application: Application, + preferencesRepository: IPreferencesRepository + ): IHapticFeedbackInteractor = + HapticFeedbackInteractor(application, preferencesRepository) + + companion object { + const val DATA_BASE_NAME = "saves-db" + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/repository/DimensionRepository.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/repository/DimensionRepository.kt new file mode 100644 index 00000000..66c053a0 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/repository/DimensionRepository.kt @@ -0,0 +1,38 @@ +package dev.lucasnlm.antimine.common.level.repository + +import android.content.Context +import android.content.res.Resources +import android.content.res.TypedArray +import android.util.DisplayMetrics +import dev.lucasnlm.antimine.common.R +import dev.lucasnlm.antimine.core.preferences.IPreferencesRepository + +interface IDimensionRepository { + fun areaSize(): Float + fun displaySize(): DisplayMetrics + fun actionBarSize(): Int +} + +class DimensionRepository( + private val context: Context, + private val preferencesRepository: IPreferencesRepository +) : IDimensionRepository { + + override fun areaSize(): Float = if (preferencesRepository.useLargeAreas()) { + context.resources.getDimension(R.dimen.accessible_field_size) + } else { + context.resources.getDimension(R.dimen.field_size) + } + + override fun displaySize(): DisplayMetrics = Resources.getSystem().displayMetrics + + override fun actionBarSize(): Int { + val styledAttributes: TypedArray = + context.theme.obtainStyledAttributes( + IntArray(1) { android.R.attr.actionBarSize } + ) + val actionBarSize: Int = styledAttributes.getDimension(0, 0.0f).toInt() + styledAttributes.recycle() + return actionBarSize + } +} \ No newline at end of file diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/repository/DrawableRepository.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/repository/DrawableRepository.kt new file mode 100644 index 00000000..c6f709d9 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/repository/DrawableRepository.kt @@ -0,0 +1,44 @@ +package dev.lucasnlm.antimine.common.level.repository + +import android.content.Context +import android.graphics.drawable.Drawable +import androidx.core.content.ContextCompat +import dev.lucasnlm.antimine.common.R + +class DrawableRepository { + private var flag: Drawable? = null + private var redFlag: Drawable? = null + private var mineExploded: Drawable? = null + private var mine: Drawable? = null + private var mineLow: Drawable? = null + private var question: Drawable? = null + + fun provideFlagDrawable(context: Context) = + flag ?: ContextCompat.getDrawable(context, R.drawable.flag).also { flag = it } + + fun provideRedFlagDrawable(context: Context) = + redFlag ?: ContextCompat.getDrawable(context, R.drawable.red_flag).also { redFlag = it } + + fun provideQuestionDrawable(context: Context) = + question ?: ContextCompat.getDrawable(context, R.drawable.question).also { question = it } + + fun provideMineExploded(context: Context) = + mineExploded ?: ContextCompat.getDrawable( + context, + R.drawable.mine_exploded + ).also { mineExploded = it } + + fun provideMine(context: Context) = + mine ?: ContextCompat.getDrawable(context, R.drawable.mine).also { mine = it } + + fun provideMineLow(context: Context) = + mineLow ?: ContextCompat.getDrawable(context, R.drawable.mine_low).also { mineLow = it } + + fun free() { + flag = null + mineExploded = null + mine = null + mineLow = null + question = null + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/repository/SavesRepository.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/repository/SavesRepository.kt new file mode 100644 index 00000000..e276c9d9 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/repository/SavesRepository.kt @@ -0,0 +1,22 @@ +package dev.lucasnlm.antimine.common.level.repository + +import dev.lucasnlm.antimine.common.level.database.dao.SaveDao +import dev.lucasnlm.antimine.common.level.database.data.Save +import javax.inject.Inject + +interface ISavesRepository { + suspend fun getNewSaveId(): Int + suspend fun fetchCurrentSave(): Save? + suspend fun saveGame(save: Save): Long? +} + +class SavesRepository @Inject constructor( + private val savesDao: SaveDao +) : ISavesRepository { + + override suspend fun getNewSaveId(): Int = savesDao.getSaveCounts() + 1 + + override suspend fun fetchCurrentSave(): Save? = savesDao.loadCurrent() + + override suspend fun saveGame(save: Save) = savesDao.insertAll(save).firstOrNull() +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/utils/Clock.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/utils/Clock.kt new file mode 100644 index 00000000..079df170 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/utils/Clock.kt @@ -0,0 +1,39 @@ +package dev.lucasnlm.antimine.common.level.utils + +import java.util.Timer +import java.util.TimerTask + +class Clock { + private var elapsedTimeSeconds: Long = 0 + private var timer: Timer? = null + + val isStopped: Boolean + get() = (timer == null) + + fun reset(initialValue: Long = 0L) { + stop() + this.elapsedTimeSeconds = initialValue + } + + fun time() = elapsedTimeSeconds + + fun stop() { + timer?.apply { + cancel() + purge() + } + timer = null + } + + fun start(onTick: (seconds: Long) -> Unit) { + stop() + timer = Timer().apply { + scheduleAtFixedRate(object : TimerTask() { + override fun run() { + elapsedTimeSeconds++ + onTick(elapsedTimeSeconds) + } + }, 1000L, 1000L) + } + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/utils/HapticFeedbackInteractor.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/utils/HapticFeedbackInteractor.kt new file mode 100644 index 00000000..b24255ce --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/utils/HapticFeedbackInteractor.kt @@ -0,0 +1,44 @@ +package dev.lucasnlm.antimine.common.level.utils + +import android.app.Application +import android.content.Context.VIBRATOR_SERVICE +import android.os.Build +import android.os.VibrationEffect +import android.os.Vibrator +import dev.lucasnlm.antimine.core.preferences.IPreferencesRepository + +interface IHapticFeedbackInteractor { + fun toggleFlagFeedback() + fun explosionFeedback() +} + +class HapticFeedbackInteractor( + application: Application, + private val preferencesRepository: IPreferencesRepository +) : IHapticFeedbackInteractor { + private val vibrator: Vibrator = application.getSystemService(VIBRATOR_SERVICE) as Vibrator + + override fun toggleFlagFeedback() { + if (preferencesRepository.useHapticFeedback()) { + vibrateTo(70, 240) + vibrateTo(10, 100) + } + } + + override fun explosionFeedback() { + if (preferencesRepository.useHapticFeedback()) { + vibrateTo(400, -1) + } + } + + private fun vibrateTo(time: Long, amplitude: Int) { + if (Build.VERSION.SDK_INT >= 26) { + vibrator.vibrate( + VibrationEffect.createOneShot(time, amplitude) + ) + } else { + @Suppress("DEPRECATION") + vibrator.vibrate(time) + } + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaAdapter.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaAdapter.kt new file mode 100644 index 00000000..4142247a --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaAdapter.kt @@ -0,0 +1,109 @@ +package dev.lucasnlm.antimine.common.level.view + +import android.content.Context +import android.graphics.Paint +import android.graphics.RectF +import android.graphics.Typeface +import android.util.Log +import android.view.LayoutInflater +import android.view.ViewGroup +import androidx.recyclerview.widget.RecyclerView +import dev.lucasnlm.antimine.common.R +import dev.lucasnlm.antimine.common.level.data.Area +import dev.lucasnlm.antimine.common.level.viewmodel.GameViewModel + +class AreaAdapter( + context: Context, + private val viewModel: GameViewModel +) : RecyclerView.Adapter() { + + private var field = listOf() + private var isLowBitAmbient = false + private var isAmbientMode = false + private val paintSettings: AreaPaintSettings + + private val clickEnabled: Boolean + get() = viewModel.isGameActive() + + init { + setHasStableIds(true) + paintSettings = createAreaPaintSettings(context, viewModel.useAccessibilityMode()) + } + + fun setAmbientMode(isAmbientMode: Boolean, isLowBitAmbient: Boolean) { + this.isLowBitAmbient = isLowBitAmbient + this.isAmbientMode = isAmbientMode + } + + fun bindField(area: List) { + this.field = area + notifyDataSetChanged() + } + + override fun getItemCount() = field.size + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): FieldViewHolder { + val layout = if (viewModel.useAccessibilityMode()) { R.layout.view_accessibility_field } else { R.layout.view_field } + val view = LayoutInflater.from(parent.context).inflate(layout, parent, false) + val holder = FieldViewHolder(view) + + holder.itemView.setOnLongClickListener { target -> + target.requestFocus() + + val position = holder.adapterPosition + if (position == RecyclerView.NO_POSITION) { + Log.d(TAG, "Item no longer exists.") + } else if (clickEnabled) { + viewModel.onLongClick(position) + } + + true + } + + holder.itemView.setOnClickListener { + val position = holder.adapterPosition + if (position == RecyclerView.NO_POSITION) { + Log.d(TAG, "Item no longer exists.") + } else if (clickEnabled) { + viewModel.onClickArea(position) + } + } + + return holder + } + + private fun getItem(position: Int) = field[position] + + override fun getItemId(position: Int): Long = getItem(position).id.toLong() + + override fun onBindViewHolder(holder: FieldViewHolder, position: Int) { + val field = getItem(position) + holder.areaView.bindField(field, isAmbientMode, isLowBitAmbient, paintSettings) + } + + companion object { + private const val TAG = "AreaAdapter" + + private fun createAreaPaintSettings(context: Context, useLargeArea: Boolean): AreaPaintSettings { + val resources = context.resources + val padding = resources.getDimension(R.dimen.field_padding) + val size = if (useLargeArea) { + resources.getDimension(R.dimen.accessible_field_size).toInt() + } else { + resources.getDimension(R.dimen.field_size).toInt() + } + return AreaPaintSettings( + Paint().apply { + isAntiAlias = true + isDither = true + style = Paint.Style.FILL + textSize = 18.0f * context.resources.displayMetrics.density + typeface = Typeface.DEFAULT_BOLD + textAlign = Paint.Align.CENTER + }, + RectF(padding, padding, size - padding, size - padding), + resources.getDimension(R.dimen.field_radius) + ) + } + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaPaintSettings.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaPaintSettings.kt new file mode 100644 index 00000000..340f1e39 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaPaintSettings.kt @@ -0,0 +1,11 @@ +package dev.lucasnlm.antimine.common.level.view + +import android.graphics.Paint +import android.graphics.RectF + +data class AreaPaintSettings( + val painter: Paint, + val rectF: RectF, + val radius: Float +) + diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaView.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaView.kt new file mode 100755 index 00000000..65b9d172 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/view/AreaView.kt @@ -0,0 +1,273 @@ +package dev.lucasnlm.antimine.common.level.view + +import android.annotation.SuppressLint +import android.content.Context +import android.graphics.* +import androidx.core.content.ContextCompat +import android.util.AttributeSet +import android.view.View +import dev.lucasnlm.antimine.common.level.data.Area +import android.graphics.RectF +import dev.lucasnlm.antimine.common.level.repository.DrawableRepository +import android.util.TypedValue +import android.graphics.drawable.Drawable +import android.os.Build +import androidx.core.view.ViewCompat +import dev.lucasnlm.antimine.common.R +import dev.lucasnlm.antimine.common.level.data.Mark + +class AreaView : View { + private var covered = true + private var minesAround: Int = -1 + private var mark = Mark.None + private var hasMine = false + private var mistake = false + private var isAmbientMode = false + private var isLowBitAmbient = false + private var highlighted = false + + private lateinit var paintSettings: AreaPaintSettings + private val drawableRepository = DrawableRepository() + + constructor(context: Context) + : super(context) + + constructor(context: Context, attrs: AttributeSet?) + : super(context, attrs) + + constructor(context: Context, attrs: AttributeSet?, defStyleAttr: Int) + : super(context, attrs, defStyleAttr) + + fun bindField(area: Area, isAmbientMode: Boolean, isLowBitAmbient: Boolean, paintSettings: AreaPaintSettings) { + this.paintSettings = paintSettings + minesAround = if (covered) -1 else area.minesAround + + bindContentDescription(area) + + val changed = arrayOf( + this.isAmbientMode != isAmbientMode, + covered != area.isCovered, + minesAround != area.minesAround, + mark != area.mark, + hasMine != area.hasMine, + mistake != area.mistake, + minesAround != area.minesAround, + highlighted != area.highlighted + ).find { it } ?: false + + // Used on Wear OS + this.isAmbientMode = isAmbientMode + this.isLowBitAmbient = isLowBitAmbient + + paintSettings.painter.isAntiAlias = !isAmbientMode || isAmbientMode && !isLowBitAmbient + + covered = area.isCovered + mark = area.mark + hasMine = area.hasMine + mistake = area.mistake + minesAround = area.minesAround + highlighted = area.highlighted + if (Build.VERSION.SDK_INT >= 23) { + this.foreground = when { + !isAmbientMode && covered -> getRippleEffect(context) + else -> null + } + } + + if (changed) { + invalidate() + } + } + + @SuppressLint("InlinedApi") + private fun bindContentDescription(area: Area) { + contentDescription = when { + area.mark == Mark.Flag -> { + context.getString(if (area.mistake) R.string.desc_wrongly_marked_area else R.string.desc_marked_area) + } + area.mark == Mark.Question -> context.getString(R.string.desc_marked_area) + area.isCovered -> context.getString(R.string.desc_convered_area) + !area.isCovered && area.minesAround > 0 -> area.minesAround.toString() + !area.isCovered && area.hasMine -> context.getString(R.string.exploded_mine) + else -> "" + } + + ViewCompat.setImportantForAccessibility( + this, + when { + area.minesAround != 0 -> IMPORTANT_FOR_ACCESSIBILITY_YES + area.hasMine -> IMPORTANT_FOR_ACCESSIBILITY_YES + area.mistake -> IMPORTANT_FOR_ACCESSIBILITY_YES + area.mark != Mark.None -> IMPORTANT_FOR_ACCESSIBILITY_YES + !area.isCovered -> IMPORTANT_FOR_ACCESSIBILITY_NO_HIDE_DESCENDANTS + else -> IMPORTANT_FOR_ACCESSIBILITY_YES + } + ) + } + + override fun onDraw(canvas: Canvas) { + super.onDraw(canvas) + + paintSettings.run { + if (isAmbientMode) { + setBackgroundResource(android.R.color.black) + } else { + setBackgroundResource(android.R.color.transparent) + } + + if (covered) { + if (isAmbientMode) { + painter.apply { + style = Paint.Style.STROKE + strokeWidth = 2.0f + isAntiAlias = !isLowBitAmbient + color = ContextCompat.getColor(context, android.R.color.white) + } + } else { + painter.apply { + style = Paint.Style.FILL + isAntiAlias = !isLowBitAmbient + color = ContextCompat.getColor(context, R.color.view_cover) + alpha = if(highlighted) 155 else 255 + } + } + + painter.run { + canvas.drawRoundRect(rectF, radius, radius, this) + } + + when (mark) { + Mark.Flag -> { + val padding = resources.getDimension(R.dimen.mark_padding).toInt() + + val flag = if (mistake) { + drawableRepository.provideRedFlagDrawable(context) + } else { + drawableRepository.provideFlagDrawable(context) + } + + flag?.setBounds( + rectF.left.toInt() + padding, + rectF.top.toInt() + padding, + rectF.right.toInt() - padding, + rectF.bottom.toInt() - padding + ) + flag?.draw(canvas) + } + Mark.Question -> { + val question = drawableRepository.provideQuestionDrawable(context) + + question?.setBounds( + rectF.left.toInt(), + rectF.top.toInt(), + rectF.right.toInt(), + rectF.bottom.toInt() + ) + question?.draw(canvas) + } + else -> {} + } + } else { + if (isAmbientMode) { + painter.apply { + style = Paint.Style.STROKE + strokeWidth = 0.5f + isAntiAlias = !isLowBitAmbient + color = ContextCompat.getColor(context, android.R.color.white) + } + } else { + painter.apply { + style = Paint.Style.FILL + isAntiAlias = !isLowBitAmbient + color = ContextCompat.getColor(context, R.color.view_clean) + } + } + + painter.run { + canvas.drawRoundRect(rectF, radius, radius, this) + } + + if (hasMine) { + val mine = when { + isAmbientMode -> drawableRepository.provideMineLow(context) + mistake -> drawableRepository.provideMineExploded(context) + else -> drawableRepository.provideMine(context) + } + + mine?.setBounds( + rectF.left.toInt(), + rectF.top.toInt(), + rectF.right.toInt(), + rectF.bottom.toInt() + ) + mine?.draw(canvas) + } else if (minesAround > 0) { + val color = if (isAmbientMode) { + R.color.ambient_color_white + } else { + when (minesAround) { + 1 -> R.color.mines_arround_1 + 2 -> R.color.mines_arround_2 + 3 -> R.color.mines_arround_3 + 4 -> R.color.mines_arround_4 + 5 -> R.color.mines_arround_5 + 6 -> R.color.mines_arround_6 + 7 -> R.color.mines_arround_7 + else -> R.color.mines_arround_8 + } + } + + painter.color = ContextCompat.getColor(context, color) + drawText(canvas, painter, minesAround.toString()) + } + + if (highlighted) { + painter.apply { + style = Paint.Style.STROKE + strokeWidth = 2.0f + isAntiAlias = !isLowBitAmbient + color = if (isAmbientMode) { + ContextCompat.getColor(context, R.color.white) + } else { + ContextCompat.getColor(context, R.color.highlight) + } + + canvas.drawRoundRect(rectF, radius, radius, this) + } + } + } + + if (isFocused) { + painter.apply { + style = Paint.Style.STROKE + strokeWidth = 4f + isAntiAlias = !isLowBitAmbient + color = ContextCompat.getColor(context, android.R.color.holo_orange_dark) + } + + canvas.drawRoundRect(rectF, radius, radius, painter) + } + } + } + + private fun getRippleEffect(context: Context): Drawable? { + val outValue = TypedValue() + context.theme.resolveAttribute( + android.R.attr.selectableItemBackground, outValue, true + ) + return ContextCompat.getDrawable(context, outValue.resourceId) + } + + private fun drawText(canvas: Canvas, paint: Paint, text: String) { + paintSettings.run { + val bounds = RectF(rectF).apply { + right = paint.measureText(text, 0, text.length) + bottom = paint.descent() - paint.ascent() + left += (rectF.width() - right) / 2.0f + top += (rectF.height() - bottom) / 2.0f + } + + canvas.drawText(text, rectF.width() * 0.5f, bounds.top - paint.ascent(), paint) + } + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/view/FieldViewHolder.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/view/FieldViewHolder.kt new file mode 100644 index 00000000..7a4c4758 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/view/FieldViewHolder.kt @@ -0,0 +1,46 @@ +package dev.lucasnlm.antimine.common.level.view + +import android.os.Build +import android.view.KeyEvent +import android.view.View +import androidx.recyclerview.widget.RecyclerView +import dev.lucasnlm.antimine.common.R + +class FieldViewHolder(view: View) : RecyclerView.ViewHolder(view) { + val areaView: AreaView = view.findViewById(R.id.area) + + init { + view.isFocusable = false + areaView.isFocusable = true + areaView.setOnKeyListener { _, keyCode, keyEvent -> + var handled = false + + if (keyCode == KeyEvent.KEYCODE_ENTER || keyCode == KeyEvent.KEYCODE_DPAD_CENTER) { + when (keyEvent.action) { + KeyEvent.ACTION_DOWN -> { + longPressAt = System.currentTimeMillis() + handled = true + } + KeyEvent.ACTION_UP -> { + if (System.currentTimeMillis() - longPressAt > 300L) { + view.performLongClick() + } else { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH_MR1) { + view.callOnClick() + } else { + view.performClick() + } + } + handled = true + } + } + } + + handled + } + } + + companion object { + var longPressAt: Long = 0L + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/view/UnlockedHorizontalScrollView.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/view/UnlockedHorizontalScrollView.kt new file mode 100644 index 00000000..20cb8797 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/view/UnlockedHorizontalScrollView.kt @@ -0,0 +1,31 @@ +package dev.lucasnlm.antimine.common.level.view + +import android.annotation.SuppressLint +import android.content.Context +import android.util.AttributeSet +import android.view.MotionEvent +import android.widget.HorizontalScrollView +import androidx.recyclerview.widget.RecyclerView + +class UnlockedHorizontalScrollView : HorizontalScrollView { + private var recyclerView: RecyclerView? = null + + constructor(context: Context) : super(context) + + constructor(context: Context, attrs: AttributeSet) + : super(context, attrs) + + constructor(context: Context, attrs: AttributeSet, defStyleAttr: Int) + : super(context, attrs, defStyleAttr) + + fun setTarget(recyclerView: RecyclerView) { + this.recyclerView = recyclerView + } + + @SuppressLint("ClickableViewAccessibility") + override fun onTouchEvent(event: MotionEvent): Boolean = + super.onTouchEvent(event) or recyclerView!!.onTouchEvent(event) + + override fun onInterceptTouchEvent(event: MotionEvent): Boolean = + super.onInterceptTouchEvent(event) or recyclerView!!.onInterceptTouchEvent(event) +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/viewmodel/GameViewModel.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/viewmodel/GameViewModel.kt new file mode 100644 index 00000000..4b5d04c2 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/viewmodel/GameViewModel.kt @@ -0,0 +1,257 @@ +package dev.lucasnlm.antimine.common.level.viewmodel + +import android.app.Application +import androidx.lifecycle.MutableLiveData +import androidx.lifecycle.ViewModel +import dev.lucasnlm.antimine.common.level.GameModeFactory +import dev.lucasnlm.antimine.common.level.LevelFacade +import dev.lucasnlm.antimine.common.level.data.* +import dev.lucasnlm.antimine.common.level.database.data.Save +import dev.lucasnlm.antimine.common.level.repository.IDimensionRepository +import dev.lucasnlm.antimine.common.level.repository.ISavesRepository +import dev.lucasnlm.antimine.common.level.utils.Clock +import dev.lucasnlm.antimine.common.level.utils.IHapticFeedbackInteractor +import dev.lucasnlm.antimine.core.analytics.AnalyticsManager +import dev.lucasnlm.antimine.core.analytics.Event +import dev.lucasnlm.antimine.core.preferences.IPreferencesRepository +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.GlobalScope +import kotlinx.coroutines.launch +import kotlinx.coroutines.withContext + +class GameViewModel( + val application: Application, + val eventObserver: MutableLiveData, + private val savesRepository: ISavesRepository, + private val dimensionRepository: IDimensionRepository, + private val preferencesRepository: IPreferencesRepository, + private val hapticFeedbackInteractor: IHapticFeedbackInteractor, + private val analyticsManager: AnalyticsManager, + private val clock: Clock +) : ViewModel() { + private lateinit var levelFacade: LevelFacade + private var currentDifficulty: DifficultyPreset = DifficultyPreset.Standard + private var initialized = false + + val field = MutableLiveData>() + val fieldRefresh = MutableLiveData() + val elapsedTimeSeconds = MutableLiveData() + val mineCount = MutableLiveData() + val difficulty = MutableLiveData() + val levelSetup = MutableLiveData() + + private fun startNewGame(gameId: Int, difficultyPreset: DifficultyPreset): LevelSetup { + clock.reset() + elapsedTimeSeconds.postValue(0L) + currentDifficulty = difficultyPreset + + val setup = GameModeFactory.fromDifficultyPreset( + difficultyPreset, dimensionRepository, preferencesRepository + ) + + levelFacade = LevelFacade(gameId, setup) + + mineCount.postValue(setup.mines) + difficulty.postValue(difficultyPreset) + levelSetup.postValue(setup) + field.postValue(levelFacade.field.toList()) + + eventObserver.postValue(GameEvent.StartNewGame) + + analyticsManager.sentEvent(Event.NewGame(setup, levelFacade.seed, useAccessibilityMode())) + + return setup + } + + private fun resumeGameFromSave(save: Save): LevelSetup { + clock.reset(save.duration) + elapsedTimeSeconds.postValue(save.duration) + + val setup = save.levelSetup + levelFacade = LevelFacade(save) + + mineCount.postValue(setup.mines) + difficulty.postValue(save.levelSetup.preset) + levelSetup.postValue(setup) + field.postValue(levelFacade.field.toList()) + + when { + levelFacade.hasAnyMineExploded() -> eventObserver.postValue(GameEvent.ResumeGameOver) + levelFacade.checkVictory() -> eventObserver.postValue(GameEvent.ResumeVictory) + else -> eventObserver.postValue(GameEvent.ResumeGame) + } + + analyticsManager.sentEvent(Event.ResumePreviousGame()) + + return setup + } + + suspend fun startNewGame(difficultyPreset: DifficultyPreset = currentDifficulty): LevelSetup = + withContext(Dispatchers.IO) { + val newGameId = savesRepository.getNewSaveId() + startNewGame(newGameId, difficultyPreset) + } + + suspend fun onCreate(newGame: DifficultyPreset? = null): LevelSetup = withContext(Dispatchers.IO) { + val lastGame = if (newGame == null) savesRepository.fetchCurrentSave() else null + + if (lastGame != null) { + currentDifficulty = lastGame.levelSetup.preset + } else if (newGame != null) { + currentDifficulty = newGame + } + + if (lastGame == null) { + val newGameId = savesRepository.getNewSaveId() + startNewGame(newGameId, currentDifficulty) + } else { + resumeGameFromSave(lastGame) + }.also { + initialized = true + } + } + + fun pauseGame() { + if (initialized) { + if (levelFacade.hasMines) { + eventObserver.postValue(GameEvent.Pause) + } + clock.stop() + } + } + + suspend fun saveGame() { + if (initialized) { + if (levelFacade.hasMines) { + savesRepository.saveGame( + levelFacade.getSaveState().copy(duration = elapsedTimeSeconds.value ?: 0L) + ) + } + } + } + + fun resumeGame() { + if (initialized) { + if (levelFacade.hasMines) { + eventObserver.postValue(GameEvent.Resume) + } + } + } + + fun onLongClick(index: Int) { + levelFacade.turnOffAllHighlighted() + + if (levelFacade.hasCoverOn(index)) { + if (levelFacade.switchMarkAt(index)) { + refreshField(index) + hapticFeedbackInteractor.toggleFlagFeedback() + } + + analyticsManager.sentEvent(Event.LongPressArea(index)) + } else { + levelFacade.openNeighbors(index) + refreshGameStatus() + + analyticsManager.sentEvent(Event.LongPressMultipleArea(index)) + } + + field.postValue(levelFacade.field.toList()) + + if (levelFacade.hasMines) { + mineCount.postValue(levelFacade.remainingMines()) + } + } + + fun onClickArea(index: Int) { + levelFacade.turnOffAllHighlighted() + + if (levelFacade.hasMarkOn(index)) { + levelFacade.removeMark(index) + hapticFeedbackInteractor.toggleFlagFeedback() + refreshField(index) + } else { + if (!levelFacade.hasMines) { + levelFacade.plantMinesExcept(index, true) + } + + levelFacade.clickArea(index) + + field.postValue(levelFacade.field.toList()) + } + + refreshGameStatus() + analyticsManager.sentEvent(Event.PressArea(index)) + } + + private fun refreshMineCount() = mineCount.postValue(levelFacade.remainingMines()) + + private fun refreshGameStatus() { + refreshMineCount() + + when { + levelFacade.hasAnyMineExploded() -> { + hapticFeedbackInteractor.explosionFeedback() + eventObserver.postValue(GameEvent.GameOver) + } + levelFacade.checkVictory() -> eventObserver.postValue(GameEvent.Victory) + else -> { + if (preferencesRepository.useFlagAssistant()){ + levelFacade.runFlagAssistant() + } + eventObserver.postValue(GameEvent.Running) + } + } + } + + fun runClock() { + if (isGameActive()) { + clock.run { + if (isStopped) start { + elapsedTimeSeconds.postValue(it) + } + } + } + } + + fun stopClock() { + clock.stop() + } + + fun revealAllEmptyAreas() { + levelFacade.revealAllEmptyAreas() + } + + fun gameOver() { + levelFacade.run { + analyticsManager.sentEvent(Event.GameOver(clock.time(), getStats())) + + showAllMines() + showWrongFlags() + } + + GlobalScope.launch { + saveGame() + } + } + + fun victory() { + levelFacade.run { + analyticsManager.sentEvent(Event.Victory(clock.time(), getStats(), currentDifficulty)) + + showAllMines() + showWrongFlags() + } + + GlobalScope.launch { + saveGame() + } + } + + fun isGameActive() = (levelFacade.checkVictory() || levelFacade.hasAnyMineExploded()).not() + + fun useAccessibilityMode() = preferencesRepository.useLargeAreas() + + private fun refreshField(index: Int) { + fieldRefresh.postValue(index) + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/common/level/viewmodel/GameViewModelFactory.kt b/common/src/main/java/dev/lucasnlm/antimine/common/level/viewmodel/GameViewModelFactory.kt new file mode 100644 index 00000000..4d6fc791 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/common/level/viewmodel/GameViewModelFactory.kt @@ -0,0 +1,38 @@ +package dev.lucasnlm.antimine.common.level.viewmodel + +import android.app.Application +import androidx.lifecycle.MutableLiveData +import androidx.lifecycle.ViewModel +import androidx.lifecycle.ViewModelProvider +import dev.lucasnlm.antimine.common.level.data.GameEvent +import dev.lucasnlm.antimine.common.level.repository.IDimensionRepository +import dev.lucasnlm.antimine.common.level.repository.ISavesRepository +import dev.lucasnlm.antimine.common.level.utils.Clock +import dev.lucasnlm.antimine.common.level.utils.IHapticFeedbackInteractor +import dev.lucasnlm.antimine.core.analytics.AnalyticsManager +import dev.lucasnlm.antimine.core.preferences.IPreferencesRepository +import javax.inject.Inject + +class GameViewModelFactory @Inject constructor( + private val application: Application, + private val gameEventObserver: MutableLiveData, + private val savesRepository: ISavesRepository, + private val dimensionRepository: IDimensionRepository, + private val preferencesRepository: IPreferencesRepository, + private val hapticFeedbackInteractor: IHapticFeedbackInteractor, + private val analyticsManager: AnalyticsManager, + private val clock: Clock +) : ViewModelProvider.Factory { + + override fun create(modelClass: Class): T = + if (modelClass.isAssignableFrom(GameViewModel::class.java)) { + @Suppress("UNCHECKED_CAST") + GameViewModel( + application, gameEventObserver, savesRepository, + dimensionRepository, preferencesRepository, hapticFeedbackInteractor, + analyticsManager, clock + ) as T + } else { + throw IllegalArgumentException("ViewModel Not Found") + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/analytics/AmplitudeAnalyticsManager.kt b/common/src/main/java/dev/lucasnlm/antimine/core/analytics/AmplitudeAnalyticsManager.kt new file mode 100644 index 00000000..b3bc2451 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/analytics/AmplitudeAnalyticsManager.kt @@ -0,0 +1,27 @@ +package dev.lucasnlm.antimine.core.analytics + +import android.app.Application +import android.content.Context +import com.amplitude.api.Amplitude +import com.amplitude.api.AmplitudeClient +import dev.lucasnlm.antimine.common.R +import org.json.JSONObject + +class AmplitudeAnalyticsManager( + private val application: Application +) : AnalyticsManager { + + private var amplitudeClient: AmplitudeClient? = null + + override fun setup(context: Context, userProperties: Map) { + val key = context.getString(R.string.amplitude_key) + amplitudeClient = Amplitude.getInstance().initialize(application, key).apply { + setUserProperties(JSONObject(userProperties)) + enableForegroundTracking(application) + } + } + + override fun sentEvent(event: Event) { + amplitudeClient?.logEvent(event.title, JSONObject(event.extra)) + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/analytics/AnalyticsManager.kt b/common/src/main/java/dev/lucasnlm/antimine/core/analytics/AnalyticsManager.kt new file mode 100644 index 00000000..90610d3f --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/analytics/AnalyticsManager.kt @@ -0,0 +1,8 @@ +package dev.lucasnlm.antimine.core.analytics + +import android.content.Context + +interface AnalyticsManager { + fun setup(context: Context, userProperties: Map) + fun sentEvent(event: Event) +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/analytics/DebugAnalyticsManager.kt b/common/src/main/java/dev/lucasnlm/antimine/core/analytics/DebugAnalyticsManager.kt new file mode 100644 index 00000000..967f7eea --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/analytics/DebugAnalyticsManager.kt @@ -0,0 +1,18 @@ +package dev.lucasnlm.antimine.core.analytics + +import android.content.Context +import android.util.Log + +class DebugAnalyticsManager : AnalyticsManager { + override fun setup(context: Context, userProperties: Map) { + Log.d(TAG, "Setup Analytics using $userProperties") + } + + override fun sentEvent(event: Event) { + Log.d(TAG, "Sent event: '${event.title}' with ${event.extra}") + } + + companion object { + const val TAG = "DebugAnalyticsManager" + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/analytics/Event.kt b/common/src/main/java/dev/lucasnlm/antimine/core/analytics/Event.kt new file mode 100644 index 00000000..6641edb4 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/analytics/Event.kt @@ -0,0 +1,82 @@ +package dev.lucasnlm.antimine.core.analytics + +import dev.lucasnlm.antimine.common.level.data.DifficultyPreset +import dev.lucasnlm.antimine.common.level.data.GameStats +import dev.lucasnlm.antimine.common.level.data.LevelSetup + +sealed class Event( + val title: String, val extra: Map = mapOf() +) { + class Open : Event("Open game") + + class NewGame(levelSetup: LevelSetup, seed: Long, useAccessibilityMode: Boolean) : + Event("New Game", mapOf( + "Seed" to seed.toString(), + "Difficulty Preset" to levelSetup.preset.text, + "Width" to levelSetup.width.toString(), + "Height" to levelSetup.height.toString(), + "Mines" to levelSetup.mines.toString(), + "Accessibility" to useAccessibilityMode.toString() + ) + ) + + class ResumePreviousGame : Event("Resume previous game") + + class LongPressArea(index: Int) : Event("Long press area", + mapOf("Index" to index.toString()) + ) + + class LongPressMultipleArea(index: Int) : Event("Long press to open multiple", + mapOf("Index" to index.toString()) + ) + + class PressArea(index: Int) : Event("Press area", + mapOf("Index" to index.toString()) + ) + + class GameOver(time: Long, gameStats: GameStats) : Event("Game Over", + mapOf( + "Time" to time.toString(), + "Right Mines" to gameStats.rightMines.toString(), + "Total Mines" to gameStats.totalMines.toString(), + "Total Area" to gameStats.totalArea.toString() + ) + ) + + class Victory(time: Long, gameStats: GameStats, difficultyPreset: DifficultyPreset) : Event( + "Victory", + mapOf( + "Time" to time.toString(), + "Difficulty" to difficultyPreset.text, + "Right Mines" to gameStats.rightMines.toString(), + "Total Mines" to gameStats.totalMines.toString(), + "Total Area" to gameStats.totalArea.toString() + ) + ) + + class Resume : Event("Back to the game") + + class Quit : Event("Quit game") + + class OpenDrawer : Event("Opened Drawer") + + class CloseDrawer : Event("Closed Drawer") + + class OpenAbout : Event("Open About") + + class OpenSettings : Event("Open Settings") + + class ShowRatingRequest(usages: Int) : Event("Shown Rating Request", + mapOf( + "Usages" to usages.toString() + )) + + class TapRatingRequest(from: String) : Event("Rating Request", + mapOf( + "From" to from + )) + + class TapGameReset(resign: Boolean) : Event("Game reset", + mapOf("Resign" to resign.toString()) + ) +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/di/CommonComponent.kt b/common/src/main/java/dev/lucasnlm/antimine/core/di/CommonComponent.kt new file mode 100644 index 00000000..d292d639 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/di/CommonComponent.kt @@ -0,0 +1,18 @@ +package dev.lucasnlm.antimine.core.di + +import dagger.Component +import dagger.android.support.AndroidSupportInjectionModule + +@Component( + modules = [ + AndroidSupportInjectionModule::class, + CommonModule::class + ] +) +abstract class LevelComponent { + @Component.Builder + interface Builder { + fun levelModule(levelModule: CommonModule): Builder + fun build(): LevelComponent + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/di/CommonModule.kt b/common/src/main/java/dev/lucasnlm/antimine/core/di/CommonModule.kt new file mode 100644 index 00000000..75a23ba5 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/di/CommonModule.kt @@ -0,0 +1,38 @@ +package dev.lucasnlm.antimine.core.di + +import android.app.Application +import dagger.Module +import dagger.Provides +import dev.lucasnlm.antimine.common.BuildConfig +import dev.lucasnlm.antimine.core.analytics.AmplitudeAnalyticsManager +import dev.lucasnlm.antimine.core.analytics.AnalyticsManager +import dev.lucasnlm.antimine.core.analytics.DebugAnalyticsManager +import dev.lucasnlm.antimine.core.preferences.IPreferencesRepository +import dev.lucasnlm.antimine.core.preferences.PreferencesInteractor +import dev.lucasnlm.antimine.core.preferences.PreferencesRepository +import javax.inject.Singleton + +@Module +class CommonModule { + @Singleton + @Provides + fun providePreferencesRepository( + preferencesInteractor: PreferencesInteractor + ): IPreferencesRepository = PreferencesRepository(preferencesInteractor) + + @Singleton + @Provides + fun providePreferencesInteractor( + application: Application + ): PreferencesInteractor = PreferencesInteractor(application) + + @Singleton + @Provides + fun provideAnalyticsManager( + application: Application + ): AnalyticsManager = if (BuildConfig.DEBUG) { + DebugAnalyticsManager() + } else { + AmplitudeAnalyticsManager(application) + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/preferences/PreferencesInteractor.kt b/common/src/main/java/dev/lucasnlm/antimine/core/preferences/PreferencesInteractor.kt new file mode 100644 index 00000000..e429d64b --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/preferences/PreferencesInteractor.kt @@ -0,0 +1,42 @@ +package dev.lucasnlm.antimine.core.preferences + +import android.app.Application +import androidx.preference.PreferenceManager +import dev.lucasnlm.antimine.common.level.data.LevelSetup +import javax.inject.Inject + +class PreferencesInteractor @Inject constructor( + private val application: Application +) { + private val preferences by lazy { + PreferenceManager.getDefaultSharedPreferences(application) + } + + fun getCustomMode() = LevelSetup( + preferences.getInt(PREFERENCE_CUSTOM_GAME_WIDTH, 9), + preferences.getInt(PREFERENCE_CUSTOM_GAME_HEIGHT, 9), + preferences.getInt(PREFERENCE_CUSTOM_GAME_MINES, 9) + ) + + fun updateCustomMode(customLevelSetup: LevelSetup) { + preferences.edit().apply { + putInt(PREFERENCE_CUSTOM_GAME_WIDTH, customLevelSetup.width) + putInt(PREFERENCE_CUSTOM_GAME_HEIGHT, customLevelSetup.height) + putInt(PREFERENCE_CUSTOM_GAME_MINES, customLevelSetup.mines) + }.apply() + } + + fun getBoolean(key: String, defaultValue: Boolean) = preferences.getBoolean(key, defaultValue) + + fun putBoolean(key: String, value: Boolean) = preferences.edit().putBoolean(key, value).apply() + + fun getInt(key: String, defaultValue: Int) = preferences.getInt(key, defaultValue) + + fun putInt(key: String, value: Int) = preferences.edit().putInt(key, value).apply() + + companion object { + private const val PREFERENCE_CUSTOM_GAME_WIDTH = "preference_custom_game_width" + private const val PREFERENCE_CUSTOM_GAME_HEIGHT = "preference_custom_game_height" + private const val PREFERENCE_CUSTOM_GAME_MINES = "preference_custom_game_mines" + } +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/preferences/PreferencesRepository.kt b/common/src/main/java/dev/lucasnlm/antimine/core/preferences/PreferencesRepository.kt new file mode 100644 index 00000000..8e1b03ef --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/preferences/PreferencesRepository.kt @@ -0,0 +1,49 @@ +package dev.lucasnlm.antimine.core.preferences + +import dev.lucasnlm.antimine.common.level.data.LevelSetup + +interface IPreferencesRepository { + fun customGameMode(): LevelSetup + fun updateCustomGameMode(levelSetup: LevelSetup) + fun getBoolean(key: String, defaultValue: Boolean): Boolean + fun getInt(key: String, defaultValue: Int): Int + fun putBoolean(key: String, value: Boolean) + fun putInt(key: String, value: Int) + + fun useFlagAssistant(): Boolean + fun useHapticFeedback(): Boolean + fun useLargeAreas(): Boolean +} + +class PreferencesRepository( + private val preferencesInteractor: PreferencesInteractor +) : IPreferencesRepository { + + override fun customGameMode(): LevelSetup = + preferencesInteractor.getCustomMode() + + override fun updateCustomGameMode(levelSetup: LevelSetup) = + preferencesInteractor.updateCustomMode(levelSetup) + + override fun getBoolean(key: String, defaultValue: Boolean): Boolean = + preferencesInteractor.getBoolean(key, defaultValue) + + override fun putBoolean(key: String, value: Boolean) = + preferencesInteractor.putBoolean(key, value) + + override fun getInt(key: String, defaultValue: Int): Int = + preferencesInteractor.getInt(key, defaultValue) + + override fun putInt(key: String, value: Int) = + preferencesInteractor.putInt(key, value) + + override fun useFlagAssistant(): Boolean = + getBoolean("preference_assistant", true) + + override fun useHapticFeedback(): Boolean = + getBoolean("preference_vibration", true) + + override fun useLargeAreas(): Boolean = + getBoolean("preference_large_area", false) + +} diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/scope/ActivityScope.kt b/common/src/main/java/dev/lucasnlm/antimine/core/scope/ActivityScope.kt new file mode 100644 index 00000000..82f84270 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/scope/ActivityScope.kt @@ -0,0 +1,7 @@ +package dev.lucasnlm.antimine.core.scope + +import javax.inject.Scope + +@Scope +@Retention(AnnotationRetention.RUNTIME) +annotation class ActivityScope diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/scope/FragmentScope.kt b/common/src/main/java/dev/lucasnlm/antimine/core/scope/FragmentScope.kt new file mode 100644 index 00000000..8f476192 --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/scope/FragmentScope.kt @@ -0,0 +1,7 @@ +package dev.lucasnlm.antimine.core.scope + +import javax.inject.Scope + +@Scope +@Retention(AnnotationRetention.RUNTIME) +annotation class FragmentScope diff --git a/common/src/main/java/dev/lucasnlm/antimine/core/utils/DarkModeUtils.kt b/common/src/main/java/dev/lucasnlm/antimine/core/utils/DarkModeUtils.kt new file mode 100644 index 00000000..19f24d4a --- /dev/null +++ b/common/src/main/java/dev/lucasnlm/antimine/core/utils/DarkModeUtils.kt @@ -0,0 +1,12 @@ +package dev.lucasnlm.antimine.core.utils + +import android.content.Context +import android.content.res.Configuration.UI_MODE_NIGHT_YES +import android.content.res.Configuration.UI_MODE_NIGHT_MASK + +fun isDarkModeEnabled(context: Context): Boolean { + return when (context.resources.configuration.uiMode and UI_MODE_NIGHT_MASK) { + UI_MODE_NIGHT_YES -> true + else -> false + } +} diff --git a/common/src/main/res/drawable-night/mine_exploded.xml b/common/src/main/res/drawable-night/mine_exploded.xml new file mode 100644 index 00000000..652db736 --- /dev/null +++ b/common/src/main/res/drawable-night/mine_exploded.xml @@ -0,0 +1,14 @@ + + + + diff --git a/common/src/main/res/drawable-night/red_flag.xml b/common/src/main/res/drawable-night/red_flag.xml new file mode 100644 index 00000000..8a2c2f71 --- /dev/null +++ b/common/src/main/res/drawable-night/red_flag.xml @@ -0,0 +1,10 @@ + + + diff --git a/common/src/main/res/drawable-night/replay.xml b/common/src/main/res/drawable-night/replay.xml new file mode 100644 index 00000000..2f8e48c8 --- /dev/null +++ b/common/src/main/res/drawable-night/replay.xml @@ -0,0 +1,9 @@ + + + diff --git a/common/src/main/res/drawable-night/splash.xml b/common/src/main/res/drawable-night/splash.xml new file mode 100644 index 00000000..75613ea0 --- /dev/null +++ b/common/src/main/res/drawable-night/splash.xml @@ -0,0 +1,13 @@ + + + + + + + + + + + diff --git a/common/src/main/res/drawable-night/splash_image.png b/common/src/main/res/drawable-night/splash_image.png new file mode 100644 index 0000000000000000000000000000000000000000..df6009ff43ff9166d607022a4173713877406b55 GIT binary patch literal 6291 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

)NUE&dF6tW z#23^Nij!zJrYdi9gr{?4dbh(z!--}(DUs5#rQw$cVNIsR`7l5_Py=LfnFON*+zt)6 z-#j*GPy(q#jhse~s}NOjI+m$t-t>F~+>G#I>NP+bW)5Dt(*&v$9)Y(l<+#5c#rK<0 zW-T7b5=UBKfFgde^Tz`hW!Y@fA3*`simr?MfY1L?# + + diff --git a/common/src/main/res/drawable/close.xml b/common/src/main/res/drawable/close.xml new file mode 100644 index 00000000..fd8599cf --- /dev/null +++ b/common/src/main/res/drawable/close.xml @@ -0,0 +1,9 @@ + + + diff --git a/common/src/main/res/drawable/done.xml b/common/src/main/res/drawable/done.xml new file mode 100644 index 00000000..436553a0 --- /dev/null +++ b/common/src/main/res/drawable/done.xml @@ -0,0 +1,9 @@ + + + diff --git a/common/src/main/res/drawable/flag.xml b/common/src/main/res/drawable/flag.xml new file mode 100755 index 00000000..f8abb7b2 --- /dev/null +++ b/common/src/main/res/drawable/flag.xml @@ -0,0 +1,10 @@ + + + diff --git a/common/src/main/res/drawable/main_logo.xml b/common/src/main/res/drawable/main_logo.xml new file mode 100644 index 00000000..a61170fb --- /dev/null +++ b/common/src/main/res/drawable/main_logo.xml @@ -0,0 +1,18 @@ + + + + diff --git a/common/src/main/res/drawable/mine.xml b/common/src/main/res/drawable/mine.xml new file mode 100755 index 00000000..2b38bcd4 --- /dev/null +++ b/common/src/main/res/drawable/mine.xml @@ -0,0 +1,14 @@ + + + + diff --git a/common/src/main/res/drawable/mine_exploded.xml b/common/src/main/res/drawable/mine_exploded.xml new file mode 100644 index 00000000..3637dc92 --- /dev/null +++ b/common/src/main/res/drawable/mine_exploded.xml @@ -0,0 +1,14 @@ + + + + diff --git a/common/src/main/res/drawable/mine_icon.xml b/common/src/main/res/drawable/mine_icon.xml new file mode 100755 index 00000000..a9af31dc --- /dev/null +++ b/common/src/main/res/drawable/mine_icon.xml @@ -0,0 +1,10 @@ + + + diff --git a/common/src/main/res/drawable/mine_low.xml b/common/src/main/res/drawable/mine_low.xml new file mode 100644 index 00000000..395f69a0 --- /dev/null +++ b/common/src/main/res/drawable/mine_low.xml @@ -0,0 +1,10 @@ + + + diff --git a/common/src/main/res/drawable/question.xml b/common/src/main/res/drawable/question.xml new file mode 100644 index 00000000..ab6ebaf1 --- /dev/null +++ b/common/src/main/res/drawable/question.xml @@ -0,0 +1,5 @@ + + + diff --git a/common/src/main/res/drawable/red_flag.xml b/common/src/main/res/drawable/red_flag.xml new file mode 100644 index 00000000..ac684a5f --- /dev/null +++ b/common/src/main/res/drawable/red_flag.xml @@ -0,0 +1,10 @@ + + + diff --git a/common/src/main/res/drawable/replay.xml b/common/src/main/res/drawable/replay.xml new file mode 100644 index 00000000..bbde5a12 --- /dev/null +++ b/common/src/main/res/drawable/replay.xml @@ -0,0 +1,9 @@ + + + diff --git a/common/src/main/res/drawable/splash.xml b/common/src/main/res/drawable/splash.xml new file mode 100644 index 00000000..f28c45cb --- /dev/null +++ b/common/src/main/res/drawable/splash.xml @@ -0,0 +1,13 @@ + + + + + + + + + + + diff --git a/common/src/main/res/drawable/splash_image.png b/common/src/main/res/drawable/splash_image.png new file mode 100644 index 0000000000000000000000000000000000000000..14d65152f3b0c21b585e24f871e74b827cbb3756 GIT binary patch literal 9316 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 efkv`(-NDq5lufEHOp^ literal 0 HcmV?d00001 diff --git a/common/src/main/res/drawable/timer.xml b/common/src/main/res/drawable/timer.xml new file mode 100755 index 00000000..4b8b566f --- /dev/null +++ b/common/src/main/res/drawable/timer.xml @@ -0,0 +1,10 @@ + + + diff --git a/common/src/main/res/drawable/title.xml b/common/src/main/res/drawable/title.xml new file mode 100644 index 00000000..18125bcd --- /dev/null +++ b/common/src/main/res/drawable/title.xml @@ -0,0 +1,17 @@ + + + + diff --git a/common/src/main/res/drawable/unchecked.xml b/common/src/main/res/drawable/unchecked.xml new file mode 100644 index 00000000..5ed61368 --- /dev/null +++ b/common/src/main/res/drawable/unchecked.xml @@ -0,0 +1,9 @@ + + + diff --git a/common/src/main/res/layout-watch/fragment_level.xml b/common/src/main/res/layout-watch/fragment_level.xml new file mode 100644 index 00000000..6c025af6 --- /dev/null +++ b/common/src/main/res/layout-watch/fragment_level.xml @@ -0,0 +1,34 @@ + + + + + + + + + diff --git a/common/src/main/res/layout/fragment_level.xml b/common/src/main/res/layout/fragment_level.xml new file mode 100644 index 00000000..df391cfc --- /dev/null +++ b/common/src/main/res/layout/fragment_level.xml @@ -0,0 +1,32 @@ + + + + + + + + + diff --git a/common/src/main/res/layout/view_accessibility_field.xml b/common/src/main/res/layout/view_accessibility_field.xml new file mode 100644 index 00000000..08ff4ab0 --- /dev/null +++ b/common/src/main/res/layout/view_accessibility_field.xml @@ -0,0 +1,14 @@ + + + + + + diff --git a/common/src/main/res/layout/view_field.xml b/common/src/main/res/layout/view_field.xml new file mode 100644 index 00000000..676353fc --- /dev/null +++ b/common/src/main/res/layout/view_field.xml @@ -0,0 +1,14 @@ + + + + + + diff --git a/common/src/main/res/mipmap-anydpi-v26/ic_launcher.xml b/common/src/main/res/mipmap-anydpi-v26/ic_launcher.xml new file mode 100644 index 00000000..db54720e --- /dev/null +++ b/common/src/main/res/mipmap-anydpi-v26/ic_launcher.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/common/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml b/common/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml new file mode 100644 index 00000000..db54720e --- /dev/null +++ b/common/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/common/src/main/res/mipmap-hdpi/ic_launcher.png b/common/src/main/res/mipmap-hdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..cc1d3ea89fbf802fd2fb9b51940b4f6cfe7953f6 GIT binary patch literal 4603 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 literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-hdpi/ic_launcher_foreground.png b/common/src/main/res/mipmap-hdpi/ic_launcher_foreground.png new file mode 100644 index 0000000000000000000000000000000000000000..30462649783223921400035da52416a0e8fc8646 GIT binary patch literal 1277 zcmaJ>`#%#39M!w4RI4>xuGo;m+~!%UrCjsKjF1~$j}%k8%ot`fmD{vY6d9!`8ljQx zt`X*wdF@(UuT3JKZJFF~tvt8l>L2KQKIfbt&iC{A{&Wu02kWM%yH8g~M@P@Y-Nkp! zOTKgK<~42k3^LHs+1Tpga>}1PJUtrBDD(SSXEsTq8TKhl#ueR9Zn}MI9>Fr>MEMO^ zi1Q8-PeJuPePik#4>>p#n{d(A_d0PNwZsebl;CA|XW{O5FRT(}GgC`OD5bNtM0>)o z`|ViG%A4-<4aeeFRu>;mJX}0JahbpN`iHnilMW3Hv6V`tz-yH=m&NBT}xP+!-lXU_h|^lWVa7(Hb_V${G= z8B1p2Zr^y4L0R5Twqv#=&t77kO{Ot3@D;G8Kxt=4V3T|>f#6si*od^ZPk6kY)oZZB zEv5JZplrxcG;hW&XnyUkD#K*PLWP(;1{IQXa|dv)u&X_dyZlD_xW+~0*z)*T(^$`S z=z?$u_)+ndB3Zg@0H5@Q-nxesZ_^(msB#~Xrez#%%VO}kg)+YUD1EQIMa2{-W#@R@m;PrVY0pbGz%VNo{`N@$?$zhQVxjfM=f?WEd~1i7CZM9UtWCBf2(C&C{bZMgz|d*4q@0y^<$wn)xTy zZX6ZDz!f#&Z66+&c9Tf~19H7PdT-!Gi&85(bWdyAdj0lWSS z$cN$GUc+m?r9x3d29>tHD~`NzQY8n&s(pdVRcqvxxhDlL=c$+*l`4?ly%JJg)D8K#vnZIpPiMq3U@ff%+(~IH|*&B=7B9 zn9*7(`czDu7R7;5M1^){G>>d_RWW!F4^5pRM%vlC5I!oW9$4~?z2M*c5sj(RU!g+6 zPw8dUHcLvuq0ndJQNv-|c?5W*`qelQXw};q0Siw+va3?7KZ_@0%gNCZNQ^D~T=~Sw k4!kAC=Z9ml7KV*2nD58!nC^@NgTCK-ps_C1&S7c)1NGl#umAu6 literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-hdpi/ic_launcher_round.png b/common/src/main/res/mipmap-hdpi/ic_launcher_round.png new file mode 100644 index 0000000000000000000000000000000000000000..b9420ff8f6a0da31448db2ce231af69521afab0b GIT binary patch literal 4395 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 literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-ldpi/ic_launcher.png b/common/src/main/res/mipmap-ldpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..539943f2ddbe204f441dd4de99bcaaee023678bd GIT binary patch literal 2188 zcmV;72y^#|P)(^b8FWQhbW?9;ba!ELWdL_~cP?peYja~^ zaAhuUa%Y?FJQ@H12nk6y$L1QYfs`bI zHdWP<)~ecO&TuLyu2{!n) zUhl=TyI$kn*_k8Z)QcFd;tLa< zUhV;Js3DiCu=1%C+Cn--@?6{(j-}#XI#6S+RwXfM`y(BWr+2jRh69g4DaZx6RFK4{ zKbiu{bpeg5Ag3%=C`qhp&@V=d3s@wmo6#=|h`eQ5rK&kPx3xG9?_S62AKL&~aR3ru z*mB0Ex7Dl!6CjT*Sd%ZrZC5@G=(@RNrj`V%t8|vhL@hhol_T3*c-_7&kW>f2S?2sP z0?8a$RR)$!QGsAoG2|E53!Ho+99tY}F(9h1$**y${J^gD%8@D8_)w zV(?^&icFhTj|4I9QpvARq<1PUP<@SBqd1hkd)CW`cDAv)C$}!&nft{EBynI(1(+o$ zPnM~AILM>@@l*Q`Cg2HsEG;qID(%c_wE3`7s(6X&n`rgA$Y|Xwc;GAWK zxps^2^a&2jzUu}98mld|ctu*LGzH>`jgU-}yow}sp8HJc@p^&2zNw)F4mH+lRjR|$ zxvN7tvb#g7-@9S)HrY0C@HmGreD7}|fEy!(1G@w$O%)M{(cLM5Sb+(&js?l`U3Tsb7*>@f>9E9SS+77#`Xc;^!`KD=yR`A?TP3d{H>FK|-% z+8XtRZ7qD=6Puto)rHOo0sQcn^V_prSvD}H6ZUVJ4@5NG)s;oykvJ;~$3%G`ka(XJ z7HS9TtEu*>-B9yf=`%BB0K>uTVP^ENtAxqK!tI=c@?5}o_CT$#B5!V<*2&5}Y+Zd7 zl~`8#dlLXS72uhtvh39y{kYSHOHKcu#6hM7G}W@=@)fL&88$VnsQ`0kD45Z=cG_r7 zb2%@i%?$$IeRT1Y(G==M)i8S@OU< za619M@O%USj09od7#35vN3H8g3%9vHlo-s*QRwO|X@|EAa$zpGQ@O`7^QCPy@Y-3H z^~4AOzdmfEym%o@I0jt4iPVd4u~>8>b^O2ABY))ppyo=WDi=||&KfI<$>~(UT`K5q zApt#7DIDs#GOk!B>+%3HXY!X-?M|mLG#5}Jo)-!FiQdffIpU+ zkW+~+Rgg@Hut3*sGG6)%^uaOn#OO`)<=Yd+f&f1-5CDuD#vLtJ zx}k9j^^a1$EJKqRan%6=1T33yCkgcXgngkKQ-3-qhV-a;G#WEr?FmcA6W98gHxvVlOoB54_k|(OU!%ka7oorVu{{uqSjT)`2@Wg(K$su_^n~Xe53q6w3aU-v0ptfG>7p>nP#? O0000(^b8FWQhbW?9;ba!ELWdL_~cP?peYja~^ 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 M07*qoM6N<$f*L2W{{R30 literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-mdpi/ic_launcher_foreground.png b/common/src/main/res/mipmap-mdpi/ic_launcher_foreground.png new file mode 100644 index 0000000000000000000000000000000000000000..5b0281abe5036a72acf571481f9c1b27e01bb628 GIT binary patch literal 765 zcmeAS@N?(olHy`uVBq!ia0vp^IUvlz1|<8_!p|}=Fdgx9aSW-r_4fAe><~wh;~#rj zO!gfpEq}jn^MkM?MX{gkNAx1Py2RXA;uW~M%A%Hv#m_zB#j;U&%7R51Aua)1dKQE! zMg34}E&N)Zu>JDuJK7IjaBu0%D3HI8$ar0!dst>hdLc0VN7>bb+}gNtVUw@Xe4oBs0Nq22|TS4D4) zdfT_IJoI}m*Ok}qyUso1IK67Q)$;Ea>vvtrFj@EN>)C(R_QAQ!K3lGs_}G&9=k-}{ zORT&-6mFN8FFFus$(mL=M`V+4>vEaHn||HN+WJaG(4jYKaobr-_B2yhDaI`VJLDWA z!t6~<`zLyIWo_MMHfaOf*8^tHRVM%YX(}6-V9q5YwcLAB%3P_hwoM!CGOOfNCvFHV zJ7#T=tNm?~%a3K1TW5VcXt?2aOU7PH(ZSxV$;|KI6Ik@O7=cr4{a^-Cdm0A8PMIR)`%HN7y n@VDngUD%P8teD9NC?e1r|IF{LbC-+dWRSe4tDnm{r-UW|z`R*M literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-mdpi/ic_launcher_round.png b/common/src/main/res/mipmap-mdpi/ic_launcher_round.png new file mode 100644 index 0000000000000000000000000000000000000000..51734f576fd89128ccd37da8c4442278aaa0ad95 GIT binary patch literal 2851 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 BZYcl& literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-xhdpi/ic_launcher.png b/common/src/main/res/mipmap-xhdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..7a85b3ca0610e19ea5f313872dc9aa17a4f0b1b9 GIT binary patch literal 6561 zcmV;S8D8dzP)$Nh?)OA50{Woz($P5;WTbI1h})ZK#7@x4}_#v6OPXQnW%TiU=W0 z2$%$b6>Lu_TMEuf;scK7y=6%CT2Wg{Z4;!h(Y6zm<#kIwRx&WO24 zfY#j}TVrbU(UL1~3R!ZI6`*K$E5wpc#>#HcqITx61u``M2W1z?7QwwkMDGoN_w^I{ zMxjbYP8V(OkWSuHE|;J09n9~+9FvB*Mu42(UboHEwHr&WyrDiKn%f$n-?ELdtc$U* z4QQ$ZJO!BgsXw$GKnh?y4ICN*_V)u%{6b(XiE=Wh;+AxOQufNv^bO|zA8^8U%sS+3 z58xzN-EGlp48!=U?P*sm?i9_Jtbq1MpHjGJ8L*@ihy?vb*3K|$O#(syp#U(i321Ku z)+}Tc%aE=oLh-a%VrXJz#L`DYQF}0(g){S0{2=B80c~yJ=2+Ca+0gU{K-#i(iz#;9 zW)^>RJ)^r7Xovzt{@?jP1PB76L7*uLY+eS~4n*0dU_31vg%B6R(+gHuAIRm(h1$>V z+?aI&Ry8B+Qt(E@2s{WWJ1$*Cq0e5e=(^27OYD7BGnvJ&pz^Gt1Hk~$(+;%OGi8dx zIF=A$DYeaTd3c^xIy7EDx%#u3JZ6=Eo_4FNAR;%JhVf;oiMGqvz`E&5MVGAxV&OBZ znhS*>pTyP%AZjrcY|@4jWRegYN^BGw!icAGZfg2HO%1a~Kt~4}T`&A;T{9lkbkVeZ zBU^uPEz>2-fw~YNt2i<#fPewaYXsVxnDQm)ePbfXB(Ly1nP?8l!^wO(U%ehLU~IF| zX?wfXCA{!0rlH>(2*~;?*TMYEcBZvGz7Z9o`h5}sx&#^`Kx+)h6iE(`3klF#2b>N? zXegU?Gu7|;+?a6!dfKfn=D_WSVcus2M8nma*|>Ep)B44=Tl@-$0A2HqdTTwC?LZ8~ zh3P4IF?&>Oj?&TO43jXU1T1a~H#=n9tZU|%gO-SG-w5LipI}-szxIn?L4}0ID6nWA zD48QUm=K2Ni4CqoqA8^H&ZIh;F#@_<0}EXdxYf}0dn`*dY+n!KrmGY!pAT38-&oER zh@&KiH){`9SLweTkytbVk|R4z~Z)Wvnv9(Xu9z=Dg}6e0V^WWe#{M;KzPF(?nE@=nbqq2>fEYPaS)WZZ1%#ytU( z>b9kOj2FK-nUR!30PtvDO8$^&yO=^R;zRYf3;@SFoK`;C88* ze{NaSaNTA$ZrjeZdg1Kt{L@iUN3k%_(ZpmskOOgHIG)&|nDWh0>aC?=t{DM6t>G4i zu|w1KdqbA2-?kpc=RVG~{J7e=kjS>ys7OLA3@mH|k~xw?3DR68*18@|wAG9L@r+ZG z1D=`?u%Iolz#;Q9hOXZi42Xv98+?PiqI>Qweg!8Ca{y?LF%`=Yg9)L#N^G=UD%S;h zD3kY+)$X_I1T1a~H@jrqA`SC_U_i#M-pIxuev)bV9A5kiDkL;SfF&KEbRP0pT$+wY z7c+~o#-Kb>ZNpqu0v5Cd7Pw^GqU+|}mPxT~8(`eB)mJ-~IX86v`A{JtXaY@jpnQqM zV1lHl#5$LeuM6^6H6@{H1oX6qTO2Y!BX#qEV1Q!RUBu=eT+6g-fj^lK!{S$PvLtl3 zF%`>@{bRy(l(>u`)AQ=Z;cD50su0lC6>Mi1I}BaFFKEfw)(tRjzm{ovm%sRPS?!z+ z6%wLBprZ-M6i5t>3*jkRW4lN$rTJ$eOV)4S2;&Q%WLowi z-ub8FWOcN-6O_vOxg*CD8x>Pxx`tVs>JPhG%pRMJ8+57PWmy#6x}NnNS1Rgm^$l(< z7XO!T6aM*?b2og$WsHwq%vJ4|$e&YZ6R_hdMGT?8{H6%!3dTJSlh)DXK0TPQ4@{-) zDG{)^E!=FAaf7BC4+cyGuHD4?9Uo;{G_Q6$zi0#Z|3G3Oe(vX9-v?a2+C?N--To^i ztXT-ug%n9a#`C*mQ&x$udH@!-X#agGwMQ4Mf_mDmE|<(5x~|`A1w`G}4bX49`nbWZ zp`ZWlCk0-5^Zbn$Y`@m#BWtQF1;DAfqqYW6wm@QNoQRp$I*e>X7$X;Yym>(cbaq-D ztOB=dy76bhfM~dO6C0o2&a`?#Z5O|HgmBk)B|J5`=ifR^_`S;*_2KG|rD7B6LO^?? zf2h+R7pCKhOFgDSlYgi)Y42##1oX6qTR`Sbnqho7WQqE18(@6?Bd)smaMyG6B}Rn#LVpcq78iETN0TmmQU?DD)9SPtC}5kRwSV@3iPxyrSg#d zaT#zGUCJJfghlCKwv6Jr+BhcxJ?&PP%jOPEH|_}pL~QE@7`JY%&S6T?28PCgbpFhL zi45U;PYe9=?WuP%o&lO-!1|u|RgvQYxgsH+2F6mrP=cmagLHBhwJ{3HmPi~+5P{?x zC5&`ijE2TD?pf0oXPsSV6b%$=xI@?U`^^AFuiFgsw(SZlx~t=Ae(_O>r+;1hNG)Ih z4G~7ZW*c`Qz*5VY0{Xk+0z`^f24 z*&iUls;|4kkn(5c=H5nb!9JVXHbR<;R{Cc>2|u+5H!S{R4!LtY&mHO|P9w z-BD8%nAgZOk|eD!PL?idA+vEXowa*Ue*P2!x?01l1BU*kveGYIzXXvRE@yG+3coJD z8lJ@P1mTbWPQsZ}cU0cT-af)@+j-h5mI%-^KZ{BP8qbhEm=F;|a;z>a4yLlsSmoIh z1ax#n8l}LZvKRQ`qIPQiHn=k*G0jWBgxEP&L0UnxHX5C9M{ zqE{)>wiqVWf9xXW`iSptp3QBW3cj?1VN{XOTB5a~dTlZLu}CBcT(gnc(5X&AyCgvM z+W?<{pvQ8hkg~I*3AA}RQ+uQT<7%cEx?36l?xWKSc4h*bmNIU?rh4P7tYKR{uyH9< zMZH%=&wVC8}uwe<~b04d%ij)$# zd<`RFK~o+qK~TaaV}%gZq!?s7sJWpO0{F{2)SRlAFf`zyJH4vKJLl@BWCH+5Xw+0@ zS|x;3AqEM{^dFdsYeJirGCq09+%d>^->k5-v-bT6;k%^)17IcrP*7-06GCv_)Ilxv zfSMugu(oGr`k4%(p_%E28ECBs3>_rE1w&U!Q6^8h9z?+g$CESj_KtL(@csWYccS8N z{8%8Fo0+~wlR&-*b`_{>tAYR|Gy%#M!9(NziBUBJuruM2r)F-#lR+v6Jo?kxdLPXG z+76Ec`4U(P2!@nGBOGO=`aKV7JVVHrYJ1MxHwygI3v12(2xX`GQos!S)&CGRw=dJh*B+;;ODMDdTD8~r z6Am1MD@ctwu6lE1B$p5Xm^}NiQr^!evJk&{m(;J{^tRwFIXk8BUN|~ZItrY0BpJ$-`b5}VD23>G zr%y!Mn?SAgI8mxGUC$_B_%BZS#lOyNDfrYD#=rmM^rQ63#vM%f(*Ixog`hldM7iEG z&S2^vDx;#R0C@kF`nBcFh1jN<(V9G4S4Jx zg;*6NmB_E~d+QMJ_dgTJ|N z8l!%$e7X<<=xJv>^~<>v1)q#NuUEK!OLf#wItT394}9-wfw2@9@z7zVcaxiL%U}eI6DivnA*rXG@z9+-sdwl ze1GhmOW4v9YK@w5hYRyhVqv*t(=yin!|ySccKU{Sx{`1{y!Zy;4J*qAAhF`gy4d_J=#K&DY!0(#p0%H^XYQ|n}15AYYadmk91spEu{1&rz2 zwoCySO@e>$tf1HTQ#q0131a7a%Cn#9A1j_S#dqHAmCU-Cx`>tpseMITY)qsn)H~v< zqp$_iWo=1Vy^!&vmkEk5ypu=%#KZiS8Vu?ACRMh#8}Q2Tar04<0+C`Rxtx0qY9qG+0YpEQ@FbiT^Iq|)+>DR z#@Z&H@+F{TgCBlM(9XSZ6KPHf<$T?A?PrgS6^=}%jY%umRKdyBMPyNcM@nU}Dw!ke zzyQfb?Tk>sf0uu)Bw@o6;D7yGpg765JPQH*{g-N2gTxHviolU!@Hc-Vuxmfc;~8(1 z*nY%d_r-(p!r{rbbwL&`ohv6AqcrG(59CYo@&HcIqf#_(uPg-IHtl z#w!$VzqU5JpV`0PmMMTAenQ~2z3|4Q*>|!LdZl*34C`xd) zRF-Qpc{2OPNL$>_Y#Ja_uMG*S7c!pTMd!U)qiewT@A9gWPBQXRfLK?yv0pzA@k6zIlM%WSWQAD?ccNxAW*wcK=k`oz^mSc3K@m znL7aUfrc2xKC&9dJ+~^Z3xTF{;cYG=TL3b7@Iy}sdi5Q4#?wd%W&bBh_LE13iv3e< zcUn+dh9mieN8XD%A0<-V!^bUg`{)xbGwKFVK`5{fw zi$_MX2dC5iv`?YaxpE>>M}tC%BL!Ptk;qVBG)dyBbqq%BHS$^7Fle*(tpp>rAL>AxtJ?CmSOoD(-+TPaF8__)T|aSQ*g5N5u+$25=EO(UaCm~mMavk5Ha84& z#o(s$euVp*J@67K6=zlYbN2AU(b4RCwd}vSw4JIC>j^OLDVD`W@ic{w zj*?upfDsPNjwDoaJIBU-SM#;K>?V>t46pnbg1i@chibd?E10pbD5bLH@o2pmmx@w_ zlDs69gE^KGa`RGVB3}|_61gIfEr1_+O3-V2*&RrB^15fUxoC-gU56GgM*}ZHG?e^WYwVBf*D9=fkT7fM}8`3_d%3W z86J~d{*DmtD@TU%z2|SG`k1w^sb(_vL_?H@JP*4|4s97tiO}I8l8ZY5%k)cLs`aiS zrGQi(I6Uk}xNjanIi2&4v0J_iT>8aSPO_?nS^b`HGV7)rLo&sJ@O*q`PIO%d!8>5%narP0RWHz+-S@!3fFy0lhl|kP;@*iLhtmFa9qIg{D4id zUu3DBR!Cba`=?jQtY0Ae@*BjDKQFMi4^ATG4Qu57Qn}vOS;wBiL3_?z5t$?Duv1_x zXp6P$CO&Js`p1KoTvv8rZCVEEZe?1(n9<$pOGKscYBcGmkw+(hUGEWkhlzi80BlIk z7far5gPi9b7ysNhnjQo&$?~q5!CWK&0H`B^1~U>_p-BHck=*D(E)H8vu?TcQqzOT2 zj4&|+6Irlg0L*SO3nyD3mgF4(yj_~({$rt#IW(BX1kO^SJe!!S1ONa!0n2F(M}nHX zN`bmGDYmFG#T`ezASCydp?+=+W^;!Cd$#9tHq2=P zPDXt_>gue>JWXZCZHle%lxjC5msnAmL16`NG=SV*hr&&nQf?@j#I!%MHWT>&`C|d! TJbm^R00000NkvXXu0mjfDIRMX literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png b/common/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png new file mode 100644 index 0000000000000000000000000000000000000000..db0d44b3fdfa83d165b704ead8dba6683c695c46 GIT binary patch literal 1575 zcmbVMi#rnv6!+y8%1C7jQ!4Uk%HwjA?G|lokw;TP3wey0B@9_b(o*w`FvX~tM`O%J zQ_^F)>1*@YW)su6H!&8sJCj#;{R^G%JLi1o`_B2D?>pyr&TU_B1Yo1_Mhy)O0P^^; zQ)(~%soERV_VzCDtcJ#V1@hQY|J32xQDSyYxT)4$OKf24(+bHU439!_j71|(-PqDt z*oHUvTlps+ywN(nWA9MGGAFLJBz+}J=ccxWdBQt58Ri_kTeU6W#cHLpiIt2IRI08H zEnI;rV|fqa+9i{#S0^v?iI=5aP<7v5lMGWYFE5-V+9|4LI=S<(uDv8gHU zu`vh)I&PRQy`%bAz4fa;Z3wPk%Z6n|#t6Qm1Uria%?4}_fVtQ1bqV71tS>LHlEuYE z-90)#98we`^gX*SDmOO?g;;$=z{bPdnM5Le4!9H= zekTyu;)h9GUU7N=_27CXg*nZ^=9li)y`NaN9bax7R|@@x;Akq(lS_2r%7Y&PVe|&u zb6f*1bwd7Pbu`*)fInFsF~s{Ykco8_&89O8{&r;mQJMKQo?QU&Dsp zD~Avl?E0g}Fvg6KV7kbhlVPKeb1RAI5Sc2{g@6iwR&44yo@@yL?gJEqn&RZp6y={y z`Gf9moXmm8Ma`KkZ#2!;3U);y29I`I`OE!X`-VGQaB|epe)hq3 zTjznGy=*tuag_&b@M;ON;~KX8qt@`W079;)M0Vquta@~iflE;a#OTN96#EaBwb|LV zYjEmIZ?>>LaiMp%A@6i9JiG+#1Ck9NGFH6T<=;SK<*2tWb8k{Aqt~J$A|m{?d@1S` zp&dgsCZ$k$ysu0lyFm#dM}NmP6qkKxQN2)Rdq4%*-&8V#2 zecOmxq>f{%JR-B`no4l)f180_U^xp@Y6k5 z1v&y^iZgXF&4akh(+-ew1uJLW0-TDmN6{8@p$8zmx>5X#9U&n;*KHL$h}Ef0-v<4B zfcoUry;7o8mz0X{%uh?%`e6Gt>uK&d<_dk0Qff4*8sIY>=w^n>Fb=5Z?9;_>N#LlJ ziaoeX&a`+%DQ}-}JA=05j0fk&Ol4w%b3V)`j-Y?9+%6rc+dDN= z+Q;Kp3%v8x*ISHN)h%Y++V`knb$>DrCR=iyB^ie$xPV{HWwHLQ-mNAYv2McCI@;Q8>^{^!rC46{|8cvA6* SjLzl!jFB+!V@>X%ng0X4J?f_b literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-xhdpi/ic_launcher_round.png b/common/src/main/res/mipmap-xhdpi/ic_launcher_round.png new file mode 100644 index 0000000000000000000000000000000000000000..1bc2989e3d782ff4f3b7e18752f46548a3d668df GIT binary patch literal 6149 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 literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-xxhdpi/ic_launcher_round.png b/common/src/main/res/mipmap-xxhdpi/ic_launcher_round.png new file mode 100644 index 0000000000000000000000000000000000000000..a420036970d6fdbb19681aa4101cdca8ad932a11 GIT binary patch literal 10169 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

    a7w;c>6khKI(X)D(hTB*(1bg(#FKLh)aBLJ}5u z`NRooEFsannx!nDDP$J}r&?$e%WAK7OYC*4F^m4jVNX-@-IJ)9Kjn2Q<7v<6RI|%eB(G1Uu<08a|sLPwY-r8^m@Fu{_>hXb(+7Q>-S}r nS!S7KmRV+*WtLfHnd$joE>zfk2@$pZ00000NkvXXu0mjf>EGz} literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/common/src/main/res/mipmap-xxxhdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..34dbb55f89f3a9a31924fe988deb94e6a77554dc GIT binary patch literal 14618 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 literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png b/common/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png new file mode 100644 index 0000000000000000000000000000000000000000..9cc33e82424bfb37398a7ea3f23f9b7e8910fc96 GIT binary patch literal 3701 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 literal 0 HcmV?d00001 diff --git a/common/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png b/common/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png new file mode 100644 index 0000000000000000000000000000000000000000..5b319d0c525925f9b95c02a64af155841c80b740 GIT binary patch literal 14768 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 literal 0 HcmV?d00001 diff --git a/common/src/main/res/raw/android_sdk.txt b/common/src/main/res/raw/android_sdk.txt new file mode 100644 index 00000000..06270f0e --- /dev/null +++ b/common/src/main/res/raw/android_sdk.txt @@ -0,0 +1,142 @@ +This is the Android Software Development Kit License Agreement + +1. Introduction + +1.1 The Android Software Development Kit (referred to in this License Agreement as the "SDK" and specifically including the Android system files, packaged APIs, and Google APIs add-ons) is licensed to you subject to the terms of this License Agreement. This License Agreement forms a legally binding contract between you and Google in relation to your use of the SDK. + +1.2 “Android” means the Android software stack for devices, as made available under the Android Open Source Project, which is located at the following URL: http://source.android.com/, as updated from time to time. + +1.3 A "compatible implementation" means any Android device that (i) complies with the Android Compatibility Definition document, which can be found at the Android compatibility website (http://source.android.com/compatibility) and which may be updated from time to time; and (ii) successfully passes the Android Compatibility Test Suite (CTS). + +1.4 "Google" means Google Inc., a Delaware corporation with principal place of business at 1600 Amphitheatre Parkway, Mountain View, CA 94043, United States. + + +2. Accepting this License Agreement + +2.1 In order to use the SDK, you must first agree to this License Agreement. You may not use the SDK if you do not accept this License Agreement. + +2.2 By clicking to accept, you hereby agree to the terms of this License Agreement. + +2.3 You may not use the SDK and may not accept the License Agreement if you are a person barred from receiving the SDK under the laws of the United States or other countries including the country in which you are resident or from which you use the SDK. + +2.4 If you are agreeing to be bound by this License Agreement on behalf of your employer or other entity, you represent and warrant that you have full legal authority to bind your employer or such entity to this License Agreement. If you do not have the requisite authority, you may not accept the License Agreement or use the SDK on behalf of your employer or other entity. + + +3. SDK License from Google + +3.1 Subject to the terms of this License Agreement, Google grants you a limited, worldwide, royalty-free, non-assignable, non-exclusive, and non-sublicensable license to use the SDK solely to develop applications to run on the Android platform. + +3.2 You may not use this SDK to develop applications for other platforms (including non-compatible implementations of Android) or to develop another SDK. You are of course free to develop applications for other platforms, including non-compatible implementations of Android, provided that this SDK is not used for that purpose. + +3.3 You agree that Google or third parties own all legal right, title and interest in and to the SDK, including any Intellectual Property Rights that subsist in the SDK. "Intellectual Property Rights" means any and all rights under patent law, copyright law, trade secret law, trademark law, and any and all other proprietary rights. Google reserves all rights not expressly granted to you. + +3.4 You may not use the SDK for any purpose not expressly permitted by this License Agreement. Except to the extent required by applicable third party licenses, you may not: (a) copy (except for backup purposes), modify, adapt, redistribute, decompile, reverse engineer, disassemble, or create derivative works of the SDK or any part of the SDK; or (b) load any part of the SDK onto a mobile handset or any other hardware device except a personal computer, combine any part of the SDK with other software, or distribute any software or device incorporating a part of the SDK. + +3.5 Use, reproduction and distribution of components of the SDK licensed under an open source software license are governed solely by the terms of that open source software license and not this License Agreement. + +3.6 You agree that the form and nature of the SDK that Google provides may change without prior notice to you and that future versions of the SDK may be incompatible with applications developed on previous versions of the SDK. You agree that Google may stop (permanently or temporarily) providing the SDK (or any features within the SDK) to you or to users generally at Google's sole discretion, without prior notice to you. + +3.7 Nothing in this License Agreement gives you a right to use any of Google's trade names, trademarks, service marks, logos, domain names, or other distinctive brand features. + +3.8 You agree that you will not remove, obscure, or alter any proprietary rights notices (including copyright and trademark notices) that may be affixed to or contained within the SDK. + + +4. Use of the SDK by You + +4.1 Google agrees that it obtains no right, title or interest from you (or your licensors) under this License Agreement in or to any software applications that you develop using the SDK, including any intellectual property rights that subsist in those applications. + +4.2 You agree to use the SDK and write applications only for purposes that are permitted by (a) this License Agreement and (b) any applicable law, regulation or generally accepted practices or guidelines in the relevant jurisdictions (including any laws regarding the export of data or software to and from the United States or other relevant countries). + +4.3 You agree that if you use the SDK to develop applications for general public users, you will protect the privacy and legal rights of those users. If the users provide you with user names, passwords, or other login information or personal information, you must make the users aware that the information will be available to your application, and you must provide legally adequate privacy notice and protection for those users. If your application stores personal or sensitive information provided by users, it must do so securely. If the user provides your application with Google Account information, your application may only use that information to access the user's Google Account when, and for the limited purposes for which, the user has given you permission to do so. + +4.4 You agree that you will not engage in any activity with the SDK, including the development or distribution of an application, that interferes with, disrupts, damages, or accesses in an unauthorized manner the servers, networks, or other properties or services of any third party including, but not limited to, Google or any mobile communications carrier. + +4.5 You agree that you are solely responsible for (and that Google has no responsibility to you or to any third party for) any data, content, or resources that you create, transmit or display through Android and/or applications for Android, and for the consequences of your actions (including any loss or damage which Google may suffer) by doing so. + +4.6 You agree that you are solely responsible for (and that Google has no responsibility to you or to any third party for) any breach of your obligations under this License Agreement, any applicable third party contract or Terms of Service, or any applicable law or regulation, and for the consequences (including any loss or damage which Google or any third party may suffer) of any such breach. + + +5. Your Developer Credentials + +5.1 You agree that you are responsible for maintaining the confidentiality of any developer credentials that may be issued to you by Google or which you may choose yourself and that you will be solely responsible for all applications that are developed under your developer credentials. + + +6. Privacy and Information + +6.1 In order to continually innovate and improve the SDK, Google may collect certain usage statistics from the software including but not limited to a unique identifier, associated IP address, version number of the software, and information on which tools and/or services in the SDK are being used and how they are being used. Before any of this information is collected, the SDK will notify you and seek your consent. If you withhold consent, the information will not be collected. + +6.2 The data collected is examined in the aggregate to improve the SDK and is maintained in accordance with Google's Privacy Policy. + + +7. Third Party Applications + +7.1 If you use the SDK to run applications developed by a third party or that access data, content or resources provided by a third party, you agree that Google is not responsible for those applications, data, content, or resources. You understand that all data, content or resources which you may access through such third party applications are the sole responsibility of the person from which they originated and that Google is not liable for any loss or damage that you may experience as a result of the use or access of any of those third party applications, data, content, or resources. + +7.2 You should be aware the data, content, and resources presented to you through such a third party application may be protected by intellectual property rights which are owned by the providers (or by other persons or companies on their behalf). You may not modify, rent, lease, loan, sell, distribute or create derivative works based on these data, content, or resources (either in whole or in part) unless you have been specifically given permission to do so by the relevant owners. + +7.3 You acknowledge that your use of such third party applications, data, content, or resources may be subject to separate terms between you and the relevant third party. In that case, this License Agreement does not affect your legal relationship with these third parties. + + +8. Using Android APIs + +8.1 Google Data APIs + +8.1.1 If you use any API to retrieve data from Google, you acknowledge that the data may be protected by intellectual property rights which are owned by Google or those parties that provide the data (or by other persons or companies on their behalf). Your use of any such API may be subject to additional Terms of Service. You may not modify, rent, lease, loan, sell, distribute or create derivative works based on this data (either in whole or in part) unless allowed by the relevant Terms of Service. + +8.1.2 If you use any API to retrieve a user's data from Google, you acknowledge and agree that you shall retrieve data only with the user's explicit consent and only when, and for the limited purposes for which, the user has given you permission to do so. + + +9. Terminating this License Agreement + +9.1 This License Agreement will continue to apply until terminated by either you or Google as set out below. + +9.2 If you want to terminate this License Agreement, you may do so by ceasing your use of the SDK and any relevant developer credentials. + +9.3 Google may at any time, terminate this License Agreement with you if: +(A) you have breached any provision of this License Agreement; or +(B) Google is required to do so by law; or +(C) the partner with whom Google offered certain parts of SDK (such as APIs) to you has terminated its relationship with Google or ceased to offer certain parts of the SDK to you; or +(D) Google decides to no longer provide the SDK or certain parts of the SDK to users in the country in which you are resident or from which you use the service, or the provision of the SDK or certain SDK services to you by Google is, in Google's sole discretion, no longer commercially viable. + +9.4 When this License Agreement comes to an end, all of the legal rights, obligations and liabilities that you and Google have benefited from, been subject to (or which have accrued over time whilst this License Agreement has been in force) or which are expressed to continue indefinitely, shall be unaffected by this cessation, and the provisions of paragraph 14.7 shall continue to apply to such rights, obligations and liabilities indefinitely. + + +10. DISCLAIMER OF WARRANTIES + +10.1 YOU EXPRESSLY UNDERSTAND AND AGREE THAT YOUR USE OF THE SDK IS AT YOUR SOLE RISK AND THAT THE SDK IS PROVIDED "AS IS" AND "AS AVAILABLE" WITHOUT WARRANTY OF ANY KIND FROM GOOGLE. + +10.2 YOUR USE OF THE SDK AND ANY MATERIAL DOWNLOADED OR OTHERWISE OBTAINED THROUGH THE USE OF THE SDK IS AT YOUR OWN DISCRETION AND RISK AND YOU ARE SOLELY RESPONSIBLE FOR ANY DAMAGE TO YOUR COMPUTER SYSTEM OR OTHER DEVICE OR LOSS OF DATA THAT RESULTS FROM SUCH USE. + +10.3 GOOGLE FURTHER EXPRESSLY DISCLAIMS ALL WARRANTIES AND CONDITIONS OF ANY KIND, WHETHER EXPRESS OR IMPLIED, INCLUDING, BUT NOT LIMITED TO THE IMPLIED WARRANTIES AND CONDITIONS OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. + + +11. LIMITATION OF LIABILITY + +11.1 YOU EXPRESSLY UNDERSTAND AND AGREE THAT GOOGLE, ITS SUBSIDIARIES AND AFFILIATES, AND ITS LICENSORS SHALL NOT BE LIABLE TO YOU UNDER ANY THEORY OF LIABILITY FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, CONSEQUENTIAL OR EXEMPLARY DAMAGES THAT MAY BE INCURRED BY YOU, INCLUDING ANY LOSS OF DATA, WHETHER OR NOT GOOGLE OR ITS REPRESENTATIVES HAVE BEEN ADVISED OF OR SHOULD HAVE BEEN AWARE OF THE POSSIBILITY OF ANY SUCH LOSSES ARISING. + + +12. Indemnification + +12.1 To the maximum extent permitted by law, you agree to defend, indemnify and hold harmless Google, its affiliates and their respective directors, officers, employees and agents from and against any and all claims, actions, suits or proceedings, as well as any and all losses, liabilities, damages, costs and expenses (including reasonable attorneys fees) arising out of or accruing from (a) your use of the SDK, (b) any application you develop on the SDK that infringes any copyright, trademark, trade secret, trade dress, patent or other intellectual property right of any person or defames any person or violates their rights of publicity or privacy, and (c) any non-compliance by you with this License Agreement. + + +13. Changes to the License Agreement + +13.1 Google may make changes to the License Agreement as it distributes new versions of the SDK. When these changes are made, Google will make a new version of the License Agreement available on the website where the SDK is made available. + + +14. General Legal Terms + +14.1 This License Agreement constitutes the whole legal agreement between you and Google and governs your use of the SDK (excluding any services which Google may provide to you under a separate written agreement), and completely replaces any prior agreements between you and Google in relation to the SDK. + +14.2 You agree that if Google does not exercise or enforce any legal right or remedy which is contained in this License Agreement (or which Google has the benefit of under any applicable law), this will not be taken to be a formal waiver of Google's rights and that those rights or remedies will still be available to Google. + +14.3 If any court of law, having the jurisdiction to decide on this matter, rules that any provision of this License Agreement is invalid, then that provision will be removed from this License Agreement without affecting the rest of this License Agreement. The remaining provisions of this License Agreement will continue to be valid and enforceable. + +14.4 You acknowledge and agree that each member of the group of companies of which Google is the parent shall be third party beneficiaries to this License Agreement and that such other companies shall be entitled to directly enforce, and rely upon, any provision of this License Agreement that confers a benefit on (or rights in favor of) them. Other than this, no other person or company shall be third party beneficiaries to this License Agreement. + +14.5 EXPORT RESTRICTIONS. THE SDK IS SUBJECT TO UNITED STATES EXPORT LAWS AND REGULATIONS. YOU MUST COMPLY WITH ALL DOMESTIC AND INTERNATIONAL EXPORT LAWS AND REGULATIONS THAT APPLY TO THE SDK. THESE LAWS INCLUDE RESTRICTIONS ON DESTINATIONS, END USERS AND END USE. + +14.6 The rights granted in this License Agreement may not be assigned or transferred by either you or Google without the prior written approval of the other party. Neither you nor Google shall be permitted to delegate their responsibilities or obligations under this License Agreement without the prior written approval of the other party. + +14.7 This License Agreement, and your relationship with Google under this License Agreement, shall be governed by the laws of the State of California without regard to its conflict of laws provisions. You and Google agree to submit to the exclusive jurisdiction of the courts located within the county of Santa Clara, California to resolve any legal matter arising from this License Agreement. Notwithstanding this, you agree that Google shall still be allowed to apply for injunctive remedies (or an equivalent type of urgent legal relief) in any jurisdiction. diff --git a/common/src/main/res/raw/apache2.txt b/common/src/main/res/raw/apache2.txt new file mode 100644 index 00000000..7a4a3ea2 --- /dev/null +++ b/common/src/main/res/raw/apache2.txt @@ -0,0 +1,202 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. \ No newline at end of file diff --git a/common/src/main/res/raw/mine_explosion_sound.ogg b/common/src/main/res/raw/mine_explosion_sound.ogg new file mode 100644 index 0000000000000000000000000000000000000000..a1a571835646826562c8d0565d188b856cd8c828 GIT binary patch literal 14878 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 literal 0 HcmV?d00001 diff --git a/common/src/main/res/raw/mockito.txt b/common/src/main/res/raw/mockito.txt new file mode 100644 index 00000000..9a90317d --- /dev/null +++ b/common/src/main/res/raw/mockito.txt @@ -0,0 +1,9 @@ +The MIT License + +Copyright (c) 2007 Mockito contributors + +Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. \ No newline at end of file diff --git a/common/src/main/res/raw/sounds.txt b/common/src/main/res/raw/sounds.txt new file mode 100644 index 00000000..d05aa6d2 --- /dev/null +++ b/common/src/main/res/raw/sounds.txt @@ -0,0 +1,6 @@ +Mine Explosion + +By shaynecantly +Link https://www.freesound.org/people/shaynecantly/sounds/131555/ +License: Creative Commons 3.0 +License URL: http://creativecommons.org/licenses/by/3.0/ diff --git a/common/src/main/res/values-es/strings.xml b/common/src/main/res/values-es/strings.xml new file mode 100644 index 00000000..59d58ef4 --- /dev/null +++ b/common/src/main/res/values-es/strings.xml @@ -0,0 +1,82 @@ + + + Anti-Mine + Deshabilitar las minas ocultas del campo de minas. + Minas Restantes + Dificultad + Estándar + Principiante + Intermediario + Especialista + Abrir Menú + Cerrar Menú + Ajustes + Animaciones + Respuesta táctil + Acerca de… + Sin conexión a Internet. + Estadística + Nuevo juego + Si iniciar un nuevo juego\nse perderá su progreso actual. + Cancelar + Continuar + Mostrar Licencias + Licencias + Google Play Games + Tablas de Clasificación + Conectar + Conectando… + Desconectar + Desconectado + Nuevo juego + ¿Quieres iniciar un nuevo juego? + + General + Todas las minas fueron desactivadas. + %d minas + Tiempo de juego + Campo Cubierto + Mina + Mina despiezada + El juego comienza ahora + ¡Usted ha estallado una mina! + Bandera colocada! + Bandera retirada! + Este juego utiliza el siguiente software: + No se ha podido iniciar sesión. Por favor, compruebe la conexión de red y vuelve a intentarlo. + Error desconocido. + Rever + Vacío + Perderá todos los movimientos en el juego actual. + ¡Ganaste! + ¡Perdiste! + %s - %s + Efectos de Sonido + Dejar + ¿Estás seguro? + + Personalizado + Empezar + Anchura + Altura + Minas + Bandera Automática + + Área cubierta + Área marcada + Área dudosa + Área marcada incorrectamente + + General + Vibra al explotar o al alternar la marcacion + Hace un sonido de explosión + Agrega una marca en minas resueltas automáticamente + Accesibilidad + Use áreas grandes + Aumenta el área táctil + + Nos califica ❤ + Si te gusta este juego, danos un comentario. Nos ayudará mucho. + Si ❤️️️ + No + diff --git a/common/src/main/res/values-night/colors.xml b/common/src/main/res/values-night/colors.xml new file mode 100644 index 00000000..3e1c073c --- /dev/null +++ b/common/src/main/res/values-night/colors.xml @@ -0,0 +1,32 @@ + + + #546E7A + #FFFFFF + + #212121 + #212121 + + #FFFFFF + + #527F8D + + #d5d2cc + #d5d2cc + #d5d2cc + #d5d2cc + #d5d2cc + #d5d2cc + #d5d2cc + #d5d2cc + + #FCC216 + + #FFFFFF + #2B2824 + #FFFFFF + + #757575 + #171717 + #ff424242 + + \ No newline at end of file diff --git a/common/src/main/res/values-pt/strings.xml b/common/src/main/res/values-pt/strings.xml new file mode 100644 index 00000000..261f2b68 --- /dev/null +++ b/common/src/main/res/values-pt/strings.xml @@ -0,0 +1,82 @@ + + + Anti-Mine + Encontre todas as minas escondidas no campo minado. + Minas Restantes + Dificuldade + Padrão + Iniciante + Intermediário + Experiente + Abrir Menu + Fechar Menu + Configurações + Animações + Vibração + Sobre + Sem conexão com a internet. + Estatísticas + Novo Jogo + Se você começar um novo jogo,\no progresso atual será perdido. + Cancelar + Continuar + Mostar Licenças + Licenças + Google Play Games + Placares + Conectar + Conectando… + Desconectar + Desconectado + Novo Jogo + Deseja começar um novo jogo? + Sim + Geral + Todas as minas foram desativadas. + %d minas + Tempo de Jogo + Campo Coberto + Mina + Mina explodida + Jogo iniciado + Você explodiu uma mina! + Bandeira posicionada! + Bandeira removida! + Esse jogo foi desenvolvido utilizando as seguintes bibliotecas: + Falha ao conectar. Por favor, verifique sua conexão com a internet e tente novamente. + Erro desconhecido. + Reiniciar + Vazio + Você perderá todos os movimentos do jogo atual. + Você venceu! + Você perdeu! + %s - %s + Efeitos Sonoros + Sair + Você tem certeza? + + Personalizado + Iniciar + Largura + Altura + Minas + Bandeiras Automáticas + + Área coberta + Área marcada + Área duvidosa + Área marcada erroneamente + + Geral + Vibra ao explodir ou alternar marcação + Faz um som ao explodir + Faz marcação automaticamente em minas resolvidas + Acessibilidade + Usar Área Grande + Aumenta a área tocável + + Avaliar o app ❤ + Se você está gostando do jogo, por favor deixe um comentário! Isso nos ajuda muito. + Sim ❤️️️ + Não + diff --git a/common/src/main/res/values-v21/themes.xml b/common/src/main/res/values-v21/themes.xml new file mode 100644 index 00000000..b4443c1c --- /dev/null +++ b/common/src/main/res/values-v21/themes.xml @@ -0,0 +1,21 @@ + + + + + + diff --git a/common/src/main/res/values-w600dp/dimens.xml b/common/src/main/res/values-w600dp/dimens.xml new file mode 100644 index 00000000..6c68e84e --- /dev/null +++ b/common/src/main/res/values-w600dp/dimens.xml @@ -0,0 +1,4 @@ + + 38dp + 20sp + diff --git a/common/src/main/res/values-zh/strings.xml b/common/src/main/res/values-zh/strings.xml new file mode 100644 index 00000000..f0598e4e --- /dev/null +++ b/common/src/main/res/values-zh/strings.xml @@ -0,0 +1,82 @@ + + + 反雷 - 扫雷 + 你需要清除一个隐藏着地雷的矩形面板,不能使任何地雷爆炸。 + 剩余地雷 + 难度 + 标准 + 初级 + 中级 + 专家 + 打开菜单 + 关闭选单 + 设置 + 动画 + 触觉反馈 + 关于 + 没有网络连接。 + 统计 + 新游戏 + 如果你开始一个新游戏,\n你的当前进度将会丢失。 + 取消 + 恢复 + 显示许可证 + 许可证 + Google Play 游戏 + 排行榜 + 连接 + 正在连接…… + 断开 + 已断开 + 新游戏 + 你要开始一个新游戏吗? + + 常规 + 所有地雷都已失效。 + %d个地雷 + 游戏时间 + 覆盖区域 + 地雷 + 爆炸的地雷 + 游戏已开始 + 你爆炸了一个地雷! + 标记已放置! + 标记已移除! + 该游戏使用以下第三方软件: + 登录失败。请检查你的网络连接,然后重试。 + 未知错误。 + 重试 + + 你将丢失当前游戏中的所有进度。 + 你赢了! + 你输了! + %s - %s + 音效 + 退出 + 确定吗? + + 自定义 + 开始 + 宽度 + 高度 + 地雷 + 游戏助手 + + 覆盖面积 + 标记区域 + 怀疑 + 标记区域错误 + + + + + + 辅助功能 + 使用大面积 + + + 评价应用 + + ️️️ + + diff --git a/common/src/main/res/values/colors.xml b/common/src/main/res/values/colors.xml new file mode 100644 index 00000000..5bc3980b --- /dev/null +++ b/common/src/main/res/values/colors.xml @@ -0,0 +1,32 @@ + + + #546E7A + #FFFFFF + + #212121 + #212121 + + #FFFFFF + + #527F8D + + #527F8D + #2B8D43 + #E65100 + #20A5f7 + #ED1C24 + #FFC107 + #66126B + #000000 + + #FCC216 + + #D32F2F + #2B2824 + #212121 + + #000 + #424242 + #d5d2cc + + \ No newline at end of file diff --git a/common/src/main/res/values/dimens.xml b/common/src/main/res/values/dimens.xml new file mode 100644 index 00000000..48c5c368 --- /dev/null +++ b/common/src/main/res/values/dimens.xml @@ -0,0 +1,11 @@ + + 180dp + 6dp + + 38dp + 48dp + 2dp + 1dp + 5dp + 18sp + diff --git a/common/src/main/res/values/keys.xml b/common/src/main/res/values/keys.xml new file mode 100644 index 00000000..83a9ae87 --- /dev/null +++ b/common/src/main/res/values/keys.xml @@ -0,0 +1,5 @@ + + + {-AMPLITUDE-kEY-} + {-CRASHLYTICS-KEY-} + diff --git a/common/src/main/res/values/strings.xml b/common/src/main/res/values/strings.xml new file mode 100644 index 00000000..b14f2a6e --- /dev/null +++ b/common/src/main/res/values/strings.xml @@ -0,0 +1,82 @@ + + + Anti-Mine + You have to clear a rectangular board containing hidden "mines" without detonating any of them. + Remaining mines + Difficulty + Standard + Beginner + Intermediate + Expert + Open Menu + Close Menu + Settings + Animations + Haptic Feedback + About + No internet connection. + Statistics + New Game + If you start a new game,\nyour current progress will be lost. + Cancel + Resume + Show Licenses + Licenses + Google Play Games + Leaderboards + Connect + Connecting… + Disconnect + Disconnected + New Game + Do you want to start a new game? + Yes + General + All mines were disabled. + %d mines + Game Time + Covered Field + Mine + Exploded Mine + Game Started + You exploded a mine! + Flag placed! + Flag removed! + This game uses the following third parties software: + Failed to sign in. Please check your network connection and try again. + Unknown error. + Retry + Empty + You\'ll lose all moves on current game. + You won! + You lost! + %s - %s + Sound Effects + Quit + Are you sure? + + Custom + Start + Width + Height + Mines + Game Assistant + + Covered area + Marked area + Doubtful area + Wrongly marked area + + General + Vibrates on explosion or flag toggle + Makes a sound of explosion + Adds a flag on resolved mines automatically + Accessibility + Use Large Areas + Increases the touch area + + Feedback ❤ + If you like this game, please give us a feedback. It will help us a lot. + Yes ❤️️️ + No + diff --git a/common/src/main/res/values/themes.xml b/common/src/main/res/values/themes.xml new file mode 100644 index 00000000..6f943afb --- /dev/null +++ b/common/src/main/res/values/themes.xml @@ -0,0 +1,24 @@ + + + + + + + + + + + +