From 09ff1ca5df4e5b77af01b28d19f992bf4a8e2093 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?And=C5=BEej=20Maciusovi=C4=8D?= Date: Tue, 3 Dec 2019 16:52:54 +0200 Subject: [PATCH] Rewrite in native Android (#175) --- .gitlab-ci.yml | 23 -- Gemfile.lock | 2 +- android/app/build.gradle | 167 +++------ .../network.mysterium.db.AppDatabase/1.json | 54 +++ android/app/src/debug/AndroidManifest.xml | 2 +- android/app/src/main/AndroidManifest.xml | 77 +++-- .../java/network/mysterium/AppContainer.kt | 61 ++++ .../java/network/mysterium/MainActivity.kt | 151 ++++++++ .../java/network/mysterium/MainApplication.kt | 54 +++ .../java/network/mysterium/db/AppDatabase.kt | 10 + .../mysterium/db/FavoriteProposalDao.kt | 20 ++ .../java/network/mysterium/db/TermsDao.kt | 20 ++ .../network/mysterium/logging/BugReporter.kt | 23 +- .../mysterium/service/core/DeferredNode.kt | 35 ++ .../core/MysteriumAndroidCoreService.kt | 165 ++++++--- .../service/core/MysteriumCoreService.kt | 9 +- .../mysterium/service/core/NodeRepository.kt | 147 ++++++++ .../core/Openvpn3AndroidTunnelSetupBridge.kt | 254 +++++++------- .../core/WireguardAndroidTunnelSetup.kt | 3 +- .../java/network/mysterium/ui/Countries.kt | 313 +++++++++++++++++ .../EditText.kt} | 29 +- .../network/mysterium/ui/FeedbackFragment.kt | 103 ++++++ .../network/mysterium/ui/FeedbackViewModel.kt | 61 ++++ .../java/network/mysterium/ui/Keyboard.kt | 27 ++ .../network/mysterium/ui/MainVpnFragment.kt | 260 ++++++++++++++ .../MaterialSpinner.kt} | 25 +- .../java/network/mysterium/ui/Navigation.kt | 42 +++ .../network/mysterium/ui/ProposalViewItem.kt | 89 +++++ .../network/mysterium/ui/ProposalsFragment.kt | 251 ++++++++++++++ .../mysterium/ui/ProposalsViewModel.kt | 243 +++++++++++++ .../network/mysterium/ui/SharedViewModel.kt | 247 +++++++++++++ .../network/mysterium/ui/TermsFragment.kt | 64 ++++ .../network/mysterium/ui/TermsViewModel.kt | 182 ++++++++++ .../main/java/network/mysterium/ui/Toast.kt | 26 ++ .../network/mysterium/ui/UnitFormatter.kt | 62 ++++ .../network/mysterium/vpn/MainActivity.kt | 159 --------- .../mysterium/vpn/MainApplication.java | 118 ------- .../vpn/connection/ConnectionChecker.kt | 79 ----- .../src/main/res/drawable/background_logo.png | Bin 0 -> 143919 bytes .../main/res/drawable/background_splash.xml | 12 + .../main/res/drawable/bordered_input_bg.xml | 8 + .../main/res/drawable/ic_arrow_back_24dp.xml | 5 + .../drawable/ic_arrow_downward_black_24dp.xml | 5 + .../res/drawable/ic_help_outline_34dp.xml | 5 + .../drawable/ic_help_outline_black_24dp.xml | 5 + .../res/drawable/ic_public_black_24dp.xml | 5 + .../res/drawable/ic_search_white_34dp.xml | 5 + .../main/res/drawable/ic_star_black_24dp.xml | 5 + .../drawable/ic_star_border_black_24dp.xml | 5 + .../main/res/drawable/notification_icon.png | Bin 0 -> 3209 bytes .../main/res/drawable/proposal_picker_bg.xml | 7 + .../main/res/drawable/proposals_filter_bg.xml | 24 ++ .../src/main/res/drawable/quality_high.png | Bin 0 -> 1636 bytes .../app/src/main/res/drawable/quality_low.png | Bin 0 -> 1654 bytes .../src/main/res/drawable/quality_medium.png | Bin 0 -> 1640 bytes .../src/main/res/drawable/quality_unknown.png | Bin 0 -> 1628 bytes .../main/res/drawable/round_icon_button.xml | 10 + .../app/src/main/res/drawable/selector.xml | 5 + .../src/main/res/drawable/service_openvpn.png | Bin 0 -> 4717 bytes .../main/res/drawable/service_wireguard.png | Bin 0 -> 19549 bytes .../main/res/drawable/spinner_input_arrow.png | Bin 0 -> 1326 bytes .../main/res/drawable/spinner_input_bg.xml | 17 + .../app/src/main/res/drawable/splash_logo.png | Bin 0 -> 22654 bytes .../app/src/main/res/layout/activity_main.xml | 20 ++ .../src/main/res/layout/fragment_feedback.xml | 109 ++++++ .../src/main/res/layout/fragment_main_vpn.xml | 325 ++++++++++++++++++ .../main/res/layout/fragment_proposals.xml | 173 ++++++++++ .../src/main/res/layout/fragment_terms.xml | 46 +++ .../main/res/layout/proposal_list_item.xml | 75 ++++ .../app/src/main/res/navigation/nav_graph.xml | 40 +++ android/app/src/main/res/values/arrays.xml | 13 + android/app/src/main/res/values/strings.xml | 24 ++ android/app/src/main/res/values/styles.xml | 39 ++- .../main/res/xml/network_security_config.xml | 7 - .../res/xml/network_security_config_debug.xml | 4 - .../app/src/test/java/UnitFormatterTest.kt | 40 +++ android/build.gradle | 18 +- android/gradle.properties | 3 +- android/settings.gradle | 5 - fastlane/Fastfile | 26 +- fastlane/README.md | 5 + 81 files changed, 3953 insertions(+), 799 deletions(-) create mode 100644 android/app/schemas/network.mysterium.db.AppDatabase/1.json create mode 100644 android/app/src/main/java/network/mysterium/AppContainer.kt create mode 100644 android/app/src/main/java/network/mysterium/MainActivity.kt create mode 100644 android/app/src/main/java/network/mysterium/MainApplication.kt create mode 100644 android/app/src/main/java/network/mysterium/db/AppDatabase.kt create mode 100644 android/app/src/main/java/network/mysterium/db/FavoriteProposalDao.kt create mode 100644 android/app/src/main/java/network/mysterium/db/TermsDao.kt create mode 100644 android/app/src/main/java/network/mysterium/service/core/DeferredNode.kt create mode 100644 android/app/src/main/java/network/mysterium/service/core/NodeRepository.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/Countries.kt rename android/app/src/main/java/network/mysterium/{vpn/connection/ConnectionCheckerService.kt => ui/EditText.kt} (50%) create mode 100644 android/app/src/main/java/network/mysterium/ui/FeedbackFragment.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/FeedbackViewModel.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/Keyboard.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/MainVpnFragment.kt rename android/app/src/main/java/network/mysterium/{logging/BugReporterPackage.kt => ui/MaterialSpinner.kt} (55%) create mode 100644 android/app/src/main/java/network/mysterium/ui/Navigation.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/ProposalViewItem.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/ProposalsFragment.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/ProposalsViewModel.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/SharedViewModel.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/TermsFragment.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/TermsViewModel.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/Toast.kt create mode 100644 android/app/src/main/java/network/mysterium/ui/UnitFormatter.kt delete mode 100644 android/app/src/main/java/network/mysterium/vpn/MainActivity.kt delete mode 100644 android/app/src/main/java/network/mysterium/vpn/MainApplication.java delete mode 100644 android/app/src/main/java/network/mysterium/vpn/connection/ConnectionChecker.kt create mode 100644 android/app/src/main/res/drawable/background_logo.png create mode 100644 android/app/src/main/res/drawable/background_splash.xml create mode 100644 android/app/src/main/res/drawable/bordered_input_bg.xml create mode 100644 android/app/src/main/res/drawable/ic_arrow_back_24dp.xml create mode 100644 android/app/src/main/res/drawable/ic_arrow_downward_black_24dp.xml create mode 100644 android/app/src/main/res/drawable/ic_help_outline_34dp.xml create mode 100644 android/app/src/main/res/drawable/ic_help_outline_black_24dp.xml create mode 100644 android/app/src/main/res/drawable/ic_public_black_24dp.xml create mode 100644 android/app/src/main/res/drawable/ic_search_white_34dp.xml create mode 100644 android/app/src/main/res/drawable/ic_star_black_24dp.xml create mode 100644 android/app/src/main/res/drawable/ic_star_border_black_24dp.xml create mode 100644 android/app/src/main/res/drawable/notification_icon.png create mode 100644 android/app/src/main/res/drawable/proposal_picker_bg.xml create mode 100644 android/app/src/main/res/drawable/proposals_filter_bg.xml create mode 100644 android/app/src/main/res/drawable/quality_high.png create mode 100644 android/app/src/main/res/drawable/quality_low.png create mode 100644 android/app/src/main/res/drawable/quality_medium.png create mode 100644 android/app/src/main/res/drawable/quality_unknown.png create mode 100644 android/app/src/main/res/drawable/round_icon_button.xml create mode 100644 android/app/src/main/res/drawable/selector.xml create mode 100644 android/app/src/main/res/drawable/service_openvpn.png create mode 100644 android/app/src/main/res/drawable/service_wireguard.png create mode 100644 android/app/src/main/res/drawable/spinner_input_arrow.png create mode 100644 android/app/src/main/res/drawable/spinner_input_bg.xml create mode 100644 android/app/src/main/res/drawable/splash_logo.png create mode 100644 android/app/src/main/res/layout/activity_main.xml create mode 100644 android/app/src/main/res/layout/fragment_feedback.xml create mode 100644 android/app/src/main/res/layout/fragment_main_vpn.xml create mode 100644 android/app/src/main/res/layout/fragment_proposals.xml create mode 100644 android/app/src/main/res/layout/fragment_terms.xml create mode 100644 android/app/src/main/res/layout/proposal_list_item.xml create mode 100644 android/app/src/main/res/navigation/nav_graph.xml create mode 100644 android/app/src/main/res/values/arrays.xml delete mode 100644 android/app/src/main/res/xml/network_security_config.xml delete mode 100644 android/app/src/main/res/xml/network_security_config_debug.xml create mode 100644 android/app/src/test/java/UnitFormatterTest.kt diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index 8f2797f07..7606c0ac6 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -1,33 +1,10 @@ image: mysteriumnetwork/mobile-ci:0.1.0 stages: - - install - - test - deploy -install-packages: - stage: install - script: - - yarn install - cache: - paths: - - node_modules/ - artifacts: - when: on_success - paths: - - node_modules/ - -lint-and-test: - stage: test - dependencies: - - install-packages - script: - - yarn ci - push-beta: stage: deploy - dependencies: - - install-packages when: manual only: - master diff --git a/Gemfile.lock b/Gemfile.lock index d834a7f7a..bc54b7310 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -145,4 +145,4 @@ DEPENDENCIES fastlane BUNDLED WITH - 1.16.5 + 2.0.2 diff --git a/android/app/build.gradle b/android/app/build.gradle index c9627371a..e02560d15 100644 --- a/android/app/build.gradle +++ b/android/app/build.gradle @@ -1,86 +1,11 @@ -apply plugin: "com.android.application" +apply plugin: 'com.android.application' apply plugin: 'kotlin-android' apply plugin: 'io.fabric' +apply plugin: 'kotlin-android-extensions' +apply plugin: 'kotlin-kapt' import com.android.build.OutputFile -/** - * The react.gradle file registers a task for each build variant (e.g. bundleDebugJsAndAssets - * and bundleReleaseJsAndAssets). - * These basically call `react-native bundle` with the correct arguments during the Android build - * cycle. By default, bundleDebugJsAndAssets is skipped, as in debug/dev mode we prefer to load the - * bundle directly from the development server. Below you can see all the possible configurations - * and their defaults. If you decide to add a configuration block, make sure to add it before the - * `apply from: "../../node_modules/react-native/react.gradle"` line. - * - * project.ext.react = [ - * // the name of the generated asset file containing your JS bundle - * bundleAssetName: "index.android.bundle", - * - * // the entry file for bundle generation - * entryFile: "index.android.js", - * - * // whether to bundle JS and assets in debug mode - * bundleInDebug: false, - * - * // whether to bundle JS and assets in release mode - * bundleInRelease: true, - * - * // whether to bundle JS and assets in another build variant (if configured). - * // See http://tools.android.com/tech-docs/new-build-system/user-guide#TOC-Build-Variants - * // The configuration property can be in the following formats - * // 'bundleIn${productFlavor}${buildType}' - * // 'bundleIn${buildType}' - * // bundleInFreeDebug: true, - * // bundleInPaidRelease: true, - * // bundleInBeta: true, - * - * // whether to disable dev mode in custom build variants (by default only disabled in release) - * // for example: to disable dev mode in the staging build type (if configured) - * devDisabledInStaging: true, - * // The configuration property can be in the following formats - * // 'devDisabledIn${productFlavor}${buildType}' - * // 'devDisabledIn${buildType}' - * - * // the root of your project, i.e. where "package.json" lives - * root: "../../", - * - * // where to put the JS bundle asset in debug mode - * jsBundleDirDebug: "$buildDir/intermediates/assets/debug", - * - * // where to put the JS bundle asset in release mode - * jsBundleDirRelease: "$buildDir/intermediates/assets/release", - * - * // where to put drawable resources / React Native assets, e.g. the ones you use via - * // require('./image.png')), in debug mode - * resourcesDirDebug: "$buildDir/intermediates/res/merged/debug", - * - * // where to put drawable resources / React Native assets, e.g. the ones you use via - * // require('./image.png')), in release mode - * resourcesDirRelease: "$buildDir/intermediates/res/merged/release", - * - * // by default the gradle tasks are skipped if none of the JS files or assets change; this means - * // that we don't look at files in android/ or ios/ to determine whether the tasks are up to - * // date; if you have any other folders that you want to ignore for performance reasons (gradle - * // indexes the entire tree), add them here. Alternatively, if you have JS files in android/ - * // for example, you might want to remove it from here. - * inputExcludes: ["android/**", "ios/**"], - * - * // override which node gets called and with what additional arguments - * nodeExecutableAndArgs: ["node"], - * - * // supply additional arguments to the packager - * extraPackagerArgs: [] - * ] - */ - -project.ext.react = [ - entryFile: "index.js", - enableHermes: false, // clean and rebuild if changing -] - -apply from: "../../node_modules/react-native/react.gradle" - /** * Get the version code from command line param * @@ -125,27 +50,6 @@ def enableSeparateBuildPerCPUArchitecture = false */ def enableProguardInReleaseBuilds = false -/** - * The preferred build flavor of JavaScriptCore. - * - * For example, to use the international variant, you can use: - * `def jscFlavor = 'org.webkit:android-jsc-intl:+'` - * - * The international variant includes ICU i18n library and necessary data - * allowing to use e.g. `Date.toLocaleString` and `String.localeCompare` that - * give correct results when using with locales other than en-US. Note that - * this variant is about 6MiB larger per architecture than default. - */ -def jscFlavor = 'org.webkit:android-jsc:+' -/** - * Whether to enable the Hermes VM. - * - * This should be set on project.ext.react and mirrored here. If it is not set - * on project.ext.react, JavaScript will not be compiled to Hermes Bytecode - * and the benefits of using Hermes will therefore be sharply reduced. - */ -def enableHermes = project.ext.react.get("enableHermes", false); - android { compileSdkVersion rootProject.ext.compileSdkVersion @@ -154,12 +58,23 @@ android { targetCompatibility JavaVersion.VERSION_1_8 } + testOptions { + unitTests.includeAndroidResources = true + } + defaultConfig { applicationId "network.mysterium.vpn" minSdkVersion rootProject.ext.minSdkVersion targetSdkVersion rootProject.ext.targetSdkVersion versionCode getVersionCode() versionName getVersionName() + multiDexEnabled true + + javaCompileOptions { + annotationProcessorOptions { + arguments = ["room.schemaLocation": "$projectDir/schemas".toString()] + } + } } splits { abi { @@ -179,7 +94,6 @@ android { } buildTypes { release { - // signingConfig signingConfigs.debug // TODO(am): check if this is needed minifyEnabled enableProguardInReleaseBuilds proguardFiles getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro" } @@ -205,28 +119,41 @@ android { } dependencies { - implementation project(':react-native-push-notification') - implementation project(':react-native-vector-icons') + def nav_version = "2.1.0" + implementation "androidx.navigation:navigation-fragment-ktx:$nav_version" + implementation "androidx.navigation:navigation-ui-ktx:$nav_version" + implementation('com.crashlytics.sdk.android:crashlytics:2.9.6@aar') { - transitive = true - } - if (enableHermes) { - def hermesPath = "../../node_modules/hermes-engine/android/"; - debugImplementation files(hermesPath + "hermes-debug.aar") - releaseImplementation files(hermesPath + "hermes-release.aar") - } else { - implementation jscFlavor + transitive = true } - implementation "androidx.appcompat:appcompat:1.0.0" - implementation 'com.facebook.react:react-native:+' - implementation 'cat.ereza:logcatreporter:1.2.0' - // From node_modules - implementation 'com.google.firebase:firebase-core:16.0.1' + implementation 'androidx.multidex:multidex:2.0.1' + implementation 'com.google.firebase:firebase-core:17.2.1' implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version" - - implementation 'network.mysterium:mobile-node:0.14.1' - // implementation files('libs/Mysterium.aar') + implementation 'androidx.constraintlayout:constraintlayout:1.1.3' + implementation 'androidx.legacy:legacy-support-v4:1.0.0' + implementation 'androidx.appcompat:appcompat:1.1.0' + implementation 'androidx.core:core-ktx:1.1.0' + implementation 'androidx.constraintlayout:constraintlayout:1.1.3' + implementation 'androidx.lifecycle:lifecycle-extensions:2.1.0' + implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.1.0' + implementation 'com.google.android.material:material:1.0.0' + implementation 'androidx.recyclerview:recyclerview:1.1.0' + implementation 'com.makeramen:roundedimageview:2.3.0' + implementation 'com.beust:klaxon:5.0.1' + implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-core:1.3.0' + implementation "org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.0" + implementation "org.jetbrains.kotlin:kotlin-reflect:1.3.50" + implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:2.1.0" + implementation 'androidx.room:room-runtime:2.2.2' + implementation 'androidx.room:room-ktx:2.2.2' + kapt 'androidx.room:room-compiler:2.2.2' + + testImplementation 'junit:junit:4.12' + + implementation 'network.mysterium:mobile-node:0.15.0' + // Comment network.mysterium:mobile-node and replace with your local path to use local node build. + // compile files('/Users/anjmao/go/src/github.com/mysteriumnetwork/node/build/package/Mysterium.aar') } // Run this once to be able to run the application with BUCK @@ -241,12 +168,10 @@ gradle.projectsEvaluated { } task applyGoogleServicesIfNeeded { - if(project.hasProperty('applyGoogleServices')) { + if (project.hasProperty('applyGoogleServices')) { apply plugin: 'com.google.gms.google-services' } } repositories { mavenCentral() } - -apply from: file("../../node_modules/@react-native-community/cli-platform-android/native_modules.gradle"); applyNativeModulesAppBuildGradle(project) \ No newline at end of file diff --git a/android/app/schemas/network.mysterium.db.AppDatabase/1.json b/android/app/schemas/network.mysterium.db.AppDatabase/1.json new file mode 100644 index 000000000..e4e9a108b --- /dev/null +++ b/android/app/schemas/network.mysterium.db.AppDatabase/1.json @@ -0,0 +1,54 @@ +{ + "formatVersion": 1, + "database": { + "version": 1, + "identityHash": "e51f75311548ddc3e5a322d3d572711d", + "entities": [ + { + "tableName": "FavoriteProposal", + "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` TEXT NOT NULL, PRIMARY KEY(`id`))", + "fields": [ + { + "fieldPath": "id", + "columnName": "id", + "affinity": "TEXT", + "notNull": true + } + ], + "primaryKey": { + "columnNames": [ + "id" + ], + "autoGenerate": false + }, + "indices": [], + "foreignKeys": [] + }, + { + "tableName": "Terms", + "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`version` TEXT NOT NULL, PRIMARY KEY(`version`))", + "fields": [ + { + "fieldPath": "version", + "columnName": "version", + "affinity": "TEXT", + "notNull": true + } + ], + "primaryKey": { + "columnNames": [ + "version" + ], + "autoGenerate": false + }, + "indices": [], + "foreignKeys": [] + } + ], + "views": [], + "setupQueries": [ + "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", + "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, 'e51f75311548ddc3e5a322d3d572711d')" + ] + } +} \ No newline at end of file diff --git a/android/app/src/debug/AndroidManifest.xml b/android/app/src/debug/AndroidManifest.xml index b4ee49a4e..46f2eb736 100644 --- a/android/app/src/debug/AndroidManifest.xml +++ b/android/app/src/debug/AndroidManifest.xml @@ -2,5 +2,5 @@ - + diff --git a/android/app/src/main/AndroidManifest.xml b/android/app/src/main/AndroidManifest.xml index 15a1a6e39..17da84094 100644 --- a/android/app/src/main/AndroidManifest.xml +++ b/android/app/src/main/AndroidManifest.xml @@ -1,47 +1,48 @@ + + - - - - - - - - - - - - - - + android:theme="@style/AppTheme" + tools:ignore="GoogleAppIndexingWarning"> - - - + + + + + + + + + + + + + + + - + + \ No newline at end of file diff --git a/android/app/src/main/java/network/mysterium/AppContainer.kt b/android/app/src/main/java/network/mysterium/AppContainer.kt new file mode 100644 index 000000000..d2eb717ab --- /dev/null +++ b/android/app/src/main/java/network/mysterium/AppContainer.kt @@ -0,0 +1,61 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium + +import android.content.Context +import androidx.fragment.app.FragmentActivity +import androidx.room.Room +import kotlinx.coroutines.CompletableDeferred +import network.mysterium.db.AppDatabase +import network.mysterium.logging.BugReporter +import network.mysterium.service.core.DeferredNode +import network.mysterium.service.core.MysteriumCoreService +import network.mysterium.service.core.NodeRepository +import network.mysterium.ui.ProposalsViewModel +import network.mysterium.ui.SharedViewModel +import network.mysterium.ui.TermsViewModel + +class AppContainer { + lateinit var appDatabase: AppDatabase + lateinit var nodeRepository: NodeRepository + lateinit var sharedViewModel: SharedViewModel + lateinit var proposalsViewModel: ProposalsViewModel + lateinit var termsViewModel: TermsViewModel + lateinit var bugReporter: BugReporter + lateinit var deferredMysteriumCoreService: CompletableDeferred + + fun init(ctx: Context, deferredNode: DeferredNode, mysteriumCoreService: CompletableDeferred) { + appDatabase = Room.databaseBuilder( + ctx, + AppDatabase::class.java, "mysteriumvpn" + ).build() + + deferredMysteriumCoreService = mysteriumCoreService + bugReporter = BugReporter() + nodeRepository = NodeRepository(deferredNode) + sharedViewModel = SharedViewModel(nodeRepository, bugReporter, deferredMysteriumCoreService) + proposalsViewModel = ProposalsViewModel(sharedViewModel, nodeRepository, appDatabase) + termsViewModel = TermsViewModel(appDatabase) + } + + companion object { + fun from(activity: FragmentActivity?): AppContainer { + return (activity!!.application as MainApplication).appContainer + } + } +} diff --git a/android/app/src/main/java/network/mysterium/MainActivity.kt b/android/app/src/main/java/network/mysterium/MainActivity.kt new file mode 100644 index 000000000..62c6c7ecb --- /dev/null +++ b/android/app/src/main/java/network/mysterium/MainActivity.kt @@ -0,0 +1,151 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium + +import android.app.Activity +import android.content.ComponentName +import android.content.Context +import android.content.Intent +import android.content.ServiceConnection +import android.net.VpnService +import android.os.Bundle +import android.os.IBinder +import android.util.Log +import android.widget.Toast +import androidx.appcompat.app.AppCompatActivity +import androidx.navigation.Navigation +import kotlinx.coroutines.CompletableDeferred +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.launch +import network.mysterium.service.core.DeferredNode +import network.mysterium.service.core.MysteriumAndroidCoreService +import network.mysterium.service.core.MysteriumCoreService +import network.mysterium.vpn.R + +class MainActivity : AppCompatActivity() { + private lateinit var appContainer: AppContainer + private var deferredNode = DeferredNode() + private var deferredMysteriumCoreService = CompletableDeferred() + + private val serviceConnection = object : ServiceConnection { + override fun onServiceDisconnected(name: ComponentName?) { + Log.i(TAG, "Service disconnected") + } + + override fun onServiceConnected(name: ComponentName?, service: IBinder?) { + Log.i(TAG, "Service connected") + deferredMysteriumCoreService.complete(service as MysteriumCoreService) + deferredNode.start(service) {err -> + if (err != null) { + showNodeStarError() + } + } + } + } + + override fun onCreate(savedInstanceState: Bundle?) { + setTheme(R.style.AppTheme) + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_main) + + // Initialize app DI container. + appContainer = (application as MainApplication).appContainer + appContainer.init(applicationContext, deferredNode, deferredMysteriumCoreService) + + // Bind VPN service. + ensureVpnServicePermission() + bindMysteriumService() + + // Load initial state without blocking main UI thread. + CoroutineScope(Dispatchers.Main).launch { + // Load favorite proposals from local database. + val favoriteProposals = appContainer.proposalsViewModel.loadFavoriteProposals() + + // Load initial data like current location, statistics, active proposal (if any). + appContainer.sharedViewModel.load(favoriteProposals) + + // Load initial proposals. + appContainer.proposalsViewModel.load() + } + + // Navigate to main vpn screen and check if terms are accepted in separate coroutine + // so it does not block main thread. + navigate(R.id.main_vpn_fragment) + CoroutineScope(Dispatchers.Main).launch { + val termsAccepted = appContainer.termsViewModel.checkTermsAccepted() + if (!termsAccepted) { + navigate(R.id.terms_fragment) + } + } + } + + override fun onDestroy() { + unbindMysteriumService() + super.onDestroy() + } + + override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) { + super.onActivityResult(requestCode, resultCode, data) + + when (requestCode) { + VPN_SERVICE_REQUEST -> { + if (resultCode != Activity.RESULT_OK) { + Log.w(TAG, "User forbidden VPN service") + Toast.makeText(this, "VPN connection has to be granted for MysteriumVPN to work.", Toast.LENGTH_LONG).show() + finish() + return + } + Log.i(TAG, "User allowed VPN service") + } + } + } + + private fun showNodeStarError() { + Toast.makeText(this, "Failed to initialize. Please relaunch app.", Toast.LENGTH_LONG).show() + } + + private fun bindMysteriumService() { + Log.i(TAG, "Binding service") + Intent(this, MysteriumAndroidCoreService::class.java).also { intent -> + bindService(intent, serviceConnection, Context.BIND_AUTO_CREATE) + } + } + + private fun unbindMysteriumService() { + Log.i(TAG, "Unbinding service") + unbindService(serviceConnection) + } + + private fun ensureVpnServicePermission() { + val intent: Intent = VpnService.prepare(this) ?: return + startActivityForResult(intent, VPN_SERVICE_REQUEST) + } + + private fun navigate(destination: Int) { + val navController = Navigation.findNavController(this, R.id.nav_host_fragment) + val navGraph = navController.navInflater.inflate(R.navigation.nav_graph) + navGraph.startDestination = destination + navController.graph = navGraph + } + + companion object { + private const val VPN_SERVICE_REQUEST = 1 + private const val TAG = "MainActivity" + } +} diff --git a/android/app/src/main/java/network/mysterium/MainApplication.kt b/android/app/src/main/java/network/mysterium/MainApplication.kt new file mode 100644 index 000000000..52f2bb375 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/MainApplication.kt @@ -0,0 +1,54 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium + +import android.util.Log +import androidx.multidex.MultiDexApplication +import com.crashlytics.android.Crashlytics +import com.crashlytics.android.core.CrashlyticsCore +import io.fabric.sdk.android.Fabric +import network.mysterium.ui.Countries +import network.mysterium.vpn.BuildConfig + +class MainApplication : MultiDexApplication() { + val appContainer = AppContainer() + + override fun onCreate() { + setupLogging() + super.onCreate() + Countries.loadBitmaps() + Log.i(TAG, "Application started") + } + + private fun setupLogging() { + // https://docs.fabric.io/android/crashlytics/build-tools.html?highlight=crashlyticscore + // Set up Crashlytics, disabled for debug builds + val crashlyticsKit = Crashlytics.Builder() + .core(CrashlyticsCore.Builder().disabled(BuildConfig.DEBUG).build()) + .build() + + // Initialize Fabric with the debug-disabled crashlytics. + Fabric.with(this, crashlyticsKit) + + Crashlytics.setInt("android_sdk_int", android.os.Build.VERSION.SDK_INT) + } + + companion object { + private const val TAG = "MainApplication" + } +} diff --git a/android/app/src/main/java/network/mysterium/db/AppDatabase.kt b/android/app/src/main/java/network/mysterium/db/AppDatabase.kt new file mode 100644 index 000000000..ed7362f80 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/db/AppDatabase.kt @@ -0,0 +1,10 @@ +package network.mysterium.db + +import androidx.room.Database +import androidx.room.RoomDatabase + +@Database(entities = [FavoriteProposal::class, Terms::class], version = 1) +abstract class AppDatabase : RoomDatabase() { + abstract fun favoriteProposalDao(): FavoriteProposalDao + abstract fun termsDao(): TermsDao +} diff --git a/android/app/src/main/java/network/mysterium/db/FavoriteProposalDao.kt b/android/app/src/main/java/network/mysterium/db/FavoriteProposalDao.kt new file mode 100644 index 000000000..e1a4742da --- /dev/null +++ b/android/app/src/main/java/network/mysterium/db/FavoriteProposalDao.kt @@ -0,0 +1,20 @@ +package network.mysterium.db + +import androidx.room.* + +@Entity +data class FavoriteProposal( + @PrimaryKey val id: String +) + +@Dao +interface FavoriteProposalDao { + @Query("SELECT * FROM favoriteproposal") + suspend fun getAll(): List + + @Insert + suspend fun insert(favoriteProposal: FavoriteProposal) + + @Delete + suspend fun delete(favoriteProposal: FavoriteProposal) +} diff --git a/android/app/src/main/java/network/mysterium/db/TermsDao.kt b/android/app/src/main/java/network/mysterium/db/TermsDao.kt new file mode 100644 index 000000000..f79aac705 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/db/TermsDao.kt @@ -0,0 +1,20 @@ +package network.mysterium.db + +import androidx.room.* + +@Entity +data class Terms( + @PrimaryKey val version: String +) + +@Dao +interface TermsDao { + @Query("SELECT * FROM terms LIMIT 1") + suspend fun get(): Terms? + + @Insert + suspend fun insert(terms: Terms) + + @Query("DELETE FROM terms") + suspend fun delete() +} diff --git a/android/app/src/main/java/network/mysterium/logging/BugReporter.kt b/android/app/src/main/java/network/mysterium/logging/BugReporter.kt index 62b94d88b..2cf2e5bf0 100644 --- a/android/app/src/main/java/network/mysterium/logging/BugReporter.kt +++ b/android/app/src/main/java/network/mysterium/logging/BugReporter.kt @@ -17,31 +17,10 @@ package network.mysterium.logging -import cat.ereza.logcatreporter.LogcatReporter import com.crashlytics.android.Crashlytics -import com.facebook.react.bridge.ReactApplicationContext -import com.facebook.react.bridge.ReactContextBaseJavaModule -import com.facebook.react.bridge.ReactMethod -class FeedbackException(message: String) : Exception(message) - -class BugReporter(reactContext: ReactApplicationContext) : ReactContextBaseJavaModule(reactContext) { - override fun getName(): String { - return "BugReporter" - } - - @ReactMethod - fun logException(value: String) { - LogcatReporter.reportExceptionWithLogcat(RuntimeException(value)) - } - - @ReactMethod +class BugReporter { fun setUserIdentifier(userIdentifier: String) { Crashlytics.setUserIdentifier(userIdentifier) } - - @ReactMethod - fun sendFeedback(type: String, message: String) { - LogcatReporter.reportExceptionWithLogcat(FeedbackException(type + ":" + message)) - } } diff --git a/android/app/src/main/java/network/mysterium/service/core/DeferredNode.kt b/android/app/src/main/java/network/mysterium/service/core/DeferredNode.kt new file mode 100644 index 000000000..8564669e4 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/service/core/DeferredNode.kt @@ -0,0 +1,35 @@ +package network.mysterium.service.core + +import android.util.Log +import kotlinx.coroutines.CompletableDeferred +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.launch +import mysterium.MobileNode + +// DeferredNode is a wrapper class which holds MobileNode instance promise. +// This allows to load UI without waiting for node to start. +class DeferredNode { + private var deferredNode = CompletableDeferred() + + suspend fun await(): MobileNode { + return deferredNode.await() + } + + fun start(service: MysteriumCoreService, done: (err: Exception?) -> Unit) { + CoroutineScope(Dispatchers.Main).launch { + try { + val node = service.startNode() + deferredNode.complete(node) + done(null) + } catch (err: Exception) { + Log.e(TAG, "Failed to start node", err) + done(err) + } + } + } + + companion object { + const val TAG = "DeferredNode" + } +} diff --git a/android/app/src/main/java/network/mysterium/service/core/MysteriumAndroidCoreService.kt b/android/app/src/main/java/network/mysterium/service/core/MysteriumAndroidCoreService.kt index 5c0024459..29c0fd113 100644 --- a/android/app/src/main/java/network/mysterium/service/core/MysteriumAndroidCoreService.kt +++ b/android/app/src/main/java/network/mysterium/service/core/MysteriumAndroidCoreService.kt @@ -17,64 +17,147 @@ package network.mysterium.service.core +import android.app.NotificationChannel +import android.app.NotificationManager +import android.app.PendingIntent +import android.content.Context import android.content.Intent import android.net.VpnService import android.os.Binder +import android.os.Build import android.os.IBinder import android.util.Log +import androidx.core.app.NotificationCompat import mysterium.MobileNode import mysterium.Mysterium +import network.mysterium.MainActivity +import network.mysterium.vpn.BuildConfig +import network.mysterium.vpn.R class MysteriumAndroidCoreService : VpnService() { - private var mobileNode: MobileNode? = null - - fun startMobileNode(filesPath: String) { - val openvpnBridge = Openvpn3AndroidTunnelSetupBridge(this) - val wireguardBridge = WireguardAndroidTunnelSetup(this) - - val logOptions = Mysterium.defaultLogOptions() - val options = Mysterium.defaultNetworkOptions() - - mobileNode = Mysterium.newNode(filesPath, logOptions, options) - mobileNode?.overrideOpenvpnConnection(openvpnBridge) - mobileNode?.overrideWireguardConnection(wireguardBridge) - Log.i(TAG, "started") - } - - fun stopMobileNode() { - val node = mobileNode - if (node == null) { - Log.w(TAG, "Trying to stop node when instance is not set") - return + private var mobileNode: MobileNode? = null + private val notificationsChannelId = BuildConfig.APPLICATION_ID + + // pendingAppIntent is used to navigate back to MainActivity + // when user taps on notification. + private lateinit var pendingAppIntent: PendingIntent + + fun startMobileNode(filesPath: String): MobileNode { + if (mobileNode != null) { + return mobileNode!! + } + + val openvpnBridge = Openvpn3AndroidTunnelSetupBridge(this) + val wireguardBridge = WireguardAndroidTunnelSetup(this) + + val logOptions = Mysterium.defaultLogOptions() + logOptions.filepath = filesPath + logOptions.logHTTP = false + val options = Mysterium.defaultNetworkOptions() + + mobileNode = Mysterium.newNode(filesPath, logOptions, options) + mobileNode?.overrideOpenvpnConnection(openvpnBridge) + mobileNode?.overrideWireguardConnection(wireguardBridge) + + Log.i(TAG, "started") + return mobileNode!! } - node.shutdown() - try { - node.waitUntilDies() - } catch (e: Exception) { - Log.i(TAG, "Got exception, safe to ignore: " + e.message) + fun stopMobileNode() { + val node = mobileNode + if (node == null) { + Log.w(TAG, "Trying to stop node when instance is not set") + return + } + + node.shutdown() + try { + node.waitUntilDies() + } catch (e: Exception) { + Log.i(TAG, "Got exception, safe to ignore: " + e.message) + } finally { + stopForeground(true) + } } - } - override fun onRevoke() { - Log.w(TAG, "VPN service revoked!") - } + override fun onCreate() { + super.onCreate() - inner class MysteriumCoreServiceBridge : Binder(), MysteriumCoreService { - override fun StartTequila() { - startMobileNode(filesDir.canonicalPath) + val intent = Intent(this, MainActivity::class.java).apply { + flags = Intent.FLAG_ACTIVITY_SINGLE_TOP + } + pendingAppIntent = PendingIntent.getActivity(this, 0, intent, 0) + + createNotificationChannel() + } + + override fun onDestroy() { + super.onDestroy() + stopMobileNode() + // TODO: Check if node is destroyed correctly. } - override fun StopTequila() { - stopMobileNode() + private fun createNotificationChannel() { + if (Build.VERSION.SDK_INT < 26) { + return + } + val notificationManager = getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager + val channel = NotificationChannel(notificationsChannelId, notificationsChannelId, NotificationManager.IMPORTANCE_DEFAULT) + channel.enableVibration(false) + notificationManager.createNotificationChannel(channel) } - } - override fun onBind(intent: Intent?): IBinder? { - return MysteriumCoreServiceBridge() - } + // startForeground starts service with given notifications in foreground. + fun startForeground(title: String, content: String = "") { + if (Build.VERSION.SDK_INT < 26) { + return + } + val notification = NotificationCompat.Builder(this, notificationsChannelId) + .setSmallIcon(R.drawable.notification_icon) + .setContentTitle(title) + .setPriority(NotificationCompat.PRIORITY_DEFAULT) + .setVibrate(LongArray(0)) + .setContentIntent(pendingAppIntent) + .setOnlyAlertOnce(true) + + if (content != "") { + notification.setContentText(content) + } + + startForeground(1, notification.build()) + } - companion object { - private const val TAG = "Mysterium vpn service" - } + fun stopForeground() { + stopForeground(true) + } + + override fun onRevoke() { + Log.w(TAG, "VPN service revoked!") + } + + inner class MysteriumCoreServiceBridge : Binder(), MysteriumCoreService { + override fun startNode(): MobileNode { + return startMobileNode(filesDir.canonicalPath) + } + + override fun stopNode() { + stopMobileNode() + } + + override fun showNotification(title: String, content: String) { + startForeground(title, content) + } + + override fun hideNotifications() { + stopForeground() + } + } + + override fun onBind(intent: Intent?): IBinder? { + return MysteriumCoreServiceBridge() + } + + companion object { + private const val TAG = "MysteriumVPNService" + } } diff --git a/android/app/src/main/java/network/mysterium/service/core/MysteriumCoreService.kt b/android/app/src/main/java/network/mysterium/service/core/MysteriumCoreService.kt index ccd6ed0db..3580b2b11 100644 --- a/android/app/src/main/java/network/mysterium/service/core/MysteriumCoreService.kt +++ b/android/app/src/main/java/network/mysterium/service/core/MysteriumCoreService.kt @@ -18,9 +18,14 @@ package network.mysterium.service.core import android.os.IBinder +import mysterium.MobileNode interface MysteriumCoreService : IBinder { - fun StartTequila() + fun startNode(): MobileNode - fun StopTequila() + fun stopNode() + + fun showNotification(title: String, content: String = "") + + fun hideNotifications() } diff --git a/android/app/src/main/java/network/mysterium/service/core/NodeRepository.kt b/android/app/src/main/java/network/mysterium/service/core/NodeRepository.kt new file mode 100644 index 000000000..3fc61ba19 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/service/core/NodeRepository.kt @@ -0,0 +1,147 @@ +package network.mysterium.service.core + +import android.util.Log +import com.beust.klaxon.Json +import com.beust.klaxon.Klaxon +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.withContext +import mysterium.ConnectRequest +import mysterium.GetProposalRequest +import mysterium.GetProposalsRequest +import mysterium.SendFeedbackRequest + +class ProposalItem( + @Json(name = "providerId") + val providerID: String, + + @Json(name = "serviceType") + val serviceType: String, + + @Json(name = "countryCode") + val countryCode: String, + + @Json(name = "qualityLevel") + val qualityLevel: Int +) + +class ProposalsResponse( + @Json(name = "proposals") + val proposals: List? +) + +class ProposalResponse( + @Json(name = "proposal") + val proposal: ProposalItem? +) + +class Statistics( + val duration: Long, + val bytesReceived: Long, + val bytesSent: Long +) + +class Location( + val ip: String, + val countryCode: String +) + +class Status( + val state: String, + val providerID: String, + val serviceType: String +) + +class NodeRepository(private val deferredNode: DeferredNode) { + + suspend fun getProposals(refresh: Boolean): List { + val req = GetProposalsRequest() + req.showOpenvpnProposals = true + req.showWireguardProposals = true + req.refresh = refresh + + val bytes = getProposals(req) + val proposalsResponse = parseProposals(bytes) + if (proposalsResponse?.proposals == null) { + return listOf() + } + + return proposalsResponse.proposals + } + + suspend fun getProposal(providerID: String, serviceType: String): ProposalItem? { + val req = GetProposalRequest() + req.providerID = providerID + req.serviceType = serviceType + + val bytes = getProposal(req) + val proposalsResponse = parseProposal(bytes) + return proposalsResponse?.proposal + } + + suspend fun registerConnectionStatusChangeCallback(cb: (status: String) -> Unit) { + deferredNode.await().registerConnectionStatusChangeCallback { status -> cb(status) } + } + + suspend fun registerStatisticsChangeCallback(cb: (stats: Statistics) -> Unit) { + deferredNode.await().registerStatisticsChangeCallback { duration, bytesReceived, bytesSent -> + cb(Statistics(duration, bytesReceived, bytesSent)) + } + } + + suspend fun connect(req: ConnectRequest) = withContext(Dispatchers.IO) { + deferredNode.await().connect(req) + } + + suspend fun disconnect() = withContext(Dispatchers.IO) { + deferredNode.await().disconnect() + } + + suspend fun unlockIdentity(): String = withContext(Dispatchers.IO) { + deferredNode.await().unlockIdentity() + } + + suspend fun getLocation(): Location { + val location = getLocationAsync() + return Location( + ip = location.ip, + countryCode = location.country + ) + } + + suspend fun getStatus(): Status { + val status = getStatusAsync() + return Status( + state = status.state, + providerID = status.providerID, + serviceType = status.serviceType + ) + } + + suspend fun sendFeedback(req: SendFeedbackRequest) = withContext(Dispatchers.IO) { + deferredNode.await().sendFeedback(req) + } + + private suspend fun getProposals(req: GetProposalsRequest) = withContext(Dispatchers.IO) { + deferredNode.await().getProposals(req) + } + + private suspend fun getProposal(req: GetProposalRequest) = withContext(Dispatchers.IO) { + deferredNode.await().getProposal(req) + } + + private suspend fun parseProposals(bytes: ByteArray) = withContext(Dispatchers.Default) { + Klaxon().parse(bytes.inputStream()) + } + + private suspend fun parseProposal(bytes: ByteArray) = withContext(Dispatchers.Default) { + Klaxon().parse(bytes.inputStream()) + } + + private suspend fun getLocationAsync() = withContext(Dispatchers.IO) { + deferredNode.await().location + } + + private suspend fun getStatusAsync() = withContext(Dispatchers.IO) { + deferredNode.await().status + } +} diff --git a/android/app/src/main/java/network/mysterium/service/core/Openvpn3AndroidTunnelSetupBridge.kt b/android/app/src/main/java/network/mysterium/service/core/Openvpn3AndroidTunnelSetupBridge.kt index 08f9035cf..bd468bd08 100644 --- a/android/app/src/main/java/network/mysterium/service/core/Openvpn3AndroidTunnelSetupBridge.kt +++ b/android/app/src/main/java/network/mysterium/service/core/Openvpn3AndroidTunnelSetupBridge.kt @@ -23,131 +23,131 @@ import android.util.Log import mysterium.Openvpn3TunnelSetup class Openvpn3AndroidTunnelSetupBridge(private val vpnService: VpnService) : Openvpn3TunnelSetup { - override fun socketProtect(socket: Long): Boolean { - val succeeded = vpnService.protect(socket.toInt()) - Log.i(TAG, "Protecting socket: ${socket.toInt()} res: $succeeded") - return succeeded - } - - private var builder: VpnService.Builder? = null - - private var tunnelFd: Int? = null - - override fun addAddress( - address: String, - prefixLength: Long, - gateway: String, - ipv6: Boolean, - net30: Boolean - ): Boolean { - - builder?.addAddress(address, prefixLength.toInt()) - return builder != null - } - - override fun addDnsServer(address: String, ipv6: Boolean): Boolean { - builder?.addDnsServer(address) - return builder != null - } - - override fun addProxyBypass(bypassHost: String): Boolean { - return false - } - - override fun addRoute(address: String, prefixLength: Long, metric: Long, ipv6: Boolean): Boolean { - builder?.addRoute(address, prefixLength.toInt()) - return builder != null - } - - override fun addSearchDomain(domain: String): Boolean { - builder?.addSearchDomain(domain) - return builder != null - } - - override fun addWinsServer(address: String): Boolean { - return false - } - - @Throws(Exception::class) - override fun establish(): Long { - tunnelFd = builder?.establish()?.detachFd() - return tunnelFd?.toLong() ?: -1 - } - - override fun establishLite() { - //whatever that means - } - - override fun excludeRoute(address: String, prefixLength: Long, metric: Long, ipv6: Boolean): Boolean { - return false - } - - override fun newBuilder(): Boolean { - builder = vpnService.Builder() - return true - } - - override fun persist(): Boolean { - return false - } - - override fun rerouteGw(ipv4: Boolean, ipv6: Boolean, flags: Long): Boolean { - Log.i(TAG, "Flags for gw reroute: " + flags.toString(16)) - builder?.addRoute("0.0.0.0", 1) - builder?.addRoute("128.0.0.0", 1) - return builder != null - } - - override fun setAdapterDomainSuffix(name: String): Boolean { - return false - } - - override fun setBlockIpv6(ipv6Block: Boolean): Boolean { - return false - } - - override fun setLayer(layer: Long): Boolean { - return layer == 3L - } - - override fun setMtu(mtu: Long): Boolean { - builder?.setMtu(mtu.toInt()) - return builder != null - } - - override fun setProxyAutoConfigUrl(url: String): Boolean { - return false - } - - override fun setProxyHttp(host: String, port: Long): Boolean { - return false - } - - override fun setProxyHttps(host: String, port: Long): Boolean { - return false - } - - override fun setRemoteAddress(ipAddress: String, ipv6: Boolean): Boolean { - //look into internet - return true - } - - override fun setRouteMetricDefault(metric: Long): Boolean { - return false - } - - override fun setSessionName(name: String): Boolean { - builder?.setSession(name) - return builder != null - } - - override fun teardown(disconnect: Boolean) { - tunnelFd?.let { - ParcelFileDescriptor.adoptFd(it).close() - } - } - - companion object { - private const val TAG = "Openvpn3 setup bridge" - } + override fun socketProtect(socket: Long): Boolean { + val succeeded = vpnService.protect(socket.toInt()) + Log.i(TAG, "Protecting socket: ${socket.toInt()} res: $succeeded") + return succeeded + } + + private var builder: VpnService.Builder? = null + + private var tunnelFd: Int? = null + + override fun addAddress( + address: String, + prefixLength: Long, + gateway: String, + ipv6: Boolean, + net30: Boolean + ): Boolean { + + builder?.addAddress(address, prefixLength.toInt()) + return builder != null + } + + override fun addDnsServer(address: String, ipv6: Boolean): Boolean { + builder?.addDnsServer(address) + return builder != null + } + + override fun addProxyBypass(bypassHost: String): Boolean { + return false + } + + override fun addRoute(address: String, prefixLength: Long, metric: Long, ipv6: Boolean): Boolean { + builder?.addRoute(address, prefixLength.toInt()) + return builder != null + } + + override fun addSearchDomain(domain: String): Boolean { + builder?.addSearchDomain(domain) + return builder != null + } + + override fun addWinsServer(address: String): Boolean { + return false + } + + @Throws(Exception::class) + override fun establish(): Long { + tunnelFd = builder?.establish()?.detachFd() + return tunnelFd?.toLong() ?: -1 + } + + override fun establishLite() { + //whatever that means + } + + override fun excludeRoute(address: String, prefixLength: Long, metric: Long, ipv6: Boolean): Boolean { + return false + } + + override fun newBuilder(): Boolean { + builder = vpnService.Builder() + return true + } + + override fun persist(): Boolean { + return false + } + + override fun rerouteGw(ipv4: Boolean, ipv6: Boolean, flags: Long): Boolean { + Log.i(TAG, "Flags for gw reroute: " + flags.toString(16)) + builder?.addRoute("0.0.0.0", 1) + builder?.addRoute("128.0.0.0", 1) + return builder != null + } + + override fun setAdapterDomainSuffix(name: String): Boolean { + return false + } + + override fun setBlockIpv6(ipv6Block: Boolean): Boolean { + return false + } + + override fun setLayer(layer: Long): Boolean { + return layer == 3L + } + + override fun setMtu(mtu: Long): Boolean { + builder?.setMtu(mtu.toInt()) + return builder != null + } + + override fun setProxyAutoConfigUrl(url: String): Boolean { + return false + } + + override fun setProxyHttp(host: String, port: Long): Boolean { + return false + } + + override fun setProxyHttps(host: String, port: Long): Boolean { + return false + } + + override fun setRemoteAddress(ipAddress: String, ipv6: Boolean): Boolean { + //look into internet + return true + } + + override fun setRouteMetricDefault(metric: Long): Boolean { + return false + } + + override fun setSessionName(name: String): Boolean { + builder?.setSession(name) + return builder != null + } + + override fun teardown(disconnect: Boolean) { + tunnelFd?.let { + ParcelFileDescriptor.adoptFd(it).close() + } + } + + companion object { + private const val TAG = "Openvpn3 setup bridge" + } } diff --git a/android/app/src/main/java/network/mysterium/service/core/WireguardAndroidTunnelSetup.kt b/android/app/src/main/java/network/mysterium/service/core/WireguardAndroidTunnelSetup.kt index 9129c798a..62c13ea30 100644 --- a/android/app/src/main/java/network/mysterium/service/core/WireguardAndroidTunnelSetup.kt +++ b/android/app/src/main/java/network/mysterium/service/core/WireguardAndroidTunnelSetup.kt @@ -22,7 +22,6 @@ import android.os.Build import android.util.Log import mysterium.WireguardTunnelSetup - class WireguardAndroidTunnelSetup(val vpnService: VpnService) : WireguardTunnelSetup { var tunBuilder: VpnService.Builder? = null @@ -42,7 +41,7 @@ class WireguardAndroidTunnelSetup(val vpnService: VpnService) : WireguardTunnelS } override fun addTunnelAddress(ip: String, prefixLength: Long) { - tunBuilder?.addAddress(ip , prefixLength.toInt()) + tunBuilder?.addAddress(ip, prefixLength.toInt()) } override fun protect(socket: Long) { diff --git a/android/app/src/main/java/network/mysterium/ui/Countries.kt b/android/app/src/main/java/network/mysterium/ui/Countries.kt new file mode 100644 index 000000000..9c6928d06 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/Countries.kt @@ -0,0 +1,313 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import android.util.Base64 +import android.graphics.* +import android.graphics.Bitmap + +class CountryFlag constructor(val name: String, val image: String) {} + +class Countries { + companion object { + fun loadBitmaps() { + for (v in values) { + val base64Image = v.value.image.split(",")[1] + val decodedString = Base64.decode(base64Image, Base64.DEFAULT) + val bitmap = BitmapFactory.decodeByteArray(decodedString, 0, decodedString.size) + + // Issue https://github.com/vinc3m1/RoundedImageView/issues/222 + val width = bitmap.width + val height = bitmap.height + var dstBmp: Bitmap + if (bitmap.width >= bitmap.height) { + dstBmp = Bitmap.createBitmap( + bitmap, + width / 2 - height / 2, + 0, + height, + height + ) + } else { + dstBmp = Bitmap.createBitmap( + bitmap, + 0, + height / 2 - width / 2, + width, + width + ) + } + + bitmaps[v.key] = dstBmp + } + } + + var bitmaps: MutableMap = mutableMapOf() + + val values: Map = mapOf( + "af" to CountryFlag("Afghanistan", ""), + "al" to CountryFlag("Albania", ""), + "dz" to CountryFlag("Algeria", ""), + "as" to CountryFlag("American Samoa", ""), + "ad" to CountryFlag("Andorra", ""), + "ao" to CountryFlag("Angola", ""), + "ai" to CountryFlag("Anguilla", ""), + "aq" to CountryFlag("Antarctica", ""), + "ag" to CountryFlag("Antigua and Barbuda", ""), + "ar" to CountryFlag("Argentina", ""), + "am" to CountryFlag("Armenia", ""), + "aw" to CountryFlag("Aruba", ""), + "au" to CountryFlag("Australia", ""), + "at" to CountryFlag("Austria", ""), + "az" to CountryFlag("Azerbaijan", ""), + "bs" to CountryFlag("Bahamas", ""), + "bh" to CountryFlag("Bahrain", ""), + "bd" to CountryFlag("Bangladesh", ""), + "bb" to CountryFlag("Barbados", ""), + "by" to CountryFlag("Belarus", ""), + "be" to CountryFlag("Belgium", ""), + "bz" to CountryFlag("Belize", ""), + "bj" to CountryFlag("Benin", ""), + "bm" to CountryFlag("Bermuda", ""), + "bt" to CountryFlag("Bhutan", ""), + "bo" to CountryFlag("Bolivia", ""), + "ba" to CountryFlag("Bosnia and Herzegovina", ""), + "bw" to CountryFlag("Botswana", ""), + "bv" to CountryFlag("Bouvet Island", ""), + "br" to CountryFlag("Brazil", ""), + "io" to CountryFlag("British Indian Ocean Territory", ""), + "vg" to CountryFlag("British Virgin Islands", ""), + "bn" to CountryFlag("Brunei", ""), + "bg" to CountryFlag("Bulgaria", ""), + "bf" to CountryFlag("Burkina Faso", ""), + "bi" to CountryFlag("Burundi", ""), + "kh" to CountryFlag("Cambodia", ""), + "cm" to CountryFlag("Cameroon", ""), + "ca" to CountryFlag("Canada", ""), + "cv" to CountryFlag("Cape Verde", ""), + "ky" to CountryFlag("Cayman Islands", ""), + "cf" to CountryFlag("Central African Republic", ""), + "td" to CountryFlag("Chad", ""), + "cl" to CountryFlag("Chile", ""), + "cn" to CountryFlag("China", ""), + "cx" to CountryFlag("Christmas Island", ""), + "cc" to CountryFlag("Cocos (Keeling) Islands", ""), + "co" to CountryFlag("Colombia", ""), + "km" to CountryFlag("Comoros", ""), + "ck" to CountryFlag("Cook Islands", ""), + "cr" to CountryFlag("Costa Rica", ""), + "hr" to CountryFlag("Croatia", ""), + "cu" to CountryFlag("Cuba", ""), + "cw" to CountryFlag("Curaçao", ""), + "cy" to CountryFlag("Cyprus", ""), + "cz" to CountryFlag("Czech Republic", ""), + "cd" to CountryFlag("DR Congo", ""), + "dk" to CountryFlag("Denmark", ""), + "dj" to CountryFlag("Djibouti", ""), + "dm" to CountryFlag("Dominica", ""), + "do" to CountryFlag("Dominican Republic", ""), + "ec" to CountryFlag("Ecuador", ""), + "eg" to CountryFlag("Egypt", ""), + "sv" to CountryFlag("El Salvador", ""), + "gq" to CountryFlag("Equatorial Guinea", ""), + "er" to CountryFlag("Eritrea", ""), + "ee" to CountryFlag("Estonia", ""), + "et" to CountryFlag("Ethiopia", ""), + "fk" to CountryFlag("Falkland Islands", ""), + "fo" to CountryFlag("Faroe Islands", ""), + "fj" to CountryFlag("Fiji", ""), + "fi" to CountryFlag("Finland", ""), + "fr" to CountryFlag("France", ""), + "gf" to CountryFlag("French Guiana", ""), + "pf" to CountryFlag("French Polynesia", ""), + "tf" to CountryFlag("French Southern and Antarctic Lands", ""), + "ga" to CountryFlag("Gabon", ""), + "gm" to CountryFlag("Gambia", ""), + "ge" to CountryFlag("Georgia", ""), + "de" to CountryFlag("Germany", ""), + "gh" to CountryFlag("Ghana", ""), + "gi" to CountryFlag("Gibraltar", ""), + "gr" to CountryFlag("Greece", ""), + "gl" to CountryFlag("Greenland", ""), + "gd" to CountryFlag("Grenada", ""), + "gp" to CountryFlag("Guadeloupe", ""), + "gu" to CountryFlag("Guam", ""), + "gt" to CountryFlag("Guatemala", ""), + "gg" to CountryFlag("Guernsey", ""), + "gn" to CountryFlag("Guinea", ""), + "gw" to CountryFlag("Guinea-Bissau", ""), + "gy" to CountryFlag("Guyana", ""), + "ht" to CountryFlag("Haiti", ""), + "hm" to CountryFlag("Heard Island and McDonald Islands", ""), + "hn" to CountryFlag("Honduras", ""), + "hk" to CountryFlag("Hong Kong", ""), + "hu" to CountryFlag("Hungary", ""), + "is" to CountryFlag("Iceland", ""), + "in" to CountryFlag("India", ""), + "id" to CountryFlag("Indonesia", ""), + "ir" to CountryFlag("Iran", ""), + "iq" to CountryFlag("Iraq", ""), + "ie" to CountryFlag("Ireland", ""), + "im" to CountryFlag("Isle of Man", ""), + "il" to CountryFlag("Israel", ""), + "it" to CountryFlag("Italy", ""), + "ci" to CountryFlag("Ivory Coast", ""), + "jm" to CountryFlag("Jamaica", ""), + "jp" to CountryFlag("Japan", ""), + "je" to CountryFlag("Jersey", ""), + "jo" to CountryFlag("Jordan", ""), + "kz" to CountryFlag("Kazakhstan", ""), + "ke" to CountryFlag("Kenya", ""), + "ki" to CountryFlag("Kiribati", ""), + "xk" to CountryFlag("Kosovo", ""), + "kw" to CountryFlag("Kuwait", ""), + "kg" to CountryFlag("Kyrgyzstan", ""), + "la" to CountryFlag("Laos", ""), + "lv" to CountryFlag("Latvia", ""), + "lb" to CountryFlag("Lebanon", ""), + "ls" to CountryFlag("Lesotho", ""), + "lr" to CountryFlag("Liberia", ""), + "ly" to CountryFlag("Libya", ""), + "li" to CountryFlag("Liechtenstein", ""), + "lt" to CountryFlag("Lithuania", ""), + "lu" to CountryFlag("Luxembourg", ""), + "mo" to CountryFlag("Macau", ""), + "mk" to CountryFlag("Macedonia", ""), + "mg" to CountryFlag("Madagascar", ""), + "mw" to CountryFlag("Malawi", ""), + "my" to CountryFlag("Malaysia", ""), + "mv" to CountryFlag("Maldives", ""), + "ml" to CountryFlag("Mali", ""), + "mt" to CountryFlag("Malta", ""), + "mh" to CountryFlag("Marshall Islands", ""), + "mq" to CountryFlag("Martinique", ""), + "mr" to CountryFlag("Mauritania", ""), + "mu" to CountryFlag("Mauritius", ""), + "yt" to CountryFlag("Mayotte", ""), + "mx" to CountryFlag("Mexico", ""), + "fm" to CountryFlag("Micronesia", ""), + "md" to CountryFlag("Moldova", ""), + "mc" to CountryFlag("Monaco", ""), + "mn" to CountryFlag("Mongolia", ""), + "me" to CountryFlag("Montenegro", ""), + "ms" to CountryFlag("Montserrat", ""), + "ma" to CountryFlag("Morocco", ""), + "mz" to CountryFlag("Mozambique", ""), + "mm" to CountryFlag("Myanmar", ""), + "na" to CountryFlag("Namibia", ""), + "nr" to CountryFlag("Nauru", ""), + "np" to CountryFlag("Nepal", ""), + "nl" to CountryFlag("Netherlands", ""), + "nc" to CountryFlag("New Caledonia", ""), + "nz" to CountryFlag("New Zealand", ""), + "ni" to CountryFlag("Nicaragua", ""), + "ne" to CountryFlag("Niger", ""), + "ng" to CountryFlag("Nigeria", ""), + "nu" to CountryFlag("Niue", ""), + "nf" to CountryFlag("Norfolk Island", ""), + "kp" to CountryFlag("North Korea", ""), + "mp" to CountryFlag("Northern Mariana Islands", ""), + "no" to CountryFlag("Norway", ""), + "om" to CountryFlag("Oman", ""), + "pk" to CountryFlag("Pakistan", ""), + "pw" to CountryFlag("Palau", ""), + "ps" to CountryFlag("Palestine", ""), + "pa" to CountryFlag("Panama", ""), + "pg" to CountryFlag("Papua New Guinea", ""), + "py" to CountryFlag("Paraguay", ""), + "pe" to CountryFlag("Peru", ""), + "ph" to CountryFlag("Philippines", ""), + "pn" to CountryFlag("Pitcairn Islands", ""), + "pl" to CountryFlag("Poland", ""), + "pt" to CountryFlag("Portugal", ""), + "pr" to CountryFlag("Puerto Rico", ""), + "qa" to CountryFlag("Qatar", ""), + "cg" to CountryFlag("Republic of the Congo", ""), + "ro" to CountryFlag("Romania", ""), + "ru" to CountryFlag("Russia", ""), + "rw" to CountryFlag("Rwanda", ""), + "re" to CountryFlag("Réunion", ""), + "bl" to CountryFlag("Saint Barthélemy", ""), + "kn" to CountryFlag("Saint Kitts and Nevis", ""), + "lc" to CountryFlag("Saint Lucia", ""), + "mf" to CountryFlag("Saint Martin", ""), + "pm" to CountryFlag("Saint Pierre and Miquelon", ""), + "vc" to CountryFlag("Saint Vincent and the Grenadines", ""), + "ws" to CountryFlag("Samoa", ""), + "sm" to CountryFlag("San Marino", ""), + "sa" to CountryFlag("Saudi Arabia", ""), + "sn" to CountryFlag("Senegal", ""), + "rs" to CountryFlag("Serbia", ""), + "sc" to CountryFlag("Seychelles", ""), + "sl" to CountryFlag("Sierra Leone", ""), + "sg" to CountryFlag("Singapore", ""), + "sx" to CountryFlag("Sint Maarten", ""), + "sk" to CountryFlag("Slovakia", ""), + "si" to CountryFlag("Slovenia", ""), + "sb" to CountryFlag("Solomon Islands", ""), + "so" to CountryFlag("Somalia", ""), + "za" to CountryFlag("South Africa", ""), + "gs" to CountryFlag("South Georgia", ""), + "kr" to CountryFlag("South Korea", ""), + "ss" to CountryFlag("South Sudan", ""), + "es" to CountryFlag("Spain", ""), + "lk" to CountryFlag("Sri Lanka", ""), + "sd" to CountryFlag("Sudan", ""), + "sr" to CountryFlag("Suriname", ""), + "sj" to CountryFlag("Svalbard and Jan Mayen", ""), + "sz" to CountryFlag("Swaziland", ""), + "se" to CountryFlag("Sweden", ""), + "ch" to CountryFlag("Switzerland", ""), + "sy" to CountryFlag("Syria", ""), + "st" to CountryFlag("São Tomé and Príncipe", ""), + "tw" to CountryFlag("Taiwan", ""), + "tj" to CountryFlag("Tajikistan", ""), + "tz" to CountryFlag("Tanzania", ""), + "th" to CountryFlag("Thailand", ""), + "tl" to CountryFlag("Timor-Leste", ""), + "tg" to CountryFlag("Togo", ""), + "tk" to CountryFlag("Tokelau", ""), + "to" to CountryFlag("Tonga", ""), + "tt" to CountryFlag("Trinidad and Tobago", ""), + "tn" to CountryFlag("Tunisia", ""), + "tr" to CountryFlag("Turkey", ""), + "tm" to CountryFlag("Turkmenistan", ""), + "tc" to CountryFlag("Turks and Caicos Islands", ""), + "tv" to CountryFlag("Tuvalu", ""), + "ug" to CountryFlag("Uganda", ""), + "ua" to CountryFlag("Ukraine", ""), + "ae" to CountryFlag("United Arab Emirates", ""), + "gb" to CountryFlag("United Kingdom", ""), + "us" to CountryFlag("United States", ""), + "um" to CountryFlag("United States Minor Outlying Islands", ""), + "vi" to CountryFlag("United States Virgin Islands", ""), + "uy" to CountryFlag("Uruguay", ""), + "uz" to CountryFlag("Uzbekistan", ""), + "vu" to CountryFlag("Vanuatu", ""), + "va" to CountryFlag("Vatican City", ""), + "ve" to CountryFlag("Venezuela", ""), + "vn" to CountryFlag("Vietnam", ""), + "wf" to CountryFlag("Wallis and Futuna", ""), + "eh" to CountryFlag("Western Sahara", ""), + "ye" to CountryFlag("Yemen", ""), + "zm" to CountryFlag("Zambia", ""), + "zw" to CountryFlag("Zimbabwe", ""), + "ax" to CountryFlag("Åland Islands", "") + ) + } +} diff --git a/android/app/src/main/java/network/mysterium/vpn/connection/ConnectionCheckerService.kt b/android/app/src/main/java/network/mysterium/ui/EditText.kt similarity index 50% rename from android/app/src/main/java/network/mysterium/vpn/connection/ConnectionCheckerService.kt rename to android/app/src/main/java/network/mysterium/ui/EditText.kt index 916ad6b6f..863574866 100644 --- a/android/app/src/main/java/network/mysterium/vpn/connection/ConnectionCheckerService.kt +++ b/android/app/src/main/java/network/mysterium/ui/EditText.kt @@ -15,25 +15,16 @@ * along with this program. If not, see . */ -package network.mysterium.vpn.connection +package network.mysterium.ui -import android.content.Intent -import android.util.Log -import com.facebook.react.HeadlessJsTaskService -import com.facebook.react.bridge.Arguments -import com.facebook.react.jstasks.HeadlessJsTaskConfig +import android.text.Editable +import android.text.TextWatcher +import android.widget.EditText -class ConnectionCheckerService : HeadlessJsTaskService() { - override fun getTaskConfig(intent: Intent?): HeadlessJsTaskConfig? { - if (intent == null) { - return null - } - val extras = intent.extras ?: return null - return HeadlessJsTaskConfig(NAME, Arguments.fromBundle(extras), TIMEOUT, true) - } - - companion object { - private const val NAME = "ConnectionChecker" - private const val TIMEOUT: Long = 60000 - } +fun EditText.onChange(cb: (String) -> Unit) { + this.addTextChangedListener(object: TextWatcher { + override fun afterTextChanged(s: Editable?) { cb(s.toString()) } + override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {} + override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {} + }) } diff --git a/android/app/src/main/java/network/mysterium/ui/FeedbackFragment.kt b/android/app/src/main/java/network/mysterium/ui/FeedbackFragment.kt new file mode 100644 index 000000000..fa67994ac --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/FeedbackFragment.kt @@ -0,0 +1,103 @@ +package network.mysterium.ui + +import android.annotation.SuppressLint +import android.os.Bundle +import android.util.Log +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import android.widget.* +import androidx.fragment.app.Fragment +import com.google.android.material.button.MaterialButton +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.launch +import network.mysterium.AppContainer +import network.mysterium.vpn.BuildConfig +import network.mysterium.vpn.R + +class FeedbackFragment : Fragment() { + private lateinit var feedbackViewModel: FeedbackViewModel + + private lateinit var feedbackBackButton: ImageView + private lateinit var feedbackTypeSpinner: Spinner + private lateinit var feedbackMessage: EditText + private lateinit var feedbackSubmitButton: MaterialButton + private lateinit var versionLabel: TextView + + override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, + savedInstanceState: Bundle?): View? { + + val root = inflater.inflate(R.layout.fragment_feedback, container, false) + val nodeRepository = AppContainer.from(activity).nodeRepository + feedbackViewModel = FeedbackViewModel(nodeRepository) + + feedbackBackButton = root.findViewById(R.id.feedback_back_button) + feedbackTypeSpinner = root.findViewById(R.id.feedback_type_spinner) + feedbackMessage = root.findViewById(R.id.feedback_message) + feedbackSubmitButton = root.findViewById(R.id.feedback_submit_button) + versionLabel = root.findViewById(R.id.vpn_version_label) + + updateVersionLabel() + + // Handle back press. + feedbackBackButton.setOnClickListener { + hideKeyboard(root) + navigateTo(root, Screen.MAIN) + } + + // Add feedback types data. + initFeedbackTypesDropdown(root) + + // Handle text change. + feedbackMessage.onChange { feedbackViewModel.setMessage(it) } + + // Handle submit. + feedbackSubmitButton.setOnClickListener { + hideKeyboard(root) + handleFeedbackSubmit(root) + } + + onBackPress { + navigateTo(root, Screen.MAIN) + } + + return root + } + + private fun handleFeedbackSubmit(root: View) { + feedbackSubmitButton.isEnabled = false + navigateTo(root, Screen.MAIN) + showMessage(root.context, getString(R.string.feedback_submit_success)) + + // Do not wait for feedback to send response as it may take some time. + CoroutineScope(Dispatchers.Main).launch { + try { + feedbackViewModel.submit() + } catch (e: Exception) { + Log.e(TAG, "Failed to send user feedback", e) + } + } + } + + private fun initFeedbackTypesDropdown(root: View) { + ArrayAdapter.createFromResource( + root.context, + R.array.feedback_types, + android.R.layout.simple_spinner_item + ).also { adapter -> + adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item) + feedbackTypeSpinner.adapter = adapter + feedbackTypeSpinner.onItemSelected { feedbackViewModel.setFeedbackType(it) } + } + } + + @SuppressLint("SetTextI18n") + private fun updateVersionLabel() { + versionLabel.text = "${BuildConfig.VERSION_NAME}.${BuildConfig.VERSION_CODE}" + } + + companion object { + private const val TAG = "FeedbackFragment" + } +} diff --git a/android/app/src/main/java/network/mysterium/ui/FeedbackViewModel.kt b/android/app/src/main/java/network/mysterium/ui/FeedbackViewModel.kt new file mode 100644 index 000000000..e3b6ab190 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/FeedbackViewModel.kt @@ -0,0 +1,61 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import androidx.lifecycle.ViewModel +import mysterium.SendFeedbackRequest +import network.mysterium.service.core.NodeRepository + +enum class FeedbackType(val type: Int) { + BUG(0), + CONNECTIVITY_ISSUE(1), + POSITIVE_FEEDBACK(2); + + override fun toString(): String { + return when(this) { + BUG -> "bug" + CONNECTIVITY_ISSUE -> "connectivity" + POSITIVE_FEEDBACK -> "positive" + } + } + + companion object { + fun parse(type: Int): FeedbackType { + return values().find { it.type == type } ?: BUG + } + } +} + +class FeedbackViewModel(private val nodeRepository: NodeRepository): ViewModel() { + private var feebackType = FeedbackType.BUG + private var message = "" + + fun setFeedbackType(type: Int) { + feebackType = FeedbackType.parse(type) + } + + fun setMessage(msg: String) { + message = msg + } + + suspend fun submit() { + val req = SendFeedbackRequest() + req.description = "Platform: Android, Feedback Type: $feebackType, Message: $message" + nodeRepository.sendFeedback(req) + } +} diff --git a/android/app/src/main/java/network/mysterium/ui/Keyboard.kt b/android/app/src/main/java/network/mysterium/ui/Keyboard.kt new file mode 100644 index 000000000..dbf24c044 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/Keyboard.kt @@ -0,0 +1,27 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import android.app.Activity +import android.view.View +import android.view.inputmethod.InputMethodManager + +fun hideKeyboard(fragmentView: View) { + val imm = fragmentView.context.getSystemService(Activity.INPUT_METHOD_SERVICE) as InputMethodManager + imm.hideSoftInputFromWindow(fragmentView.windowToken, 0) +} diff --git a/android/app/src/main/java/network/mysterium/ui/MainVpnFragment.kt b/android/app/src/main/java/network/mysterium/ui/MainVpnFragment.kt new file mode 100644 index 000000000..21744074c --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/MainVpnFragment.kt @@ -0,0 +1,260 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import android.content.Context +import android.os.Bundle +import android.util.Log +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import android.widget.ImageView +import android.widget.RelativeLayout +import android.widget.TextView +import androidx.constraintlayout.widget.ConstraintLayout +import androidx.fragment.app.Fragment +import androidx.lifecycle.Observer +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.Job +import kotlinx.coroutines.launch +import network.mysterium.MainApplication +import network.mysterium.vpn.R + +class MainVpnFragment : Fragment() { + private lateinit var sharedViewModel: SharedViewModel + private lateinit var proposalsViewModel: ProposalsViewModel + + private var job: Job? = null + private lateinit var connStatusLabel: TextView + private lateinit var conStatusIP: TextView + private lateinit var vpnStatusCountry: ImageView + private lateinit var selectProposalLayout: ConstraintLayout + private lateinit var feedbackButton: ImageView + private lateinit var vpnSelectedProposalCountryLabel: TextView + private lateinit var vpnSelectedProposalProviderLabel: TextView + private lateinit var vpnSelectedProposalCountryIcon: ImageView + private lateinit var vpnProposalPickerFavoriteLayput: RelativeLayout + private lateinit var vpnProposalPickerFavoriteImage: ImageView + private lateinit var connectionButton: TextView + private lateinit var vpnStatsDurationLabel: TextView + private lateinit var vpnStatsBytesSentLabel: TextView + private lateinit var vpnStatsBytesReceivedLabel: TextView + + override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, + savedInstanceState: Bundle?): View? { + + val appContainer = (activity!!.application as MainApplication).appContainer + sharedViewModel = appContainer.sharedViewModel + proposalsViewModel = appContainer.proposalsViewModel + + val root = inflater.inflate(R.layout.fragment_main_vpn, container, false) + + // Initialize UI elements. + connStatusLabel = root.findViewById(R.id.vpn_status_label) + conStatusIP = root.findViewById(R.id.vpn_status_ip) + vpnStatusCountry = root.findViewById(R.id.vpn_status_country) + selectProposalLayout = root.findViewById(R.id.vpn_select_proposal_layout) + feedbackButton = root.findViewById(R.id.vpn_feedback_button) + vpnSelectedProposalCountryLabel = root.findViewById(R.id.vpn_selected_proposal_country_label) + vpnSelectedProposalProviderLabel = root.findViewById(R.id.vpn_selected_proposal_provider_label) + vpnSelectedProposalCountryIcon = root.findViewById(R.id.vpn_selected_proposal_country_icon) + vpnProposalPickerFavoriteLayput = root.findViewById(R.id.vpn_proposal_picker_favorite_layout) + vpnProposalPickerFavoriteImage = root.findViewById(R.id.vpn_proposal_picker_favorite_image) + connectionButton = root.findViewById(R.id.vpn_connection_button) + vpnStatsDurationLabel = root.findViewById(R.id.vpn_stats_duration) + vpnStatsBytesReceivedLabel = root.findViewById(R.id.vpn_stats_bytes_received) + vpnStatsBytesSentLabel = root.findViewById(R.id.vpn_stats_bytes_sent) + + feedbackButton.setOnClickListener { + navigateTo(root, Screen.FEEDBACK) + } + + selectProposalLayout.setOnClickListener { + handleSelectProposalPress(root) + } + + vpnProposalPickerFavoriteLayput.setOnClickListener { + handleFavoriteProposalPress(root) + } + + connectionButton.setOnClickListener { + handleConnectionPress(root.context) + } + + sharedViewModel.selectedProposal.observe(this, Observer { updateSelectedProposal(it) }) + + sharedViewModel.connectionState.observe(this, Observer { + updateConnStateLabel(it) + updateConnButtonState(it) + }) + + sharedViewModel.statistics.observe(this, Observer { updateStatsLabels(it) }) + + sharedViewModel.location.observe(this, Observer { updateLocation(it) }) + + onBackPress { emulateHomePress() } + + return root + } + + override fun onDestroy() { + super.onDestroy() + job?.cancel() + } + + private fun updateLocation(it: LocationViewItem) { + conStatusIP.text = "IP: ${it.ip}" + if (it.countryFlagImage == null) { + vpnStatusCountry.setImageResource(R.drawable.ic_public_black_24dp) + } else { + vpnStatusCountry.setImageBitmap(it.countryFlagImage) + } + } + + private fun updateSelectedProposal(it: ProposalViewItem) { + vpnSelectedProposalCountryLabel.text = it.countryName + vpnSelectedProposalCountryIcon.setImageBitmap(it.countryFlagImage) + vpnSelectedProposalProviderLabel.text = it.providerID + vpnSelectedProposalProviderLabel.visibility = View.VISIBLE + vpnProposalPickerFavoriteImage.setImageResource(it.isFavoriteResID) + } + + private fun updateStatsLabels(it: StatisticsViewItem) { + vpnStatsDurationLabel.text = it.duration + vpnStatsBytesReceivedLabel.text = it.bytesReceived + vpnStatsBytesSentLabel.text = it.bytesSent + } + + private fun updateConnStateLabel(it: ConnectionState) { + val connStateText = when (it) { + ConnectionState.NOT_CONNECTED, ConnectionState.UNKNOWN -> getString(R.string.conn_state_not_connected) + ConnectionState.CONNECTED -> getString(R.string.conn_state_connected) + ConnectionState.CONNECTING -> getString(R.string.conn_state_connecting) + ConnectionState.DISCONNECTING -> getString(R.string.conn_state_disconnecting) + } + + connStatusLabel.text = connStateText + } + + private fun handleSelectProposalPress(root: View) { + navigateToProposals(root) + } + + private fun handleFavoriteProposalPress(root: View) { + val selectedProposal = sharedViewModel.selectedProposal.value + if (selectedProposal == null) { + navigateToProposals(root) + return + } + + vpnProposalPickerFavoriteLayput.isEnabled = false + proposalsViewModel.toggleFavoriteProposal(selectedProposal.id) { updatedProposal -> + if (updatedProposal != null) { + vpnProposalPickerFavoriteImage.setImageResource(updatedProposal.isFavoriteResID) + } + + vpnProposalPickerFavoriteLayput.isEnabled = true + } + } + + private fun navigateToProposals(root: View) { + if (sharedViewModel.canConnect()) { + navigateTo(root, Screen.PROPOSALS) + } else { + showMessage(root.context, getString(R.string.disconnect_to_select_proposal)) + } + } + + private fun updateConnButtonState(it: ConnectionState) { + connectionButton.text = when (it) { + ConnectionState.NOT_CONNECTED, ConnectionState.UNKNOWN -> getString(R.string.connect_button_connect) + ConnectionState.CONNECTED -> getString(R.string.connect_button_disconnect) + ConnectionState.CONNECTING -> getString(R.string.connect_button_cancel) + ConnectionState.DISCONNECTING -> getString(R.string.connect_button_disconnecting) + } + + connectionButton.isEnabled = when (it) { + ConnectionState.DISCONNECTING -> false + else -> true + } + } + + private fun handleConnectionPress(ctx: Context) { + if (sharedViewModel.canConnect()) { + connect(ctx) + return + } + + if (sharedViewModel.canDisconnect()) { + disconnect(ctx) + return + } + + cancel() + } + + private fun connect(ctx: Context) { + val proposal: ProposalViewItem? = sharedViewModel.selectedProposal.value + if (proposal == null) { + showMessage(ctx, "Select proposal!") + return + } + job?.cancel() + connectionButton.isEnabled = false + job = CoroutineScope(Dispatchers.Main).launch { + try { + sharedViewModel.connect(proposal.providerID, proposal.serviceType.type) + } catch (e: kotlinx.coroutines.CancellationException) { + // Do nothing. + } catch (e: Exception) { + showMessage(ctx, "Failed to connect. Please try again.") + Log.e(TAG, "Failed to connect", e) + } + } + } + + private fun disconnect(ctx: Context) { + connectionButton.isEnabled = false + job?.cancel() + job = CoroutineScope(Dispatchers.Main).launch { + try { + sharedViewModel.disconnect() + } catch (e: Exception) { + showMessage(ctx, "Failed to disconnect. Please try again.") + Log.e(TAG, "Failed to disconnect", e) + } + } + } + + private fun cancel() { + connectionButton.isEnabled = false + job?.cancel() + job = CoroutineScope(Dispatchers.Main).launch { + try { + sharedViewModel.disconnect() + } catch (e: Exception) { + Log.e(TAG, "Failed to cancel", e) + } + } + } + + companion object { + private const val TAG = "ProposalsFragment" + } +} diff --git a/android/app/src/main/java/network/mysterium/logging/BugReporterPackage.kt b/android/app/src/main/java/network/mysterium/ui/MaterialSpinner.kt similarity index 55% rename from android/app/src/main/java/network/mysterium/logging/BugReporterPackage.kt rename to android/app/src/main/java/network/mysterium/ui/MaterialSpinner.kt index 5b04e84b2..9df46074b 100644 --- a/android/app/src/main/java/network/mysterium/logging/BugReporterPackage.kt +++ b/android/app/src/main/java/network/mysterium/ui/MaterialSpinner.kt @@ -15,21 +15,20 @@ * along with this program. If not, see . */ -package network.mysterium.logging +package network.mysterium.ui -import com.facebook.react.ReactPackage -import com.facebook.react.bridge.NativeModule -import com.facebook.react.bridge.ReactApplicationContext -import com.facebook.react.uimanager.ViewManager +import android.view.View +import android.widget.AdapterView +import android.widget.Spinner -class BugReporterPackage : ReactPackage { - override fun createNativeModules(reactContext: ReactApplicationContext): MutableList { - return mutableListOf( - BugReporter(reactContext) - ) - } +fun Spinner.onItemSelected(cb: (position: Int) -> Unit) { + this.onItemSelectedListener = object : AdapterView.OnItemSelectedListener { + override fun onItemSelected(parent: AdapterView<*>?, view: View?, position: Int, id: Long) { + cb(position) + } + + override fun onNothingSelected(parent: AdapterView<*>?) { + } - override fun createViewManagers(reactContext: ReactApplicationContext): MutableList> { - return mutableListOf() } } diff --git a/android/app/src/main/java/network/mysterium/ui/Navigation.kt b/android/app/src/main/java/network/mysterium/ui/Navigation.kt new file mode 100644 index 000000000..e7f9b393b --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/Navigation.kt @@ -0,0 +1,42 @@ +package network.mysterium.ui + +import android.content.Intent +import android.view.View +import androidx.activity.OnBackPressedCallback +import androidx.fragment.app.Fragment +import androidx.navigation.findNavController +import network.mysterium.vpn.R + +enum class Screen { + MAIN, + FEEDBACK, + PROPOSALS +} + +fun navigateTo(view: View, destination: Screen) { + val navController = view.findNavController() + val to = when(destination) { + Screen.MAIN -> R.id.action_go_to_vpn_screen + Screen.FEEDBACK -> R.id.action_go_to_feedback_screen + Screen.PROPOSALS -> R.id.action_go_to_proposals_screen + } + navController.navigate(to) +} + +fun Fragment.onBackPress(cb: () -> Unit) { + val callback = object : OnBackPressedCallback(true) { + override fun handleOnBackPressed() { + cb() + } + } + this.requireActivity().onBackPressedDispatcher.addCallback(this.viewLifecycleOwner, callback) +} + +// Default back behaviour fully closes app, but we only want to minimize it. +// To do so we emulate home button press. +fun Fragment.emulateHomePress() { + val startMain = Intent(Intent.ACTION_MAIN) + startMain.addCategory(Intent.CATEGORY_HOME) + startMain.flags = Intent.FLAG_ACTIVITY_NEW_TASK + this.startActivity(startMain) +} diff --git a/android/app/src/main/java/network/mysterium/ui/ProposalViewItem.kt b/android/app/src/main/java/network/mysterium/ui/ProposalViewItem.kt new file mode 100644 index 000000000..53208499d --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/ProposalViewItem.kt @@ -0,0 +1,89 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import android.graphics.Bitmap +import network.mysterium.service.core.ProposalItem +import network.mysterium.db.FavoriteProposal +import network.mysterium.vpn.R + +class ProposalViewItem constructor( + val id: String, + val providerID: String, + val serviceType: ServiceType, + val countryCode: String +) { + var countryFlagImage: Bitmap? = null + var serviceTypeResID: Int = R.drawable.service_openvpn + var qualityResID: Int = R.drawable.quality_unknown + var qualityLevel: QualityLevel = QualityLevel.UNKNOWN + var countryName: String = "" + var isFavorite: Boolean = false + var isFavoriteResID: Int = R.drawable.ic_star_border_black_24dp + + fun toggleFavorite() { + isFavorite = !isFavorite + isFavoriteResID = if (isFavorite) { + R.drawable.ic_star_black_24dp + } else { + R.drawable.ic_star_border_black_24dp + } + } + + companion object { + fun parse(it: ProposalItem, favoriteProposals: Map): ProposalViewItem { + val res = ProposalViewItem( + id = it.providerID+it.serviceType, + providerID = it.providerID, + serviceType = ServiceType.parse(it.serviceType), + countryCode = it.countryCode.toLowerCase()) + + if (Countries.bitmaps.contains(res.countryCode)) { + res.countryFlagImage = Countries.bitmaps[res.countryCode] + res.countryName = Countries.values[res.countryCode]?.name ?: "" + } + + res.serviceTypeResID = mapServiceTypeResourceID(res.serviceType) + res.qualityLevel = QualityLevel.parse(it.qualityLevel) + res.qualityResID = mapQualityLevelResourceID(res.qualityLevel) + res.isFavorite = favoriteProposals.containsKey(res.id) + if (res.isFavorite) { + res.isFavoriteResID = R.drawable.ic_star_black_24dp + } + + return res + } + + private fun mapServiceTypeResourceID(serviceType: ServiceType): Int { + return when(serviceType) { + ServiceType.OPENVPN -> R.drawable.service_openvpn + ServiceType.WIREGUARD -> R.drawable.service_wireguard + else -> R.drawable.service_openvpn + } + } + + private fun mapQualityLevelResourceID(qualityLevel: QualityLevel): Int { + return when(qualityLevel) { + QualityLevel.HIGH -> R.drawable.quality_high + QualityLevel.MEDIUM -> R.drawable.quality_medium + QualityLevel.LOW -> R.drawable.quality_low + QualityLevel.UNKNOWN -> R.drawable.quality_unknown + } + } + } +} diff --git a/android/app/src/main/java/network/mysterium/ui/ProposalsFragment.kt b/android/app/src/main/java/network/mysterium/ui/ProposalsFragment.kt new file mode 100644 index 000000000..3be88c09c --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/ProposalsFragment.kt @@ -0,0 +1,251 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import android.graphics.Color +import android.os.Bundle +import android.util.Log +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import android.widget.* +import androidx.core.content.ContextCompat +import androidx.fragment.app.Fragment +import androidx.lifecycle.Observer +import androidx.recyclerview.widget.DividerItemDecoration +import androidx.recyclerview.widget.LinearLayoutManager +import androidx.recyclerview.widget.RecyclerView +import androidx.swiperefreshlayout.widget.SwipeRefreshLayout +import com.makeramen.roundedimageview.RoundedImageView +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.launch +import network.mysterium.AppContainer +import network.mysterium.MainApplication +import network.mysterium.vpn.R + +class ProposalsFragment : Fragment() { + + private lateinit var proposalsViewModel: ProposalsViewModel + private lateinit var appContainer: AppContainer + + private lateinit var proposalsCloseButton: TextView + private lateinit var proposalsSearchInput: EditText + private lateinit var proposalsFiltersAllButton: TextView + private lateinit var proposalsFiltersOpenvpnButton: TextView + private lateinit var proposalsFiltersWireguardButton: TextView + private lateinit var proposalsFiltersSort: Spinner + private lateinit var proposalsSwipeRefresh: SwipeRefreshLayout + private lateinit var proposalsList: RecyclerView + private lateinit var proposalsProgressBar: ProgressBar + + override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? { + + val root = inflater.inflate(R.layout.fragment_proposals, container, false) + + appContainer = (activity!!.application as MainApplication).appContainer + proposalsViewModel = appContainer.proposalsViewModel + + // Initialize UI elements. + proposalsCloseButton = root.findViewById(R.id.proposals_close_button) + proposalsSearchInput = root.findViewById(R.id.proposals_search_input) + proposalsFiltersAllButton = root.findViewById(R.id.proposals_filters_all_button) + proposalsFiltersOpenvpnButton = root.findViewById(R.id.proposals_filters_openvpn_button) + proposalsFiltersWireguardButton = root.findViewById(R.id.proposals_filters_wireguard_button) + proposalsFiltersSort = root.findViewById(R.id.proposals_filters_sort) + proposalsSwipeRefresh = root.findViewById(R.id.proposals_list_swipe_refresh) + proposalsList = root.findViewById(R.id.proposals_list) + proposalsProgressBar = root.findViewById(R.id.proposals_progress_bar) + + proposalsCloseButton.setOnClickListener { handleClose(root) } + + initProposalsList(root) + initProposalsSortDropdown(root) + initProposalsServiceTypeFilter(root) + initProposalsSearchFilter() + + onBackPress { + navigateTo(root, Screen.MAIN) + } + + return root + } + + private fun navigateToMainVpnFragment(root: View) { + navigateTo(root, Screen.MAIN) + } + + private fun initProposalsSearchFilter() { + if (proposalsViewModel.filter.searchText != "") { + proposalsSearchInput.setText(proposalsViewModel.filter.searchText) + } + + proposalsSearchInput.onChange { proposalsViewModel.filterBySearchText(it) } + } + + private fun initProposalsServiceTypeFilter(root: View) { + // Set current active filter. + val activeTabButton = when (proposalsViewModel.filter.serviceType) { + ServiceTypeFilter.ALL -> proposalsFiltersAllButton + ServiceTypeFilter.OPENVPN -> proposalsFiltersOpenvpnButton + ServiceTypeFilter.WIREGUARD -> proposalsFiltersWireguardButton + } + setFilterTabActiveStyle(root, activeTabButton) + + proposalsFiltersAllButton.setOnClickListener { + proposalsViewModel.filterByServiceType(ServiceTypeFilter.ALL) + setFilterTabActiveStyle(root, proposalsFiltersAllButton) + setFilterTabInactiveStyle(root, proposalsFiltersOpenvpnButton) + setFilterTabInactiveStyle(root, proposalsFiltersWireguardButton) + } + + proposalsFiltersOpenvpnButton.setOnClickListener { + proposalsViewModel.filterByServiceType(ServiceTypeFilter.OPENVPN) + setFilterTabActiveStyle(root, proposalsFiltersOpenvpnButton) + setFilterTabInactiveStyle(root, proposalsFiltersAllButton) + setFilterTabInactiveStyle(root, proposalsFiltersWireguardButton) + } + + proposalsFiltersWireguardButton.setOnClickListener { + proposalsViewModel.filterByServiceType(ServiceTypeFilter.WIREGUARD) + setFilterTabActiveStyle(root, proposalsFiltersWireguardButton) + setFilterTabInactiveStyle(root, proposalsFiltersAllButton) + setFilterTabInactiveStyle(root, proposalsFiltersOpenvpnButton) + } + } + + private fun initProposalsSortDropdown(root: View) { + ArrayAdapter.createFromResource(root.context, R.array.proposals_sort_types, android.R.layout.simple_spinner_item).also { adapter -> + adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item) + proposalsFiltersSort.adapter = adapter + proposalsFiltersSort.onItemSelected { item -> proposalsViewModel.sortBy(item) } + } + } + + private fun initProposalsList(root: View) { + proposalsList.layoutManager = LinearLayoutManager(root.context) + val items = ArrayList() + val proposalsListAdapter = ProposalsListAdapter(items) { handleSelectedProposal(root, it) } + proposalsList.adapter = proposalsListAdapter + proposalsList.addItemDecoration(DividerItemDecoration(root.context, DividerItemDecoration.VERTICAL)) + + proposalsSwipeRefresh.setOnRefreshListener { + proposalsViewModel.refreshProposals { + proposalsSwipeRefresh.isRefreshing = false + } + } + + // Subscribe to proposals changes. + proposalsViewModel.getProposals().observe(this, Observer { newItems -> + items.clear() + items.addAll(newItems) + proposalsListAdapter.notifyDataSetChanged() + + // Hide progress bar once proposals are loaded. + proposalsList.visibility = View.VISIBLE + proposalsProgressBar.visibility = View.GONE + }) + + // Subscribe to proposals counters. + proposalsViewModel.getProposalsCounts().observe(this, Observer { counts -> + proposalsFiltersAllButton.text = "All (${counts.all})" + proposalsFiltersOpenvpnButton.text = "Openvpn (${counts.openvpn})" + proposalsFiltersWireguardButton.text = "Wireguard (${counts.wireguard})" + }) + + proposalsViewModel.initialProposalsLoaded.observe(this, Observer {loaded -> + if (loaded) { + return@Observer + } + + // If initial proposals failed to load during app init try to load them explicitly. + proposalsList.visibility = View.GONE + proposalsProgressBar.visibility = View.VISIBLE + proposalsViewModel.refreshProposals {} + }) + } + + private fun handleClose(root: View) { + hideKeyboard(root) + navigateToMainVpnFragment(root) + } + + private fun handleSelectedProposal(root: View, proposal: ProposalViewItem) { + hideKeyboard(root) + proposalsViewModel.selectProposal(proposal) + navigateToMainVpnFragment(root) + } + + private fun setFilterTabActiveStyle(root: View, btn: TextView) { + btn.setBackgroundColor(ContextCompat.getColor(root.context, R.color.ColorMain)) + btn.setTextColor(ContextCompat.getColor(root.context, R.color.ColorWhite)) + } + + private fun setFilterTabInactiveStyle(root: View, btn: TextView) { + btn.setBackgroundColor(Color.TRANSPARENT) + btn.setTextColor(ContextCompat.getColor(root.context, R.color.ColorMain)) + } +} + +class ProposalsListAdapter(private var list: List, private var onItemClickListener: (ProposalViewItem) -> Unit) + : RecyclerView.Adapter() { + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ProposalViewHolder { + val inflater = LayoutInflater.from(parent.context) + return ProposalViewHolder(inflater, parent) + } + + override fun onBindViewHolder(holder: ProposalViewHolder, position: Int) { + val item: ProposalViewItem = list[position] + holder.bind(item) + holder.itemView.setOnClickListener { + onItemClickListener(item) + } + } + + override fun getItemCount(): Int = list.size + + inner class ProposalViewHolder(inflater: LayoutInflater, parent: ViewGroup) : + RecyclerView.ViewHolder(inflater.inflate(R.layout.proposal_list_item, parent, false)) { + + private var countryFlag: RoundedImageView? = null + private var countryName: TextView? = null + private var providerID: TextView? = null + private var serviceType: ImageView? = null + private var qualityLevel: ImageView? = null + private var favorite: ImageView? = null + + init { + countryFlag = itemView.findViewById(R.id.proposal_item_country_flag) + countryName = itemView.findViewById(R.id.proposal_item_country_name) + providerID = itemView.findViewById(R.id.proposal_item_provider_id) + serviceType = itemView.findViewById(R.id.proposal_item_service_type) + qualityLevel = itemView.findViewById(R.id.proposal_item_quality_level) + favorite = itemView.findViewById(R.id.proposal_item_favorite) + } + + fun bind(item: ProposalViewItem) { + countryFlag?.setImageBitmap(item.countryFlagImage) + countryName?.text = item.countryName + providerID?.text = item.providerID + serviceType?.setImageResource(item.serviceTypeResID) + qualityLevel?.setImageResource(item.qualityResID) + favorite?.setImageResource(item.isFavoriteResID) + } + } +} diff --git a/android/app/src/main/java/network/mysterium/ui/ProposalsViewModel.kt b/android/app/src/main/java/network/mysterium/ui/ProposalsViewModel.kt new file mode 100644 index 000000000..2ad6e4de6 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/ProposalsViewModel.kt @@ -0,0 +1,243 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import android.util.Log +import androidx.lifecycle.LiveData +import androidx.lifecycle.MutableLiveData +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import kotlinx.coroutines.launch +import network.mysterium.service.core.NodeRepository +import network.mysterium.db.AppDatabase +import network.mysterium.db.FavoriteProposal + +enum class ServiceType(val type: String) { + UNKNOWN("unknown"), + OPENVPN("openvpn"), + WIREGUARD("wireguard"); + + companion object { + fun parse(type: String): ServiceType { + return values().find { it.type == type } ?: UNKNOWN + } + } +} + +enum class ServiceTypeFilter(val type: String) { + ALL("all"), + OPENVPN("openvpn"), + WIREGUARD("wireguard") +} + +enum class ProposalSortType(val type: Int) { + COUNTRY(0), + QUALITY(1); + + companion object { + fun parse(type: Int): ProposalSortType { + if (type == 0) { + return COUNTRY + } + return QUALITY + } + } +} + +enum class QualityLevel(val level: Int) { + UNKNOWN(0), + LOW(1), + MEDIUM(2), + HIGH(3); + + companion object { + fun parse(level: Int): QualityLevel { + return values().find { it.level == level } ?: UNKNOWN + } + } +} + +class ProposalsCounts( + val openvpn: Int, + val wireguard: Int +) { + val all: Int + get() = openvpn + wireguard +} + +class ProposalsFilter( + var serviceType: ServiceTypeFilter = ServiceTypeFilter.ALL, + var searchText: String = "", + var sortBy: ProposalSortType = ProposalSortType.COUNTRY +) + +class ProposalsViewModel(private val sharedViewModel: SharedViewModel, private val nodeRepository: NodeRepository, private val appDatabase: AppDatabase) : ViewModel() { + var filter = ProposalsFilter() + var initialProposalsLoaded = MutableLiveData() + + private var favoriteProposals: MutableMap = mutableMapOf() + private var allProposals: List = listOf() + private val proposals = MutableLiveData>() + private val proposalsCounts = MutableLiveData() + + suspend fun loadFavoriteProposals(): Map { + val favorites = appDatabase.favoriteProposalDao().getAll() + favoriteProposals = favorites.map { it.id to it }.toMap().toMutableMap() + return favoriteProposals + } + + suspend fun load() { + loadInitialProposals() + } + + fun getProposals(): LiveData> { + return proposals + } + + fun getProposalsCounts(): LiveData { + return proposalsCounts + } + + fun filterByServiceType(type: ServiceTypeFilter) { + if (filter.serviceType == type) { + return + } + + filter.serviceType = type + proposals.value = filterAndSortProposals(filter, allProposals) + } + + fun filterBySearchText(value: String) { + val searchText = value.toLowerCase() + if (filter.searchText == searchText) { + return + } + + filter.searchText = searchText + proposals.value = filterAndSortProposals(filter, allProposals) + } + + fun sortBy(type: Int) { + val sortBy = ProposalSortType.parse(type) + if (filter.sortBy == sortBy) { + return + } + + filter.sortBy = sortBy + proposals.value = filterAndSortProposals(filter, allProposals) + } + + fun refreshProposals(done: () -> Unit) { + viewModelScope.launch { + loadInitialProposals(refresh = true) + done() + } + } + + fun selectProposal(item: ProposalViewItem) { + sharedViewModel.selectProposal(item) + } + + fun toggleFavoriteProposal(proposalID: String, done: (updatedProposal: ProposalViewItem?) -> Unit) { + viewModelScope.launch { + val proposal = allProposals.find { it.id == proposalID } + if (proposal == null) { + done(null) + return@launch + } + + val favoriteProposal = FavoriteProposal(proposalID) + if (proposal.isFavorite) { + deleteFavoriteProposal(favoriteProposal) + } else { + insertFavoriteProposal(favoriteProposal) + } + + proposal.toggleFavorite() + proposals.value = filterAndSortProposals(filter, allProposals) + done(proposal) + } + } + + private suspend fun insertFavoriteProposal(proposal: FavoriteProposal) { + try { + appDatabase.favoriteProposalDao().insert(proposal) + favoriteProposals[proposal.id] = proposal + } catch (e: Exception) { + Log.e(TAG, "Failed to insert favorite proposal", e) + } + } + + private suspend fun deleteFavoriteProposal(proposal: FavoriteProposal) { + try { + appDatabase.favoriteProposalDao().delete(proposal) + favoriteProposals.remove(proposal.id) + } catch (e: Exception) { + Log.e(TAG, "Failed to delete favorite proposal", e) + } + } + + private suspend fun loadInitialProposals(refresh: Boolean = false) { + try { + val nodeProposals = nodeRepository.getProposals(refresh) + allProposals = nodeProposals.map { ProposalViewItem.parse(it, favoriteProposals) } + + proposalsCounts.value = ProposalsCounts( + openvpn = allProposals.count { it.serviceType == ServiceType.OPENVPN }, + wireguard = allProposals.count { it.serviceType == ServiceType.WIREGUARD } + ) + proposals.value = filterAndSortProposals(filter, allProposals) + initialProposalsLoaded.value = true + } catch (e: Exception) { + Log.e(TAG, "Failed to load initial proposals", e) + proposals.value = listOf() + initialProposalsLoaded.value = false + } + } + + private fun filterAndSortProposals(filter: ProposalsFilter, allProposals: List): List { + return allProposals.asSequence() + // Filter by service type. + .filter { + when (filter.serviceType) { + ServiceTypeFilter.OPENVPN -> it.serviceType == ServiceType.OPENVPN + ServiceTypeFilter.WIREGUARD -> it.serviceType == ServiceType.WIREGUARD + else -> true + } + } + // Filter by search value. + .filter { + when (filter.searchText) { + "" -> true + else -> it.countryName.toLowerCase().contains(filter.searchText) or it.providerID.contains(filter.searchText) + } + } + // Sort by country or quality. + .sortedWith( + if (filter.sortBy == ProposalSortType.QUALITY) + compareByDescending { it.qualityLevel } + else + compareBy { it.countryName } + ) + .toList() + } + + companion object { + const val TAG: String = "ProposalsViewModel" + } +} diff --git a/android/app/src/main/java/network/mysterium/ui/SharedViewModel.kt b/android/app/src/main/java/network/mysterium/ui/SharedViewModel.kt new file mode 100644 index 000000000..8dd61c4a0 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/SharedViewModel.kt @@ -0,0 +1,247 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import android.graphics.Bitmap +import android.util.Log +import androidx.lifecycle.MutableLiveData +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import kotlinx.coroutines.* +import mysterium.ConnectRequest +import network.mysterium.db.FavoriteProposal +import network.mysterium.service.core.NodeRepository +import network.mysterium.service.core.Statistics +import network.mysterium.logging.BugReporter +import network.mysterium.service.core.MysteriumCoreService +import network.mysterium.service.core.Status + +enum class ConnectionState(val type: String) { + UNKNOWN("Unknown"), + CONNECTED("Connected"), + CONNECTING("Connecting"), + NOT_CONNECTED("NotConnected"), + DISCONNECTING("Disconnecting"); + + companion object { + fun parse(type: String): ConnectionState { + return values().find { it.type == type } ?: UNKNOWN + } + } +} + +class LocationViewItem( + val ip: String, + val countryFlagImage: Bitmap? +) + +class StatisticsViewItem( + val duration: String, + val bytesReceived: String, + val bytesSent: String +) { + companion object { + fun from(it: Statistics): StatisticsViewItem { + return StatisticsViewItem( + duration = UnitFormatter.timeDisplay(it.duration.toDouble()), + bytesReceived = UnitFormatter.bytesDisplay(it.bytesReceived.toDouble()), + bytesSent = UnitFormatter.bytesDisplay(it.bytesSent.toDouble()) + ) + } + } +} + +class SharedViewModel( + private val nodeRepository: NodeRepository, + private val bugReporter: BugReporter, + private val mysteriumCoreService: CompletableDeferred +) : ViewModel() { + + val selectedProposal = MutableLiveData() + val connectionState = MutableLiveData() + val statistics = MutableLiveData() + val location = MutableLiveData() + + private var isConnected = false + + suspend fun load(favoriteProposals: Map) { + unlockIdentity() + initListeners() + loadLocation() + val status = loadCurrentStatus() + loadActiveProposal(status, favoriteProposals) + } + + fun selectProposal(item: ProposalViewItem) { + selectedProposal.value = item + } + + fun canConnect(): Boolean { + val state = connectionState.value + return state == null || state == ConnectionState.NOT_CONNECTED || state == ConnectionState.UNKNOWN + } + + fun canDisconnect(): Boolean { + val state = connectionState.value + return state != null && state == ConnectionState.CONNECTED + } + + suspend fun connect(providerID: String, serviceType: String) { + try { + connectionState.value = ConnectionState.CONNECTING + // Before doing actual connection add some delay to prevent + // from trying to establish connection if user instantly clicks CANCEL. + delay(1000) + val req = ConnectRequest() + req.providerID = providerID + req.serviceType = serviceType + nodeRepository.connect(req) + isConnected = true + connectionState.value = ConnectionState.CONNECTED + loadLocation() + } catch (e: Exception) { + isConnected = false + connectionState.value = ConnectionState.NOT_CONNECTED + throw e + } + } + + suspend fun disconnect() { + try { + connectionState.value = ConnectionState.DISCONNECTING + nodeRepository.disconnect() + isConnected = false + connectionState.value = ConnectionState.NOT_CONNECTED + resetStatistics() + loadLocation() + } catch (e: Exception) { + connectionState.value = ConnectionState.NOT_CONNECTED + throw e + } finally { + mysteriumCoreService.await().hideNotifications() + } + } + + private suspend fun loadCurrentStatus(): Status? { + return try { + val status = nodeRepository.getStatus() + val state = ConnectionState.parse(status.state) + connectionState.value = state + status + } catch (e: Exception) { + Log.e(TAG, "Failed to load current status", e) + null + } + } + + private suspend fun loadActiveProposal(it: Status?, favoriteProposals: Map) { + if (it == null || it.providerID == "" || it.serviceType == "") { + return + } + + try { + val proposal = nodeRepository.getProposal(it.providerID, it.serviceType) ?: return + val proposalViewItem = ProposalViewItem.parse(proposal, favoriteProposals) + selectProposal(proposalViewItem) + } catch (e: Exception) { + Log.e(TAG, "Failed to load active proposal", e) + } + } + + // initListeners subscribes to go node library exposed callbacks for statistics and state. + private suspend fun initListeners() { + nodeRepository.registerConnectionStatusChangeCallback { + handleConnectionStatusChange(it) + } + + nodeRepository.registerStatisticsChangeCallback { + handleStatisticsChange(it) + } + } + + private fun handleConnectionStatusChange(it: String) { + val newState = ConnectionState.parse(it) + val currentState = connectionState.value + + // Update all UI related state in new coroutine on UI thread. + // This is needed since status change can be executed on separate + // inside go node library. + viewModelScope.launch { + connectionState.value = newState + if (currentState == ConnectionState.CONNECTED && newState != currentState) { + mysteriumCoreService.await().showNotification("Connection lost", "VPN connection was closed.") + resetStatistics() + loadLocation() + } + } + } + + private fun handleStatisticsChange(it: Statistics) { + // Update all UI related state in new coroutine on UI thread. + // This is needed since status change can be executed on separate + // inside go node library. + viewModelScope.launch { + val s = StatisticsViewItem( + duration = UnitFormatter.timeDisplay(it.duration.toDouble()), + bytesReceived = UnitFormatter.bytesDisplay(it.bytesReceived.toDouble()), + bytesSent = UnitFormatter.bytesDisplay(it.bytesSent.toDouble()) + ) + statistics.value = s + + // Show global notification with connected country and statistics. + // At this point we need to check if proposal is not null since + // statistics event can fire sooner than proposal is loaded. + if (selectedProposal.value != null) { + val countryName = selectedProposal.value?.countryName + mysteriumCoreService.await().showNotification("Connected to $countryName", "Received ${s.bytesReceived} | Send ${s.bytesSent}") + } + } + } + + private suspend fun unlockIdentity() { + try { + val identity = nodeRepository.unlockIdentity() + bugReporter.setUserIdentifier(identity) + } catch (e: Exception) { + Log.e(TAG, "Failed not unlock identity", e) + } + } + + private suspend fun loadLocation() { + // Try to load location with few attempts. It can fail to load when connected to VPN. + location.value = LocationViewItem(ip = "Updating", countryFlagImage = null) + for (i in 1..3) { + try { + val loc = nodeRepository.getLocation() + location.value = LocationViewItem(ip = loc.ip, countryFlagImage = Countries.bitmaps[loc.countryCode.toLowerCase()]) + break + } catch (e: Exception) { + delay(1000) + Log.e(TAG, "Failed to load location. Attempt $i.", e) + } + } + } + + private fun resetStatistics() { + statistics.value = StatisticsViewItem.from(Statistics(0, 0, 0)) + } + + companion object { + const val TAG = "SharedViewModel" + } +} diff --git a/android/app/src/main/java/network/mysterium/ui/TermsFragment.kt b/android/app/src/main/java/network/mysterium/ui/TermsFragment.kt new file mode 100644 index 000000000..86e17e6c8 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/TermsFragment.kt @@ -0,0 +1,64 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import android.os.Bundle +import android.text.Html +import androidx.fragment.app.Fragment +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import android.widget.Button +import android.widget.TextView +import androidx.navigation.findNavController +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.launch +import network.mysterium.MainApplication +import network.mysterium.vpn.R + +class TermsFragment : Fragment() { + private lateinit var termsViewModel: TermsViewModel + + private lateinit var termsTextWiew: TextView + private lateinit var termsAcceptButton: Button + + override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, + savedInstanceState: Bundle?): View? { + + val root = inflater.inflate(R.layout.fragment_terms, container, false) + + termsViewModel = (activity!!.application as MainApplication).appContainer.termsViewModel + termsTextWiew = root.findViewById(R.id.terms_text_wiew) + termsAcceptButton = root.findViewById(R.id.terms_accept_button) + + termsTextWiew.setText(Html.fromHtml(termsViewModel.termsText)) + + termsAcceptButton.setOnClickListener { + termsAcceptButton.isEnabled = false + CoroutineScope(Dispatchers.Main).launch { + termsViewModel.acceptCurrentTerms() + root.findNavController().navigate(R.id.action_go_to_vpn_screen) + } + } + + onBackPress { emulateHomePress() } + + return root + } +} diff --git a/android/app/src/main/java/network/mysterium/ui/TermsViewModel.kt b/android/app/src/main/java/network/mysterium/ui/TermsViewModel.kt new file mode 100644 index 000000000..7e0de5a5e --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/TermsViewModel.kt @@ -0,0 +1,182 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import androidx.lifecycle.ViewModel +import network.mysterium.db.AppDatabase +import network.mysterium.db.Terms + +class TermsViewModel(private val appDatabase: AppDatabase): ViewModel() { + private val termsVersion = "1" + + suspend fun acceptCurrentTerms() { + appDatabase.termsDao().delete() + appDatabase.termsDao().insert(Terms(termsVersion)) + } + + suspend fun checkTermsAccepted(): Boolean { + val terms = appDatabase.termsDao().get() ?: return false + return terms.version == termsVersion + } + + val termsText = """ +

Terms & Conditions of MysteriumVPN for Users

+ +The following Terms and Conditions (“T&C”) govern the use of the Mysterium open source software platform (“Mysterium Platform”) and MysteriumVPN Software (the “Software”). Prior to any use of the Mysterium Platform, the User confirms to understand and expressly agrees to all of the Terms. + +MysteriumVPN is an Open Source software that enables you (the “Client”) to use the internet connection and resources of our contributors (the “Nodes”) for your private browsing using Mysterium Platform. + +The Mysterium Platform rests on open-source software. This is an experimental version of the Mysterium Platform, and there is a risk that the NetSys Inc. (“Network Operator”) or other third parties not directly affiliated with the Network Operator, might unintentionally introduce weaknesses or bugs into the core infrastructural elements of the Mysterium Platform or Software causing the system to lose cryptocurrency stored in one or more Client accounts or other accounts or lose sums of other valued tokens issued on the Mysterium Platform, or causing to suffer other losses, inconvenience, or damage to its users. The user expressly knows and agrees that the user is using the Mysterium Platform and MysteriumVPN software at the user’s sole risk. + +These terms and conditions constitute an agreement between you and Network Operator and governs your access and use of MysteriumVPN software and private browsing services provided by Mysterium Platform (the “Services”). + +The Network Operator or other creators of the Mysterium Platform are entitled to discontinue this project at any time without any further explanation or prior notification at this experimental stage. + + +PLEASE READ THESE T&C CAREFULLY IF YOU WISH TO USE THE SERVICES. IF YOU DO NOT AGREE TO BE BOUND BY THESE T&C, PLEASE DO NOT USE THE SERVICES. + +

Your Use of the Services

+ +The Services are provided to you free of charge. You may not use the Services if you are under the age of 18 or if you are not the owner of the device on which you install the Software or otherwise use the Services. + +By using the services of Network Operator, you understand and agree that the Services are internet security and private browsing services, and this mechanism is not a service to be used for criminal and/or other illegal acts. By using the Services, you accept not to violate any law of any jurisdiction that you are originating from or residing at. It is your responsibility to know and comprehend any and all relevant laws related to any jurisdiction or venue that concerns you and your actions. + + +

Client’s Account

+ +Before starting to use the Services, you must download Mysterium software and create an account entering your credentials and private/public keys. You hereby agree to provide true, accurate, current and complete information as may be prompted by any registration forms regarding your registration or use of Services. + +By creating an account and starting to use the Services you agree and accept unconditionally that: + +you cannot provide your credentials and private/public keys to others; +you will not share your credentials and private/public keys publicly. + +

Prohibited Conduct

+ +You may not use the Services in any manner that could damage, disable, overburden, or impair the servers and other resources of Network Operator and the Nodes, or interfere with any third party’s use of the Services. You may not attempt to gain unauthorized access to any aspect of the Services or to information for which you have not been granted access. + +By using the Services, you commit not to carry out any of these criminal and other illegal actions in or through the Services using our resources and/or resources of the Nodes: + +extortion, blackmail, kidnapping, rape, murder, sale/purchase of stolen credit cards, sale/purchase of stolen sale/purchase, sale/purchase of illegal sale/purchase, performance of identity theft; +use of stolen credit cards, credit card fraud, wire fraud, +hacking, pharming, phishing, or spamming of any form, web scraping through our Service in any form or scale; +exploitation of or contribution to children exploitation photographically, digitally or in any other way; +port scanning, sending opt-in email, scanning for open relays or open proxies, sending unsolicited e-mail or any version or type of email sent in vast quantities even if the email is lastly sent off through another server; +assaulting in any way or form any other network or computer while using the Service; +any other activities that are against the law of the country you originate from or reside in, and/or any other activities that are not compatible with the principles of democracy, freedom of speech, freedom of expression, and human rights. + +

Suspension or Termination of the Account

+ +Network Operator does not have an obligation to monitor the activities on the Mysterium Platform, and he does not carry out such monitoring actively. + +In the case Network Operator notices accidently or is notified of any suspicious activities using your Account that may result in violation of these T&C (see section “Prohibited Conduct” for more detail), in order to maintain the integrity of the Mysterium Platform Network Operator reserves the right (but not an obligation) to carry out any actions he considers necessary in the particular situation, including, but not limited to, the suspension and/or the termination of your Account immediately without any previous notification. + +Network Operator can terminate any Client account for the violation of these T&C immediately without notice or suspend the account until further clarification, investigation, or communication with the Client. + +

Limitation of Liability

+ +NEITHER THE NETWORK OPERATOR, NOR THE NODES WILL BE LIABLE IN ANY WAY OR FORM FOR ACTIONS DONE BY THE CLIENTS AND/OR ANY ACTIONS DONE USING CLIENT’S ACCOUNT INCLUDING CRIMINAL LIABILITY AND CIVIL LIABILITY FOR ANY HARM. SOLELY THE CLIENT IS LIABLE FOR ALL HIS ACTIONS USING THE SERVICES, THE SOFTWARE AND THE RESOURCES OF NETWORK OPERATOR AND OF THE RESOURCES THE NODES. + +NETWORK OPERATOR, ITS OWNERS, EMPLOYEES, AGENTS AND OTHERS THAT ARE INVOLVED WITH THE NETWORK OPERATOR ARE NOT IN ANY WAY OR FORM LIABLE FOR ANY HARM OF ANY SORT EXECUTED OR NOT EXECUTED, RESULTING FROM OR ARISING THROUGH OR FROM THE USE OF ANY ACCOUNT REGISTERED USING MYSTERIUM. + +IN NO EVENT WILL WE OR ANY OTHER PARTY WHO HAS BEEN INVOLVED IN THE CREATION, PRODUCTION, DISTRIBUTION, PROMOTION, OR MARKETING OF SOFTWARE AND/OR THE SERVICES BE LIABLE TO YOU OR ANY OTHER PARTY FOR ANY SPECIAL, INDIRECT, INCIDENTAL, RELIANCE, EXEMPLARY, OR CONSEQUENTIAL DAMAGES, INCLUDING, WITHOUT LIMITATION, LOSS OF DATA OR PROFITS, OR FOR INABILITY TO USE THE SERVICE, EVEN IF WE OR SUCH OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. SUCH LIMITATION SHALL APPLY NOTWITHSTANDING ANY FAILURE OF ESSENTIAL PURPOSE OF ANY LIMITED REMEDY AND TO THE FULLEST EXTENT PERMITTED BY LAW. IN NO EVENT, SHALL OUR AGGREGATE LIABILITY TO YOU AND ANY OTHER PARTY, WHETHER DIRECT OR INDIRECT, EXCEED ONE HUNDRED DOLLARS (${'$'}100.00) FOR ANY AND ALL CLAIMS, DAMAGES, AND OTHER THEORY OF LIABILITY. + +

No-logging and Traceability of Your Activities

+ +Network Operator guarantees a strict no-logs policy of the Services, meaning that your activities using the Services are not actively monitored, recorded, logged, stored or passed to any third party by the Network Operator. + +Network Operator does not store connection time stamps, used bandwidth, traffic logs nor IP addresses. Network Operator does not provide any information about their Clients and the fact of the use of Services by a particular Client to any third party. + +In the case Client’s use of Services constitute a breach of these T&C or respective legal acts and this causes legal risks for the Nodes whose resources Client was using, Network Operator might be obliged to provide the information related to such particular use of Services by the officially binding order of law enforcement authorities or other governmental agencies. As Network Operator does not trace the activities of their Clients, Network Operator would not be technically able to provide the authorities with more detailed information about their particular Clients activities. However, as Network Operator has no intention to contribute to any illegal and/or unauthorized activities, and/or to involve the Nodes in any of such, the Network Operator can cooperate with the authorities on his own account. + +Using the Services you agree and accept that Network Operator is free to take the decisions to provide any help and assistance including legal advice that Network Operator might find necessary or suitable to the Nodes who might need it in relation with a possible or an ongoing official investigation related to the use of their resources by the Clients. + +By using the Services you also agree and understand that your activities using the Services in some cases can be traceable by the Nodes and their internet service providers. + +In order to maintain the Services and to ensure the proper functioning of Mysterium, the Network Operator does collect limited and anonymized personal user information and site performance data (see section “Privacy Policy” for more details). + +

Data Privacy

+ +Network Operator is committed to your privacy and does not collect, log, organize, structure, store, use, disseminate or make otherwise available any personal data of the Clients, e.g. Client’s identity, IP address, browsing history, traffic destination, data content, DNS queries from the Clients connected to Mysterium Network, or any other personal information which could be considered personal data (i.e. any information relating to an identified or identifiable natural person). +Network Operator will collect anonymous statistics to improve the performance of Mysterium Network. This information is fully anonymized and cannot be tied back to any individual users. + +Limited License + +During your use of Services, we hereby grant you a personal, non-exclusive, non-transferable, non-assignable, non-sub-licensable, revocable and limited license to access and use the Services and to install a copy of the Software on your personal device. + +You fully understand and accept that the Software is licensed to you, not sold. + +Permission is granted to anyone to use this Software for any purpose, including commercial applications, and to alter it freely (without the right of redistribution), subject to the following restrictions: + +the origin of this Software must not be misrepresented; you must not claim that you wrote the original Software. If you use this Software in a product, an acknowledgment in the product documentation would be appreciated but is not mandatory; +altered versions must be plainly marked as such, and must not be misrepresented as being the original Software or any other creation of Network Operator, or the product created under any commission of Mysterium Platform; +this notice may not be removed or altered from any derivative works based on the Software. + +In any case, Network Operator does not bear any liability for any actions based on or related to the derivative works based on the Software. The users of such derivative works use them on their own account. + +Except as expressly indicated in these T&C or within any of the Services you may not: (i) sell, lend, rent, assign, export, sublicense or otherwise transfer the Software or the Services; (ii) alter, delete or conceal any copyright, trademark or other notices in connection with the Services or the Software; (iii) interfere with or impair the use of others of the Services or with any network connected to the Services; (iv) use the Services or the Software by yourself or in conjunction with any other products to infringe upon any third party's rights, including without limitation third party's intellectual property rights, to invade third party’s privacy in any way, or to track, store, transmit or record personal information about any other user of the Services or the Software; (v) otherwise violate applicable laws including without limitation copyright and trademark laws and applicable communications regulations and statutes. + +Any such forbidden uses shall immediately and automatically terminate your license to use the Software and the Services, without derogating from any other remedies available to us at law or in equity. + +Intellectual Property Rights + +The Software, including any versions, revisions, corrections, modifications, enhancements and/or upgrades thereto, accompanying materials, services and any copies you are permitted to make under these T&C are owned by us or our licensors and are protected under intellectual property laws, including copyright laws and treaties. You agree, accept and acknowledge that all right, title, and interest in and to the Software and associated intellectual property rights (including, without limitation, any patents (registered or pending), copyrights, trade secrets, designs or trademarks), evidenced by or embodied in or attached or connected or related to the Software are and shall remain owned solely by us or our licensors. These T&C do not convey to you any interest in or to the Software or any of the Services, but only a limited, revocable right of use in accordance with the terms of these T&C. Nothing in these T&C constitutes a waiver of our intellectual property rights under any law. + +Network Operator and Mysterium logos and trademarks owned by us or our licensors, and no right, license, or interest in any such trademarks is granted hereunder. We respect the intellectual property of others, and we ask you to do the same. It is important (and a condition of these T&C) that you comply with all copyright laws and other provisions in connection with any content agreement to which you may be a party through the Services. + +

Warranty Disclaimers

+ +THE SOFTWARE AND THE SERVICES ARE PROVIDED ON AN "AS IS" AND "AS AVAILABLE" BASIS, +WITHOUT ANY WARRANTY OF ANY KIND (INCLUDING SUPPORT OR OTHER SERVICES BY US OR OUR LICENSORES). YOU AGREE THAT YOUR USE OF THE SERVICES AND SOFTWARE SHALL BE AT YOUR SOLE RISK AND RESPONSIBILITY. TO THE FULLEST EXTENT PERMITTED BY APPLICABLE LAW, WE, OUR LICENSORS, OFFICERS, DIRECTORS, EMPLOYEES, AND AGENTS DISCLAIM ALL WARRANTIES, EXPRESS OR IMPLIED INCLUDING, WITHOUT LIMITATION, ANY IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, AND NON INFRINGEMENT. WE DO NOT WARRANT THAT THE SOFTWARE OR SERVICES: (A) WILL BE ERROR OR DEFECT FREE OR OTHERWISE FREE FROM ANY INTERRUPTIONS OR OTHER FAILURES; (B) WILL MEET YOUR REQUIREMENTS; OR (C) THAT ANY ERROR WILL BE IMMEDIATELY FIXED; WE MAKE NO WARRANTIES OR REPRESENTATIONS ABOUT THE ACCURACY OR COMPLETENESS OF THE CONTENT (INCLUDING ANY USER CONTENT) OR TO ANY THIRD PARTY SITES OR APPLICATIONS OR CONTENT OR ANY PORTION OR COMPONENT OF EITHER AND ASSUME NO LIABILITY OR RESPONSIBILITY AND DISCLAIM ALL WARRANTIES FOR ANY (I) PROBLEMS OR AVAILABILITY OF INTERNET CONNECTIONS (II) ERRORS, MISTAKES, OR INACCURACIES IN SOFTWARE OR SERVICES, (III) PROPERTY DAMAGE, OF ANY NATURE WHATSOEVER, RESULTING FROM YOUR ACCESS TO AND USE OF THE SERVICES OR TO ANY THIRD PARTY SITE, (IV) ANY UNAUTHORIZED ACCESS TO YOUR DEVICE OR USE OF OUR SECURE SERVERS OR ANY AND ALL PERSONAL INFORMATION OR FINANCIAL INFORMATION STORED THEREIN, (V) ANY INTERRUPTION OR CESSATION OF TRANSMISSION REGARDING THE SERVICES, (VI) ANY BUGS, VIRUSES, TROJAN HORSES, OR OTHER MALICIOUS CODE WHICH MAY BE TRANSMITTED TO OR THROUGH THE SERVICES BY ANY THIRD PARTY, (VII) ANY LOSS OR DAMAGE OF ANY KIND INCURRED AS A RESULT OF THE USE OF ANY CONTENT POSTED, EMAILED, TRANSMITTED OR OTHERWISE MADE AVAILABLE VIA THE SERVICES. + +WE DO NOT WARRANT, ENDORSE, GUARANTEE, OR ASSUME RESPONSIBILITY FOR ANY PRODUCT OR SERVICES ADVERTISED OR OFFERED BY A THIRD PARTY THROUGH THE SERVICES OR ANY HYPERLINKED WEBSITE OR FEATURED IN ANY BANNER OR OTHER ADVERTISING, AND WE WILL NOT BE A PARTY TO ANY TRANSACTION OR OTHER ENGAGEMENT WITH SUCH ADVERTISING OR IN ANY WAY BE RESPONSIBLE FOR MONITORING ANY TRANSACTION BETWEEN YOU AND THIRD-PARTY PROVIDERS OF PRODUCTS OR SERVICES. YOU ASSUME ALL RISK AS TO THE QUALITY, FUNCTION, AND PERFORMANCE OF THE SERVICES, AND TO ALL TRANSACTIONS YOU UNDERTAKE ON THROUGH THE SERVICES. + +

Indemnification

+ +BY USING THE SERVICES YOU FULLY UNDERSTAND AND AGREE THAT IS AN EXPERIMENTAL VERSION OF THE MYSTERIUM PLATFORM, AND THERE IS A RISK THAT THE NETWORK OPERATOR OR OTHER THIRD PARTIES NOT DIRECTLY AFFILIATED WITH THE NETWORK OPERATOR, MAY INTRODUCE WEAKNESSES OR BUGS INTO THE CORE INFRASTRUCTURAL ELEMENTS OF THE MYSTERIUM PLATFORM CAUSING THE SYSTEM TO LOSE CRYPTOCURRENCY STORED IN ONE OR MORE CLIENT ACCOUNTS OR OTHER ACCOUNTS OR LOSE SUMS OF OTHER VALUED TOKENS ISSUED ON THE MYSTERIUM PLATFORM, OR CAUSING TO SUFFER OTHER LOSSES, INCONVENIENCE, OR DAMAGE TO ITS USERS. + +You hereby agree to indemnify, defend and hold us, our subsidiaries, parent corporation and affiliates, partners, sponsors and all of their respective officers, directors, owners, employees, agents, attorneys, licensors, representatives, licensees, and suppliers (collectively, "Parties"), harmless from and against any and all liabilities, losses, expenses, damages, and costs (including reasonable attorneys' fees), incurred by any of the Parties in connection with any claim arising out of your use of the Software or Services, any use or alleged use of your account, username, or your password by any person, whether or not authorized by you, your violation or breach of these T&C or your violation of the rights of any other person or entity. + +

Term and Termination

+ +These T&C become effective upon the installation of the Software until terminated by either you or us (the "Term"). You may terminate your relationship with us at any time by completely uninstalling the Software. In the case you fail to comply with these T&C or any other agreement you have concluded with us, this will terminate your Software license and this agreement. Upon termination of this agreement the Software license granted to you shall automatically expire and you shall discontinue all further use of the Software and Services. + +We have the right to take any of the following actions in our sole discretion at any time without any prior notice to you: (i) restrict, deactivate, suspend, or terminate your access to the Services, including deleting your accounts and all related information and files contained in your account; (ii) refuse, move, or remove any material that is available on or through the Services; (iii) establish additional general practices and limits concerning use of the Services. + +We may take any of the above actions for any reason, as determined by us in our sole discretion, including, but not be limited to, (a) your breach or violation of these T&C, (b) requests by law enforcement authority or other governmental agency, (c) a request by you, (d) discontinuance or material modification to the Services (or any part thereof), and (e) unexpected technical or security issues or problems. + +You agree that we will not be liable to you or any third party for taking any of these actions. + +

Miscellaneous

+ +The Software is intended for use only in compliance with applicable laws and you undertake to use it in accordance with all such applicable laws. Without derogating from the foregoing and from any other terms herein, you agree to comply with all applicable export laws and restrictions and regulations and agree that you will not export, or allow the export or re-export of the Software in violation of any such restrictions, laws or regulations. +By logging in to Mysterium and using the Services, you agree to the T&C, including all other policies incorporated by reference. + +These T&C and the relationship between you and Network Operator shall be governed by and construed in accordance with law of Panama. You agree that any legal action arising out of or relating to these T&C, Software or your use of, or inability to use, the Services shall be filed exclusively in the competent courts of Panama. + +You agree that these T&C and our rights hereunder may be assigned, in whole or in part, by us or our affiliate to any third party, at our sole discretion, including an assignment in connection with a merger, acquisition, reorganization or sale of substantially all of our assets, or otherwise, in whole or in part. You may not delegate, sublicense or assign your rights under these T&C. + +That is the whole agreement. Network Operator may rewrite the T&C from time to time. The T&C become binding from the time it is updated on our website. It is your responsibility to check for the new provisions of T&C periodically. + +

Contact us:

+ +For any questions you may contact us: team@netsys.technology + +Last update: 29-03-2018 + """.trimIndent() +} \ No newline at end of file diff --git a/android/app/src/main/java/network/mysterium/ui/Toast.kt b/android/app/src/main/java/network/mysterium/ui/Toast.kt new file mode 100644 index 000000000..515f54ad9 --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/Toast.kt @@ -0,0 +1,26 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import android.content.Context +import android.widget.Toast + +fun showMessage(ctx: Context, message: String, duration: Int = Toast.LENGTH_SHORT) { + val t = Toast.makeText(ctx, message, duration) + t.show() +} diff --git a/android/app/src/main/java/network/mysterium/ui/UnitFormatter.kt b/android/app/src/main/java/network/mysterium/ui/UnitFormatter.kt new file mode 100644 index 000000000..d6e1d69cb --- /dev/null +++ b/android/app/src/main/java/network/mysterium/ui/UnitFormatter.kt @@ -0,0 +1,62 @@ +/* + * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package network.mysterium.ui + +import kotlin.math.floor +import kotlin.math.roundToInt + +object UnitFormatter { + val KB = 1024 + val MB = 1024 * KB + val GB = 1024 * MB + + fun bytesDisplay(bytes: Double): String { + return when { + bytes < KB -> "$bytes B" + bytes < MB -> "%.2f KB".format(bytes / KB) + bytes < GB -> "%.2f MB".format(bytes / MB) + else -> "%.2f GB".format(bytes / GB) + } + } + + fun timeDisplay(seconds: Double): String { + if (seconds < 0) { + return "00:00:00" + } + + val h = floor(seconds / 3600).roundToInt() + val hh = when { + h > 9 -> h.toString() + else -> "0$h" + } + + val m = floor((seconds % 3600) / 60).roundToInt() + val mm = when { + m > 9 -> m.toString() + else -> "0$m" + } + + val s = floor(seconds % 60).roundToInt() + val ss = when { + s > 9 -> s.toString() + else -> "0$s" + } + + return "${hh}:${mm}:${ss}" + } +} diff --git a/android/app/src/main/java/network/mysterium/vpn/MainActivity.kt b/android/app/src/main/java/network/mysterium/vpn/MainActivity.kt deleted file mode 100644 index dc8e73f11..000000000 --- a/android/app/src/main/java/network/mysterium/vpn/MainActivity.kt +++ /dev/null @@ -1,159 +0,0 @@ -/* - * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package network.mysterium.vpn - -import android.app.Activity -import android.content.ComponentName -import android.content.Context -import android.content.Intent -import android.content.ServiceConnection -import android.net.VpnService -import android.os.Bundle -import android.os.IBinder -import android.util.Log -import android.widget.Toast -import com.facebook.react.ReactActivity -import network.mysterium.service.core.MysteriumAndroidCoreService -import network.mysterium.service.core.MysteriumCoreService -import network.mysterium.vpn.connection.ConnectionChecker - -class MainActivity : ReactActivity() { - /** - * Returns the name of the main component registered from JavaScript. - * This is used to schedule rendering of the component. - */ - override fun getMainComponentName(): String? { - return "MysteriumVPN" - } - - private var mysteriumService: MysteriumCoreService? - get() = _mysteriumService - set(value) { - _mysteriumService = value - startIfReady() - } - private var _mysteriumService: MysteriumCoreService? = null - - private var vpnServiceGranted: Boolean - get() = _vpnServiceGranted - set(value) { - _vpnServiceGranted = value - startIfReady() - } - private var _vpnServiceGranted: Boolean = false - - private val serviceConnection = object : ServiceConnection { - override fun onServiceDisconnected(name: ComponentName?) { - Log.i(TAG, "Service disconnected") - mysteriumService = null - } - - override fun onServiceConnected(name: ComponentName?, service: IBinder?) { - Log.i(TAG, "Service connected") - mysteriumService = service as MysteriumCoreService - } - } - - private lateinit var connectionChecker: ConnectionChecker - - override fun onCreate(savedInstanceState: Bundle?) { - super.onCreate(savedInstanceState) - - bindMysteriumService() - ensureVpnServicePermission() - - connectionChecker = ConnectionChecker(applicationContext, CONNECTION_CHECKER_INTERVAL) - } - - override fun onResume() { - super.onResume() - connectionChecker.stop() - } - - override fun onPause() { - super.onPause() - connectionChecker.start() - } - - override fun onDestroy() { - unbindMysteriumService() - connectionChecker.stop() - super.onDestroy() - } - - override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) { - when (requestCode) { - VPN_SERVICE_REQUEST -> { - if (resultCode != Activity.RESULT_OK) { - Log.w(TAG, "User forbidden VPN service") - showMessage("VPN connection has to be granted for MysteriumVPN to work.") - finish() - return - } - Log.i(TAG, "User allowed VPN service") - vpnServiceGranted = true - } - } - } - - private fun bindMysteriumService() { - Log.i(TAG, "Binding service") - Intent(this, MysteriumAndroidCoreService::class.java).also { intent -> - bindService(intent, serviceConnection, Context.BIND_AUTO_CREATE) - } - } - - private fun unbindMysteriumService() { - Log.i(TAG, "Unbinding service") - unbindService(serviceConnection) - } - - private fun ensureVpnServicePermission() { - val intent: Intent? = VpnService.prepare(this) - if (intent == null) { - vpnServiceGranted = true - return - } - startActivityForResult(intent, MainActivity.VPN_SERVICE_REQUEST) - } - - private fun startIfReady() { - val service = mysteriumService ?: return - if (!vpnServiceGranted) { - return - } - - Log.i(TAG, "Starting node") - try { - service.StartTequila() - Log.i(TAG, "Node started successfully") - } catch (tr: Throwable) { - Log.e(TAG, "Starting service failed", tr) - } - } - - private fun showMessage(message: String) { - Toast.makeText(this, message, Toast.LENGTH_LONG).show() - } - - companion object { - private const val VPN_SERVICE_REQUEST = 1 - private const val TAG = "MainActivity" - private const val CONNECTION_CHECKER_INTERVAL: Long = 3000 - } -} diff --git a/android/app/src/main/java/network/mysterium/vpn/MainApplication.java b/android/app/src/main/java/network/mysterium/vpn/MainApplication.java deleted file mode 100644 index 6f6a43ec2..000000000 --- a/android/app/src/main/java/network/mysterium/vpn/MainApplication.java +++ /dev/null @@ -1,118 +0,0 @@ -/* - * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package network.mysterium.vpn; - -import android.app.Application; -import android.util.Log; -import android.content.Context; -import com.facebook.react.PackageList; - -import com.facebook.react.ReactApplication; -import com.dieam.reactnativepushnotification.ReactNativePushNotificationPackage; -import com.oblador.vectoricons.VectorIconsPackage; -import com.facebook.react.ReactNativeHost; -import com.facebook.react.ReactPackage; -import com.facebook.soloader.SoLoader; - -import cat.ereza.logcatreporter.LogcatReporter; - -import com.crashlytics.android.core.CrashlyticsCore; -import com.crashlytics.android.Crashlytics; -import io.fabric.sdk.android.Fabric; -import network.mysterium.logging.BugReporterPackage; - -import java.lang.reflect.InvocationTargetException; -import java.util.List; - -public class MainApplication extends Application implements ReactApplication { - private static final String TAG = "MainApplication"; - - private final ReactNativeHost mReactNativeHost = new ReactNativeHost(this) { - @Override - public boolean getUseDeveloperSupport() { - return BuildConfig.DEBUG; - } - - @Override - protected List getPackages() { - @SuppressWarnings("UnnecessaryLocalVariable") - List packages = new PackageList(this).getPackages(); - packages.add(new BugReporterPackage()); - return packages; - } - - @Override - protected String getJSMainModuleName() { - return "index"; - } - }; - - @Override - public ReactNativeHost getReactNativeHost() { - return mReactNativeHost; - } - - @Override - public void onCreate() { - setupLogging(); - super.onCreate(); - SoLoader.init(this, /* native exopackage */ false); - // initializeFlipper(this); // Remove this line if you don't want Flipper enabled - Log.i(TAG, "Application started"); - } - - /** - * Loads Flipper in React Native templates. - * - * @param context - */ - private static void initializeFlipper(Context context) { - if (BuildConfig.DEBUG) { - try { - /* - We use reflection here to pick up the class that initializes Flipper, - since Flipper library is not available in release mode - */ - Class aClass = Class.forName("com.facebook.flipper.ReactNativeFlipper"); - aClass.getMethod("initializeFlipper", Context.class).invoke(null, context); - } catch (ClassNotFoundException e) { - e.printStackTrace(); - } catch (NoSuchMethodException e) { - e.printStackTrace(); - } catch (IllegalAccessException e) { - e.printStackTrace(); - } catch (InvocationTargetException e) { - e.printStackTrace(); - } - } - } - - private void setupLogging() { - // https://docs.fabric.io/android/crashlytics/build-tools.html?highlight=crashlyticscore - // Set up Crashlytics, disabled for debug builds - Crashlytics crashlyticsKit = new Crashlytics.Builder() - .core(new CrashlyticsCore.Builder().disabled(BuildConfig.DEBUG).build()) - .build(); - - // Initialize Fabric with the debug-disabled crashlytics. - Fabric.with(this, crashlyticsKit); - - LogcatReporter.install(); - Crashlytics.setInt("android_sdk_int", android.os.Build.VERSION.SDK_INT); - } -} diff --git a/android/app/src/main/java/network/mysterium/vpn/connection/ConnectionChecker.kt b/android/app/src/main/java/network/mysterium/vpn/connection/ConnectionChecker.kt deleted file mode 100644 index 72b7c6402..000000000 --- a/android/app/src/main/java/network/mysterium/vpn/connection/ConnectionChecker.kt +++ /dev/null @@ -1,79 +0,0 @@ -/* - * Copyright (C) 2019 The "mysteriumnetwork/mysterium-vpn-mobile" Authors. - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package network.mysterium.vpn.connection - -import android.content.Context -import android.content.Intent -import android.os.Bundle -import android.os.Handler -import android.util.Log -import java.lang.IllegalStateException - -/** - * Starts ConnectionCheckerService periodically at a given interval. - */ -class ConnectionChecker(private val context: Context, private val interval: Long) { - private var running: Boolean = false - - fun start() { - if (running) { - return - } - - Log.d(TAG, "Starting ConnectionChecker") - running = true - this.loopService(ignoreLastStatus = true) - } - - fun stop() { - if (!running) { - return - } - - Log.d(TAG, "Stopping ConnectionChecker") - running = false - } - - private fun loopService(ignoreLastStatus: Boolean) { - if (!running) { - return - } - startService(ignoreLastStatus) - - Handler().postDelayed({ this.loopService(ignoreLastStatus = false) }, interval) - } - - private fun startService(ignoreLastStatus: Boolean) { - Log.d(TAG, "Starting ConnectionCheckerService") - val service = Intent(context, ConnectionCheckerService::class.java) - val bundle = Bundle() - bundle.putBoolean("ignoreLastStatus", ignoreLastStatus) - service.putExtras(bundle) - try { - context.startService(service) - } catch (e: IllegalStateException) { - // We stop checker, because app is in a state where the service can not be started. - // This is usually because app was in the background for too long. - stop() - } - } - - companion object { - private const val TAG = "ConnectionChecker" - } -} diff --git a/android/app/src/main/res/drawable/background_logo.png b/android/app/src/main/res/drawable/background_logo.png new file mode 100644 index 0000000000000000000000000000000000000000..3a4023138ac08095fefe40f978ffe4086553cea5 GIT binary patch literal 143919 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�(crh(u5=R}Meps55?cZqiSiluso7*QtMb?01ir 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 + + + + + + + + + diff --git a/android/app/src/main/res/drawable/bordered_input_bg.xml b/android/app/src/main/res/drawable/bordered_input_bg.xml new file mode 100644 index 000000000..90e886ed4 --- /dev/null +++ b/android/app/src/main/res/drawable/bordered_input_bg.xml @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file diff --git a/android/app/src/main/res/drawable/ic_arrow_back_24dp.xml b/android/app/src/main/res/drawable/ic_arrow_back_24dp.xml new file mode 100644 index 000000000..6343cdbb1 --- /dev/null +++ b/android/app/src/main/res/drawable/ic_arrow_back_24dp.xml @@ -0,0 +1,5 @@ + + + diff --git a/android/app/src/main/res/drawable/ic_arrow_downward_black_24dp.xml b/android/app/src/main/res/drawable/ic_arrow_downward_black_24dp.xml new file mode 100644 index 000000000..c31b7be6a --- /dev/null +++ b/android/app/src/main/res/drawable/ic_arrow_downward_black_24dp.xml @@ -0,0 +1,5 @@ + + + diff --git a/android/app/src/main/res/drawable/ic_help_outline_34dp.xml b/android/app/src/main/res/drawable/ic_help_outline_34dp.xml new file mode 100644 index 000000000..25614790e --- /dev/null +++ b/android/app/src/main/res/drawable/ic_help_outline_34dp.xml @@ -0,0 +1,5 @@ + + + diff --git a/android/app/src/main/res/drawable/ic_help_outline_black_24dp.xml b/android/app/src/main/res/drawable/ic_help_outline_black_24dp.xml new file mode 100644 index 000000000..850ca0eb7 --- /dev/null +++ b/android/app/src/main/res/drawable/ic_help_outline_black_24dp.xml @@ -0,0 +1,5 @@ + + + diff --git a/android/app/src/main/res/drawable/ic_public_black_24dp.xml b/android/app/src/main/res/drawable/ic_public_black_24dp.xml new file mode 100644 index 000000000..56a6e9b1f --- /dev/null +++ b/android/app/src/main/res/drawable/ic_public_black_24dp.xml @@ -0,0 +1,5 @@ + + + diff --git a/android/app/src/main/res/drawable/ic_search_white_34dp.xml b/android/app/src/main/res/drawable/ic_search_white_34dp.xml new file mode 100644 index 000000000..116da9b07 --- /dev/null +++ b/android/app/src/main/res/drawable/ic_search_white_34dp.xml @@ -0,0 +1,5 @@ + + + diff --git a/android/app/src/main/res/drawable/ic_star_black_24dp.xml b/android/app/src/main/res/drawable/ic_star_black_24dp.xml new file mode 100644 index 000000000..3c6fabaef --- /dev/null +++ b/android/app/src/main/res/drawable/ic_star_black_24dp.xml @@ -0,0 +1,5 @@ + + + diff --git a/android/app/src/main/res/drawable/ic_star_border_black_24dp.xml b/android/app/src/main/res/drawable/ic_star_border_black_24dp.xml new file mode 100644 index 000000000..e7d547918 --- /dev/null +++ b/android/app/src/main/res/drawable/ic_star_border_black_24dp.xml @@ -0,0 +1,5 @@ + + + diff --git a/android/app/src/main/res/drawable/notification_icon.png b/android/app/src/main/res/drawable/notification_icon.png new file mode 100644 index 0000000000000000000000000000000000000000..57f545d20af04a8042b9bd2e73d8a2a3d3c90a85 GIT binary patch literal 3209 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 vyGgZJ + + + + + + diff --git a/android/app/src/main/res/drawable/proposals_filter_bg.xml b/android/app/src/main/res/drawable/proposals_filter_bg.xml new file mode 100644 index 000000000..57db345cb --- /dev/null +++ b/android/app/src/main/res/drawable/proposals_filter_bg.xml @@ -0,0 +1,24 @@ + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/android/app/src/main/res/drawable/quality_high.png b/android/app/src/main/res/drawable/quality_high.png new file mode 100644 index 0000000000000000000000000000000000000000..d6656bd94e68be847930cb9e89a45da5b2f3b738 GIT binary patch literal 1636 zcmeAS@N?(olHy`uVBq!ia0vp^2_VeD1|%QND7OGojKx9jP7LeL$-D$|I14-?iy0XB zj({-ZRBb+K1_llpi<;HsXMd|v6mX?tKo&&O01A4mfXs^2oCudpV1R;s0df^q33QFl`8oMT z!3BxQsUU}ewIB(hs|H&WiOU)!2_%gv$yUXsNm;4MB}lOaj5BcBcS=mxM-sNt$Ew-} zm=dgf^HVa@DqRvwQtgZk3@vpH40R1mLJSS83=FM|O>Feh)F8P4B;#C^npl!w6q28x zV+Zy{E{LmOtY@wVOeE+kpqg#;LHP(NUqLc0SQMCR?YMwCVI_ba7tN< zq=ND7&5gOU9R*kuZx%%7IsDzh_4P4Z$c!~?Y7ZVI%KSN8z+cW?$Lv;9{d%$k6Q{}q zRN{k3qrCsmMPF>~?e6b?@y6)?=il$Wm)Yxk@7gS^ap2siFF}lcIlH}CZ$!?E(>lNrs)alh8d=Z?IsFg(O7c8YUgnlpi<;HsXMd|v6mX?tKo&&O01A4mfXs^2oCudpV1R;s0df^q33QFl`8oMT z!3BxQsUU}ewIB(hs|H&WiOU)!2_%gv$yUXsNm;4MB}lOaj5BcBcS=mxM-sNt$Ew-} zm=dgf^HVa@DqRvwQtgZk3@vpH40R1mLJSS83=FNzfZjz@gX98`jB`l~Jle#0+fCb8HPZ!6K z3dXm$H|9z^2)G2EED(Lw+!}Q(?ARQ$SLHz0+jo-?hW@Jie&wzA|KHnh-S#)$|4iE8c8DL#3WM|B zEHiSo)R{tNJXd8pV>VToQR~cSA;z@Xr#cx#pH+4;Z1#QX(!lk3k4wYNWkpI3t(9>~ z2XcZf1s5E$(HBhEwsHn&(&siBPmFTKotoe7MK-pP`taXUeJt@_L}kz|+;wWt~$(69BBU BBRT*8 literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/drawable/quality_medium.png b/android/app/src/main/res/drawable/quality_medium.png new file mode 100644 index 0000000000000000000000000000000000000000..e9a622ed098f9c6c8c9d51cfda248bd030eb8d6f GIT binary patch literal 1640 zcmeAS@N?(olHy`uVBq!ia0vp^2_VeD1|%QND7OGojKx9jP7LeL$-D$|I14-?iy0XB zj({-ZRBb+K1_llpi<;HsXMd|v6mX?tKo&&O01A4mfXs^2oCudpV1R;s0df^q33QFl`8oMT z!3BxQsUU}ewIB(hs|H&WiOU)!2_%gv$yUXsNm;4MB}lOaj5BcBcS=mxM-sNt$Ew-} zm=dgf^HVa@DqRvwQtgZk3@vpH40R1mLJSS83=FM|fNn=qgX98`jB`l~JJ}=+o0}GV9o-U3d z6^w6h8u}e^5MXd@{vpsP{7+wq{ri$;{*nh5R|uOwTvO&AuB#;PUeBpA!GQ^l@E811 zaf*NG{mR(uv(Kr0{%n&!v(9f?NxJ_6hU#gHr*j!3-QFG2V0-$aH>NcDg7niR3|z+gnhdu! z)L8>&RC+SraI@5GICRFF>wwuQPezfnxFCjYfqq;I3_ed{+R$4R#Lzl(x=2Irl*vpQ zXZ9>%$XPr`#3AvSDl5>aC*Q<3F0WY`^yUBbFY*bVhvzXd^qA^3sPa~GF&JFi!7$C_ zjV8l^h{sG3mo_YCU`X!ern{o;>WwTy9tx<$gO5!0p~YNNZnT%nfr@`mS3j3^P6lpi<;HsXMd|v6mX?tKo&&O01A4mfXs^2oCudpV1R;s0df^q33QFl`8oMT z!3BxQsUU}ewIB(hs|H&WiOU)!2_%gv$yUXsNm;4MB}lOaj5BcBcS=mxM-sNt$Ew-} zm=dgf^HVa@DqRvwQtgZk3@vpH40R1mLJSS93=FL-fDw(R2FV2=8Rw$Z#FG4?ko^1{ zJFqWuL0ko6J##&%USt(e%{Ka=e1w#*Aej~{3e2^3TtHi4C4e2*zb9%Tzyjr>r;B4q z1>@VB8*`Z)1Xu%a7kPh)XK=llm%tPj|F}W%&gHvZ=1PwqFZO5RRGHv_N*vhDQM2!~ z_|o@3YwYS@yfLb`k<*`97dx#aT|J;-?$eh`ndaOmo6faiv(4U+2I1T9f*8xDEuYR+ zkaRm%>%h5BUxFC@a&~*O-iVwRr*)t+w|pgo_36vrtQH&J>WL<#S5~cLP~UcUDbpJr ze|^yhN4D(_K^VHP+&?$|*Zcoj+wZrbP0l+XkK%={T# literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/drawable/round_icon_button.xml b/android/app/src/main/res/drawable/round_icon_button.xml new file mode 100644 index 000000000..8dd5eba26 --- /dev/null +++ b/android/app/src/main/res/drawable/round_icon_button.xml @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/android/app/src/main/res/drawable/selector.xml b/android/app/src/main/res/drawable/selector.xml new file mode 100644 index 000000000..3108ff1e4 --- /dev/null +++ b/android/app/src/main/res/drawable/selector.xml @@ -0,0 +1,5 @@ + + + + + diff --git a/android/app/src/main/res/drawable/service_openvpn.png b/android/app/src/main/res/drawable/service_openvpn.png new file mode 100644 index 0000000000000000000000000000000000000000..5ec1ec63c1ddaf16a6ef82c46e860028897d232c GIT binary patch literal 4717 zcmbVPc{G&&+kR%DA&e~zS;knBRMwDvH?n1|?6ND_Dn!N_#xkUd60&a*VTdx;C~IVk z>_iwPU%UBvU%xZQ>G!_pyyrdVbMDV|U)ObC_w&c|{P9E^>TA){a?k<*K#$f|Hv$0A zAp`+1$l>K)a^2+cf;y|{sQ^HE!pS{5s{c}bjI>mN(r(U00Du67x+WTj`fp;QWgWTxsY9cK>lQ38cL0>|6q$rSTnX~+GC zn4%u}DN{!hr5;WFXG`HJ;CRk)d}KIeDC80T=fB?#kK~s@hu^?~uaUkn&`-;GQF7U7 zVYagCp9eodkTXm)1JWZ>r^CEmh?ml~lZHZUYU^t1<-cX#_yE^Gx0#Pa(KVg;gXfr9 zdh?8Tugx@kKBHB`F^5Yv>NC-yaZu)QkJ}r@Yxo7fbrF^6{VT8Uuxd+2famnY(CfG9 zxNOz;p4eG$@Dy~8qnZOmH?sTQ%~@e!S==ueIuavUJ~1~p%;YVyO@N)~N$QWc9!%2f z99+FUn?>zA@M=}Hef7FhG45uLVWBhD4Oln}Zy#)KtwI``}4qS^3 zjnSx-FecVAd>%ujE6ne?xdfz0T;4 zcubv6TJ);DW31_e)XcB!in|I&ENI1|b$@$G%A=!iX>K)adW9todb^hBaj-7*^aN&u zXe`-jAROCsaL!#@-8H;}r61aO=&%mO7o3~3Tza(qmSzmb+>L*bO*W>hCG*hc)sRLKrZt)7 zN*^@h0>)&P=)K~0?OxXM{otHuAQM)n8SIA4PS@Dr8oAUpL;4Rk=NYmIZy%%!t;%F^ z)(Z%Dn2Ed2Wzq9z|JRAs2f#sGkxho8$j60PB?TqcS0z#=e=ql&ll(UAWmba)Ynd zV|<>P)-rn8mL;(JPj1R_;gym=k-eZ5MX-nHYn9ny+BZop!*Qj8k5@p zT+%Ep>VTnmyY2^zwkvoVq4cL`b3^=DnISIo`OKSvc~ z>4FA@Q;n)KAS#)c`-LFP{i7*x_*0|46a(vS1&u=f!sPI_GpGKHzkil^i!@G4JsveQ zq^&Y89Uv|-A(S!zd;L13zJlO8H~&3}PzW+J=W;c@!STXsrXr`SD6HSa`U?D!s{12F zi=9Z;=${hrRCYGZuicH%)L1m1qDM&0IRpb|%@<4iZrn5a=PYKi`k|>&N55dvKssJ> z)pXwu^TcWC4}UNuib!lu>LCkDKueMW;qoQhjMqD^Cj_tS1SEtJIs~%P?b_C2Eh|

Art?+~WMyLV)=IE~7?- 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 C9G{l} literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/drawable/service_wireguard.png b/android/app/src/main/res/drawable/service_wireguard.png new file mode 100644 index 0000000000000000000000000000000000000000..87c373f7d3c3edab548cfddb63c584ccffa6d724 GIT binary patch literal 19549 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 literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/drawable/spinner_input_arrow.png b/android/app/src/main/res/drawable/spinner_input_arrow.png new file mode 100644 index 0000000000000000000000000000000000000000..8e1949a7ea500d7751707e4244867ccda9baa0f9 GIT binary patch literal 1326 zcmZ`(U1-x#6wY)L*Ri3Z&`n`6`rycG!z+wdT-iov2=m_-0ytfJ?A^g$#|sy zOtY=cMi4}EUvFpt-FM-0=MHpjUnz;`R-X=p0|fE>X49J3fYyy_Z(JvcmIZv)-8t5> zA8{x-8iTR$Sx!>2w3t*z0BvMd#3l&d;Lu|hK#?@EnVilUKFVZp=o!NdMVb_N(MQF? z5i+P~fb60>X(uJvNRs5WWQrRI^;G0&<)cO+R5^wz6bf{~O)FZOVZB~2<8(1Dmjf{z z`dAK%h9jpRDsQsBPYCFeCaX|ZawOhY98vPnM^QM@#!;>l%BjsvIlW>F8DwyTVQD8* zO9l*i3mJ}VDN9NluGGAiF+C+E24p}M$!+*fJ`hTqlTI;SX5QbK(#QaWE5>s6(xWRGbI+Nnr)_uZgfv9JSL8ojw2KXUyK(nqDiFE;Es0 z>+qhprKaf8^{~75afie4i=3D#9(~<26!?_5EhXy5w++vEF8x{l?f=pq-}nCejp^y> zEB@6}FFdY?jSF=hMA0Bu=*e%x6Lw)S@v^tgo?3hWh>!5V@%cBSjc~OgIX3m#ADr~x fJM!wuwYh@?@vNHg46ccS + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/android/app/src/main/res/drawable/splash_logo.png b/android/app/src/main/res/drawable/splash_logo.png new file mode 100644 index 0000000000000000000000000000000000000000..a9fea89ab0f7d708341ac7f54cbad66581c4a3df GIT binary patch literal 22654 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 xtA5eh8Iu%*YO0N)RZ9hvN2#`sjQd{;{|`kR+^ literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/layout/activity_main.xml b/android/app/src/main/res/layout/activity_main.xml new file mode 100644 index 000000000..daa7bc824 --- /dev/null +++ b/android/app/src/main/res/layout/activity_main.xml @@ -0,0 +1,20 @@ + + + + + + diff --git a/android/app/src/main/res/layout/fragment_feedback.xml b/android/app/src/main/res/layout/fragment_feedback.xml new file mode 100644 index 000000000..b6339d945 --- /dev/null +++ b/android/app/src/main/res/layout/fragment_feedback.xml @@ -0,0 +1,109 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/android/app/src/main/res/layout/fragment_main_vpn.xml b/android/app/src/main/res/layout/fragment_main_vpn.xml new file mode 100644 index 000000000..4b3b6b59c --- /dev/null +++ b/android/app/src/main/res/layout/fragment_main_vpn.xml @@ -0,0 +1,325 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +