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 000000000..3a4023138 Binary files /dev/null and b/android/app/src/main/res/drawable/background_logo.png differ diff --git a/android/app/src/main/res/drawable/background_splash.xml b/android/app/src/main/res/drawable/background_splash.xml new file mode 100644 index 000000000..8c32c102c --- /dev/null +++ b/android/app/src/main/res/drawable/background_splash.xml @@ -0,0 +1,12 @@ + + + + + + + + + + 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 000000000..57f545d20 Binary files /dev/null and b/android/app/src/main/res/drawable/notification_icon.png differ diff --git a/android/app/src/main/res/drawable/proposal_picker_bg.xml b/android/app/src/main/res/drawable/proposal_picker_bg.xml new file mode 100644 index 000000000..ec57deb22 --- /dev/null +++ b/android/app/src/main/res/drawable/proposal_picker_bg.xml @@ -0,0 +1,7 @@ + + + + + + + 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 000000000..d6656bd94 Binary files /dev/null and b/android/app/src/main/res/drawable/quality_high.png differ diff --git a/android/app/src/main/res/drawable/quality_low.png b/android/app/src/main/res/drawable/quality_low.png new file mode 100644 index 000000000..73a91bdf4 Binary files /dev/null and b/android/app/src/main/res/drawable/quality_low.png differ 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 000000000..e9a622ed0 Binary files /dev/null and b/android/app/src/main/res/drawable/quality_medium.png differ diff --git a/android/app/src/main/res/drawable/quality_unknown.png b/android/app/src/main/res/drawable/quality_unknown.png new file mode 100644 index 000000000..fdff94167 Binary files /dev/null and b/android/app/src/main/res/drawable/quality_unknown.png differ 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 000000000..5ec1ec63c Binary files /dev/null and b/android/app/src/main/res/drawable/service_openvpn.png differ 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 000000000..87c373f7d Binary files /dev/null and b/android/app/src/main/res/drawable/service_wireguard.png differ 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 000000000..8e1949a7e Binary files /dev/null and b/android/app/src/main/res/drawable/spinner_input_arrow.png differ diff --git a/android/app/src/main/res/drawable/spinner_input_bg.xml b/android/app/src/main/res/drawable/spinner_input_bg.xml new file mode 100644 index 000000000..27a2a152c --- /dev/null +++ b/android/app/src/main/res/drawable/spinner_input_bg.xml @@ -0,0 +1,17 @@ + + + + + + + + + + + + + + + + + \ 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 000000000..a9fea89ab Binary files /dev/null and b/android/app/src/main/res/drawable/splash_logo.png differ 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 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +