diff --git a/Dockerfile b/Dockerfile deleted file mode 100644 index 9c973bf1..00000000 --- a/Dockerfile +++ /dev/null @@ -1,33 +0,0 @@ -FROM thyrlian/android-sdk - -## Dependencies to run as root: -ENV DEBIAN_FRONTEND=noninteractive -RUN dpkg --add-architecture i386 && \ - apt-get -y update && \ - apt-get install -y \ - curl ca-certificates software-properties-common gpg-agent wget \ - python3.7 python3.7-dev python3-pip python2.7 python2.7-dev python3.7-venv \ - python-pip zlib1g-dev m4 zlib1g:i386 libc6-dev-i386 gawk nodejs npm unzip openjdk-8-jdk \ - autoconf autogen automake libtool libffi-dev build-essential \ - ccache git libncurses5:i386 libstdc++6:i386 \ - libgtk2.0-0:i386 libpangox-1.0-0:i386 libpangoxft-1.0-0:i386 libidn11:i386 -RUN npm install -g npm@latest -RUN npm install -g yarn react-native-cli && \ - pip2 install --upgrade cython setuptools && \ - pip2 install git+https://github.com/lbryio/buildozer.git@master && \ - ln -s /src/scripts/build-docker.sh /usr/local/bin/build && \ - adduser lbry-android --gecos GECOS --shell /bin/bash --disabled-password --home /home/lbry-android && \ - mkdir /home/lbry-android/.npm-packages && \ - echo "prefix=/home/lbry-android/.npm-packages" > /home/lbry-android/.npmrc && \ - chown -R lbry-android:lbry-android /home/lbry-android && \ - mkdir /src && \ - chown lbry-android:lbry-android /src && \ - mkdir /dist && \ - chown lbry-android:lbry-android /dist - -## Further setup done by lbry-android user: -USER lbry-android - -COPY scripts/docker-build.sh /home/lbry-android/bin/build -COPY scripts/docker-setup.sh /home/lbry-android/bin/setup -CMD ["/home/lbry-android/bin/build"] diff --git a/app/.gitignore b/app/.gitignore new file mode 100644 index 00000000..796b96d1 --- /dev/null +++ b/app/.gitignore @@ -0,0 +1 @@ +/build diff --git a/app/BUCK b/app/BUCK deleted file mode 100644 index 3f7eb728..00000000 --- a/app/BUCK +++ /dev/null @@ -1,55 +0,0 @@ -# To learn about Buck see [Docs](https://buckbuild.com/). -# To run your application with Buck: -# - install Buck -# - `npm start` - to start the packager -# - `cd android` -# - `keytool -genkey -v -keystore keystores/debug.keystore -storepass android -alias androiddebugkey -keypass android -dname "CN=Android Debug,O=Android,C=US"` -# - `./gradlew :app:copyDownloadableDepsToLibs` - make all Gradle compile dependencies available to Buck -# - `buck install -r android/app` - compile, install and run application -# - -load(":build_defs.bzl", "create_aar_targets", "create_jar_targets") - -lib_deps = [] - -create_aar_targets(glob(["libs/*.aar"])) - -create_jar_targets(glob(["libs/*.jar"])) - -android_library( - name = "all-libs", - exported_deps = lib_deps, -) - -android_library( - name = "app-code", - srcs = glob([ - "src/main/java/**/*.java", - ]), - deps = [ - ":all-libs", - ":build_config", - ":res", - ], -) - -android_build_config( - name = "build_config", - package = "com.lbryandroid", -) - -android_resource( - name = "res", - package = "com.lbryandroid", - res = "src/main/res", -) - -android_binary( - name = "app", - keystore = "//android/keystores:debug", - manifest = "src/main/AndroidManifest.xml", - package_type = "debug", - deps = [ - ":app-code", - ], -) diff --git a/app/build.gradle b/app/build.gradle index 502a6362..8ad216cf 100644 --- a/app/build.gradle +++ b/app/build.gradle @@ -1,137 +1,8 @@ -apply plugin: "com.android.application" - -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", - * - * // https://facebook.github.io/react-native/docs/performance#enable-the-ram-format - * bundleCommand: "ram-bundle", - * - * // 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: [] - * ] - */ - -task buildReactNativeBundle(type:Exec) { - println("Building React Native bundle") - workingDir new File(rootProject.projectDir, '../') - commandLine './bundle-android.sh' -} -preBuild.dependsOn buildReactNativeBundle - -task printVersionName { - doLast { - println android.defaultConfig.versionName - } -} - -project.ext.react = [ - entryFile: "index.js", - enableHermes: false, // clean and rebuild if changing -] - -/** - * Set this to true to create two separate APKs instead of one: - * - An APK that only works on ARM devices - * - An APK that only works on x86 devices - * The advantage is the size of the APK is reduced by about 4MB. - * Upload all the APKs to the Play Store and people will download - * the correct one based on the CPU architecture of their device. - */ -def enableSeparateBuildPerCPUArchitecture = false - -/** - * Run Proguard to shrink the Java bytecode in release builds. - */ -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); +apply plugin: 'com.android.application' android { - compileSdkVersion rootProject.ext.compileSdkVersion + compileSdkVersion 29 + buildToolsVersion "29.0.1" flavorDimensions "default" compileOptions { @@ -141,18 +12,14 @@ android { defaultConfig { applicationId "io.lbry.browser" - minSdkVersion rootProject.ext.minSdkVersion - targetSdkVersion rootProject.ext.targetSdkVersion - versionCode 1402 - versionName "0.14.2" - missingDimensionStrategy 'react-native-camera', 'general' - multiDexEnabled true - } - dexOptions { - javaMaxHeapSize "2048M" - preDexLibraries false - jumboMode true + minSdkVersion 21 + targetSdkVersion 29 + versionCode 1500 + versionName "0.15.0" + + testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" } + productFlavors { __32bit { versionCode android.defaultConfig.versionCode * 10 + 1 @@ -167,66 +34,53 @@ android { } } } - signingConfigs { - debug { - storeFile file('debug.keystore') - storePassword 'android' - keyAlias 'androiddebugkey' - keyPassword 'android' - } - } + buildTypes { - debug { - signingConfig signingConfigs.debug - } release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' } } + } dependencies { - implementation project(':@react-native-community_async-storage') - implementation project(':react-native-camera') - implementation project(':react-native-exception-handler') - implementation project(':react-native-fast-image') - implementation project(':react-native-fs') - implementation project(':react-native-gesture-handler') - implementation project(':react-native-reanimated') - implementation project(':react-native-snackbar') - implementation project(':react-native-video') - implementation project(':react-native-webview') - implementation project(':rn-fetch-blob') + implementation fileTree(dir: 'libs', include: ['*.jar']) + + implementation 'androidx.appcompat:appcompat:1.1.0' implementation 'androidx.legacy:legacy-support-v4:1.0.0' - implementation 'androidx.media:media:1.0.0' - implementation 'androidx.appcompat:appcompat:1.0.0' - implementation 'com.facebook.react:react-native:0.61.5' - implementation 'com.facebook.fresco:fresco:1.9.0' - implementation 'com.facebook.fresco:animated-gif:1.9.0' - implementation 'com.squareup.picasso:picasso:2.71828' - implementation 'com.google.firebase:firebase-analytics:17.2.1' - implementation 'com.google.android.gms:play-services-base:17.1.0' - implementation 'androidx.exifinterface:exifinterface:1.0.0' - implementation 'com.facebook.fresco:animated-base-support:1.3.0' - implementation 'com.facebook.fresco:animated-gif:1.10.0' - implementation 'com.google.firebase:firebase-messaging:20.1.0' + implementation 'com.google.android.material:material:1.1.0' + implementation "androidx.cardview:cardview:1.0.0" + implementation 'androidx.constraintlayout:constraintlayout:1.1.3' + implementation 'androidx.navigation:navigation-fragment:2.2.2' + implementation 'androidx.navigation:navigation-ui:2.2.2' + implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0' + implementation 'androidx.localbroadcastmanager:localbroadcastmanager:1.0.0' + implementation 'androidx.preference:preference:1.1.1' + implementation 'com.github.bumptech.glide:glide:4.11.0' + implementation 'com.squareup.okhttp3:okhttp:4.4.1' + implementation 'com.google.firebase:firebase-analytics:17.4.0' + implementation 'com.google.android.gms:play-services-base:17.2.1' + implementation 'com.google.firebase:firebase-messaging:20.1.6' + + implementation 'com.google.code.gson:gson:2.8.6' + implementation 'com.google.android.exoplayer:exoplayer-core:2.11.4' + implementation 'com.google.android.exoplayer:exoplayer-dash:2.11.4' + implementation 'com.google.android.exoplayer:exoplayer-ui:2.11.4' + implementation 'com.google.android.exoplayer:extension-cast:2.11.4' + + implementation 'com.google.android:flexbox:2.0.1' + + compileOnly 'org.projectlombok:lombok:1.18.10' + annotationProcessor 'org.projectlombok:lombok:1.18.10' + annotationProcessor 'com.github.bumptech.glide:compiler:4.11.0' + + testImplementation 'junit:junit:4.12' + androidTestImplementation 'androidx.test.ext:junit:1.1.1' + androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0' __32bitImplementation files('libs/lbrysdk-0.67.1-release__arm.aar') __64bitImplementation files('libs/lbrysdk-0.67.1-release__arm64.aar') - - if (enableHermes) { - def hermesPath = "../../node_modules/hermes-engine/android/"; - debugImplementation files(hermesPath + "hermes-debug.aar") - releaseImplementation files(hermesPath + "hermes-release.aar") - } else { - implementation jscFlavor - } -} - -// Run this once to be able to run the application with BUCK -// puts all compile dependencies into folder libs for BUCK to use -task copyDownloadableDepsToLibs(type: Copy) { - from configurations.compile - into 'libs' } apply plugin: 'com.google.gms.google-services' diff --git a/app/build_defs.bzl b/app/build_defs.bzl deleted file mode 100644 index fff270f8..00000000 --- a/app/build_defs.bzl +++ /dev/null @@ -1,19 +0,0 @@ -"""Helper definitions to glob .aar and .jar targets""" - -def create_aar_targets(aarfiles): - for aarfile in aarfiles: - name = "aars__" + aarfile[aarfile.rindex("/") + 1:aarfile.rindex(".aar")] - lib_deps.append(":" + name) - android_prebuilt_aar( - name = name, - aar = aarfile, - ) - -def create_jar_targets(jarfiles): - for jarfile in jarfiles: - name = "jars__" + jarfile[jarfile.rindex("/") + 1:jarfile.rindex(".jar")] - lib_deps.append(":" + name) - prebuilt_jar( - name = name, - binary_jar = jarfile, - ) diff --git a/app/debug.keystore b/app/debug.keystore deleted file mode 100644 index 364e105e..00000000 Binary files a/app/debug.keystore and /dev/null differ diff --git a/app/proguard-rules.pro b/app/proguard-rules.pro index 11b02572..f1b42451 100644 --- a/app/proguard-rules.pro +++ b/app/proguard-rules.pro @@ -1,10 +1,21 @@ # Add project specific ProGuard rules here. -# By default, the flags in this file are appended to flags specified -# in /usr/local/Cellar/android-sdk/24.3.3/tools/proguard/proguard-android.txt -# You can edit the include path and order by changing the proguardFiles -# directive in build.gradle. +# You can control the set of applied configuration files using the +# proguardFiles setting in build.gradle. # # For more details, see # http://developer.android.com/guide/developing/tools/proguard.html -# Add any project specific keep options here: +# If your project uses WebView with JS, uncomment the following +# and specify the fully qualified class name to the JavaScript interface +# class: +#-keepclassmembers class fqcn.of.javascript.interface.for.webview { +# public *; +#} + +# Uncomment this to preserve the line number information for +# debugging stack traces. +#-keepattributes SourceFile,LineNumberTable + +# If you keep the line number information, uncomment this to +# hide the original source file name. +#-renamesourcefileattribute SourceFile diff --git a/app/src/androidTest/java/io/lbry/browser/ExampleInstrumentedTest.java b/app/src/androidTest/java/io/lbry/browser/ExampleInstrumentedTest.java new file mode 100644 index 00000000..50f333a6 --- /dev/null +++ b/app/src/androidTest/java/io/lbry/browser/ExampleInstrumentedTest.java @@ -0,0 +1,27 @@ +package io.lbry.browser; + +import android.content.Context; + +import androidx.test.platform.app.InstrumentationRegistry; +import androidx.test.ext.junit.runners.AndroidJUnit4; + +import org.junit.Test; +import org.junit.runner.RunWith; + +import static org.junit.Assert.*; + +/** + * Instrumented test, which will execute on an Android device. + * + * @see Testing documentation + */ +@RunWith(AndroidJUnit4.class) +public class ExampleInstrumentedTest { + @Test + public void useAppContext() { + // Context of the app under test. + Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext(); + + assertEquals("io.lbry.browser", appContext.getPackageName()); + } +} diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index a08ad7b2..fdb62306 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -1,16 +1,8 @@ + - - @@ -18,68 +10,48 @@ - - - - - - - - - - - - - - - - - - - + android:windowSoftInputMode="adjustResize"> + - - - - - - - - - - - + + + - + \ No newline at end of file diff --git a/app/src/main/assets/font_awesome_5_free_solid.otf b/app/src/main/assets/font_awesome_5_free_solid.otf new file mode 100644 index 00000000..9d8a0e62 Binary files /dev/null and b/app/src/main/assets/font_awesome_5_free_solid.otf differ diff --git a/app/src/main/assets/fonts/Feather.ttf b/app/src/main/assets/fonts/Feather.ttf deleted file mode 100644 index 852c7135..00000000 Binary files a/app/src/main/assets/fonts/Feather.ttf and /dev/null differ diff --git a/app/src/main/assets/fonts/FontAwesome.ttf b/app/src/main/assets/fonts/FontAwesome.ttf deleted file mode 100644 index 35acda2f..00000000 Binary files a/app/src/main/assets/fonts/FontAwesome.ttf and /dev/null differ diff --git a/app/src/main/assets/fonts/FontAwesome5_Brands.ttf b/app/src/main/assets/fonts/FontAwesome5_Brands.ttf deleted file mode 100644 index 5f72e912..00000000 Binary files a/app/src/main/assets/fonts/FontAwesome5_Brands.ttf and /dev/null differ diff --git a/app/src/main/assets/fonts/FontAwesome5_Regular.ttf b/app/src/main/assets/fonts/FontAwesome5_Regular.ttf deleted file mode 100644 index a309313d..00000000 Binary files a/app/src/main/assets/fonts/FontAwesome5_Regular.ttf and /dev/null differ diff --git a/app/src/main/assets/fonts/FontAwesome5_Solid.ttf b/app/src/main/assets/fonts/FontAwesome5_Solid.ttf deleted file mode 100644 index 7ece3282..00000000 Binary files a/app/src/main/assets/fonts/FontAwesome5_Solid.ttf and /dev/null differ diff --git a/app/src/main/assets/fonts/Inter-Medium.otf b/app/src/main/assets/fonts/Inter-Medium.otf deleted file mode 100644 index 1bcb0a93..00000000 Binary files a/app/src/main/assets/fonts/Inter-Medium.otf and /dev/null differ diff --git a/app/src/main/ic_launcher-playstore.png b/app/src/main/ic_launcher-playstore.png new file mode 100644 index 00000000..a82c512a Binary files /dev/null and b/app/src/main/ic_launcher-playstore.png differ diff --git a/app/src/main/java/io/lbry/browser/DownloadManager.java b/app/src/main/java/io/lbry/browser/DownloadManager.java deleted file mode 100644 index 1d8e82ab..00000000 --- a/app/src/main/java/io/lbry/browser/DownloadManager.java +++ /dev/null @@ -1,415 +0,0 @@ -package io.lbry.browser; - -import android.app.NotificationChannel; -import android.app.NotificationManager; -import android.app.PendingIntent; -import android.content.BroadcastReceiver; -import android.content.Context; -import android.content.Intent; -import android.net.Uri; -import android.os.Build; -import androidx.core.app.NotificationCompat; -import androidx.core.app.NotificationManagerCompat; -import androidx.core.content.ContextCompat; - -import io.lbry.browser.receivers.NotificationDeletedReceiver; -import io.lbry.lbrysdk.LbrynetService; - -import java.text.DecimalFormat; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; -import java.util.List; -import java.util.Random; - -public class DownloadManager { - private Context context; - - private List activeDownloads = new ArrayList(); - - private List completedDownloads = new ArrayList(); - - private Map downloadIdOutpointsMap = new HashMap(); - - // maintain a map of uris to writtenBytes, so that we check if it's changed and don't flood RN with update events every 500ms - private Map writtenDownloadBytes = new HashMap(); - - private HashMap builders = new HashMap(); - - private HashMap downloadIdNotificationIdMap = new HashMap(); - - private HashMap stoppedDownloadsMap = new HashMap(); - - private static final DecimalFormat DECIMAL_FORMAT = new DecimalFormat("#"); - - private static final int MAX_FILENAME_LENGTH = 20; - - private static final int MAX_PROGRESS = 100; - - private static final String GROUP_DOWNLOADS = "io.lbry.browser.GROUP_DOWNLOADS"; - - private static final String NOTIFICATION_CHANNEL_ID = "io.lbry.browser.DOWNLOADS_NOTIFICATION_CHANNEL"; - - private static boolean channelCreated = false; - - private static NotificationCompat.Builder groupBuilder = null; - - public static final String NOTIFICATION_ID_KEY = "io.lbry.browser.notificationId"; - - public static final String ACTION_DOWNLOAD_EVENT = "io.lbry.browser.ACTION_DOWNLOAD_EVENT"; - - public static final String ACTION_START = "start"; - - public static final String ACTION_COMPLETE = "complete"; - - public static final String ACTION_UPDATE = "update"; - - public static final int DOWNLOAD_NOTIFICATION_GROUP_ID = 20; - - public static boolean groupCreated = false; - - public DownloadManager(Context context) { - this.context = context; - } - - private int generateNotificationId() { - int id = 0; - Random random = new Random(); - do { - id = random.nextInt(); - } while (id < 1000); - - return id; - } - - private void createNotificationChannel() { - // Only applies to Android 8.0 Oreo (API Level 26) or higher - if (!channelCreated && Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { - NotificationManager notificationManager = - (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE); - NotificationChannel channel = new NotificationChannel( - NOTIFICATION_CHANNEL_ID, "LBRY Downloads", NotificationManager.IMPORTANCE_LOW); - channel.setDescription("LBRY file downloads"); - channel.setSound(null, null); - notificationManager.createNotificationChannel(channel); - } - } - - private void createNotificationGroup() { - if (!groupCreated) { - Intent intent = new Intent(context, NotificationDeletedReceiver.class); - intent.putExtra(NOTIFICATION_ID_KEY, DOWNLOAD_NOTIFICATION_GROUP_ID); - - PendingIntent pendingIntent = PendingIntent.getBroadcast(context, DOWNLOAD_NOTIFICATION_GROUP_ID, intent, 0); - NotificationManagerCompat notificationManager = NotificationManagerCompat.from(context); - groupBuilder = new NotificationCompat.Builder(context, NOTIFICATION_CHANNEL_ID); - groupBuilder.setContentTitle("Active LBRY downloads") - // contentText will be displayed if there are no notifications in the group - .setContentText("There are no active LBRY downloads.") - .setSmallIcon(android.R.drawable.stat_sys_download) - .setPriority(NotificationCompat.PRIORITY_LOW) - .setGroup(GROUP_DOWNLOADS) - .setGroupSummary(true) - .setDeleteIntent(pendingIntent); - notificationManager.notify(DOWNLOAD_NOTIFICATION_GROUP_ID, groupBuilder.build()); - - groupCreated = true; - } - } - - public static PendingIntent getLaunchPendingIntent(String uri, Context context) { - Intent launchIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(uri)); - launchIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP); - PendingIntent intent = PendingIntent.getActivity(context, 0, launchIntent, 0); - return intent; - } - - public void updateWrittenBytesForDownload(String id, double writtenBytes) { - if (!writtenDownloadBytes.containsKey(id)) { - writtenDownloadBytes.put(id, writtenBytes); - } - } - - public double getWrittenBytesForDownload(String id) { - if (writtenDownloadBytes.containsKey(id)) { - return writtenDownloadBytes.get(id); - } - - return -1; - } - - public void clearWrittenBytesForDownload(String id) { - if (writtenDownloadBytes.containsKey(id)) { - writtenDownloadBytes.remove(id); - } - } - - private Intent getDeleteDownloadIntent(String uri) { - Intent intent = new Intent(); - intent.setAction(LbrynetService.ACTION_DELETE_DOWNLOAD); - intent.putExtra("uri", uri); - intent.putExtra("nativeDelete", true); - return intent; - } - - public void startDownload(String id, String filename, String outpoint) { - if (filename == null || filename.trim().length() == 0) { - return; - } - - synchronized (this) { - if (!isDownloadActive(id)) { - activeDownloads.add(id); - downloadIdOutpointsMap.put(id, outpoint); - } - - createNotificationChannel(); - createNotificationGroup(); - - PendingIntent stopDownloadIntent = PendingIntent.getBroadcast(context, 0, getDeleteDownloadIntent(id), PendingIntent.FLAG_CANCEL_CURRENT); - - NotificationManagerCompat notificationManager = NotificationManagerCompat.from(context); - NotificationCompat.Builder builder = new NotificationCompat.Builder(context, NOTIFICATION_CHANNEL_ID); - // The file URI is used as the unique ID - builder.setColor(ContextCompat.getColor(context, R.color.lbryGreen)) - .setContentIntent(getLaunchPendingIntent(id, context)) - .setContentTitle(String.format("Downloading %s", truncateFilename(filename))) - .setGroup(GROUP_DOWNLOADS) - .setPriority(NotificationCompat.PRIORITY_LOW) - .setProgress(MAX_PROGRESS, 0, false) - .setSmallIcon(android.R.drawable.stat_sys_download) - .setOngoing(true) - .addAction(android.R.drawable.ic_menu_close_clear_cancel, "Stop", stopDownloadIntent); - - int notificationId = getNotificationId(id); - downloadIdNotificationIdMap.put(id, notificationId); - builders.put(notificationId, builder); - notificationManager.notify(notificationId, builder.build()); - - if (groupCreated && groupBuilder != null) { - groupBuilder.setSmallIcon(android.R.drawable.stat_sys_download); - notificationManager.notify(DOWNLOAD_NOTIFICATION_GROUP_ID, groupBuilder.build()); - } - } - } - - public void updateDownload(String id, String filename, double writtenBytes, double totalBytes) { - if (filename == null || filename.trim().length() == 0) { - return; - } - - synchronized (this) { - createNotificationChannel(); - createNotificationGroup(); - - NotificationManagerCompat notificationManager = NotificationManagerCompat.from(context); - NotificationCompat.Builder builder = null; - int notificationId = getNotificationId(id); - if (builders.containsKey(notificationId)) { - builder = builders.get(notificationId); - } else { - PendingIntent stopDownloadIntent = PendingIntent.getBroadcast(context, 0, getDeleteDownloadIntent(id), PendingIntent.FLAG_CANCEL_CURRENT); - builder = new NotificationCompat.Builder(context, NOTIFICATION_CHANNEL_ID); - builder.setColor(ContextCompat.getColor(context, R.color.lbryGreen)) - .setContentTitle(String.format("Downloading %s", truncateFilename(filename))) - .setPriority(NotificationCompat.PRIORITY_LOW) - .setOngoing(true) - .addAction(android.R.drawable.ic_menu_close_clear_cancel, "Stop", stopDownloadIntent); - builders.put(notificationId, builder); - } - - double progress = (writtenBytes / totalBytes) * 100; - builder.setContentIntent(getLaunchPendingIntent(id, context)) - .setContentText(String.format("%.0f%% (%s / %s)", progress, formatBytes(writtenBytes), formatBytes(totalBytes))) - .setGroup(GROUP_DOWNLOADS) - .setProgress(MAX_PROGRESS, new Double(progress).intValue(), false) - .setSmallIcon(android.R.drawable.stat_sys_download); - notificationManager.notify(notificationId, builder.build()); - - if (progress >= MAX_PROGRESS) { - builder.setContentTitle(String.format("Downloaded %s", truncateFilename(filename, 30))) - .setContentText(String.format("%s", formatBytes(totalBytes))) - .setGroup(GROUP_DOWNLOADS) - .setProgress(0, 0, false) - .setSmallIcon(android.R.drawable.stat_sys_download_done) - .setOngoing(false); - builder.mActions.clear(); - notificationManager.notify(notificationId, builder.build()); - - if (downloadIdNotificationIdMap.containsKey(id)) { - downloadIdNotificationIdMap.remove(id); - } - if (builders.containsKey(notificationId)) { - builders.remove(notificationId); - } - - // If there are no more downloads and the group exists, set the icon to stop animating - if (groupCreated && groupBuilder != null && downloadIdNotificationIdMap.size() == 0) { - groupBuilder.setSmallIcon(android.R.drawable.stat_sys_download_done); - notificationManager.notify(DOWNLOAD_NOTIFICATION_GROUP_ID, groupBuilder.build()); - } - - completeDownload(id, filename, totalBytes); - } - } - } - - public void completeDownload(String id, String filename, double totalBytes) { - synchronized (this) { - if (isDownloadActive(id)) { - activeDownloads.remove(id); - } - if (!isDownloadCompleted(id)) { - completedDownloads.add(id); - } - - NotificationManagerCompat notificationManager = NotificationManagerCompat.from(context); - NotificationCompat.Builder builder = null; - int notificationId = getNotificationId(id); - if (builders.containsKey(notificationId)) { - builder = builders.get(notificationId); - } else { - builder = new NotificationCompat.Builder(context, NOTIFICATION_CHANNEL_ID); - builder.setPriority(NotificationCompat.PRIORITY_LOW); - builders.put(notificationId, builder); - } - - builder.setContentTitle(String.format("Downloaded %s", truncateFilename(filename, 30))) - .setContentText(String.format("%s", formatBytes(totalBytes))) - .setGroup(GROUP_DOWNLOADS) - .setProgress(0, 0, false) - .setSmallIcon(android.R.drawable.stat_sys_download_done) - .setOngoing(false); - builder.mActions.clear(); - notificationManager.notify(notificationId, builder.build()); - - // If there are no more downloads and the group exists, set the icon to stop animating - checkGroupDownloadIcon(notificationManager); - } - } - - public void abortDownload(String id) { - synchronized (this) { - if (downloadIdNotificationIdMap.containsKey(id)) { - removeDownloadNotification(id); - } - activeDownloads.remove(id); - } - } - - public boolean isDownloadActive(String id) { - return (activeDownloads.contains(id)); - } - - public boolean isDownloadCompleted(String id) { - return (completedDownloads.contains(id)); - } - - public boolean hasActiveDownloads() { - return activeDownloads.size() > 0; - } - - public List getActiveDownloads() { - return activeDownloads; - } - - public List getCompletedDownloads() { - return completedDownloads; - } - - public String getOutpointForDownload(String uri) { - if (downloadIdOutpointsMap.containsKey(uri)) { - return downloadIdOutpointsMap.get(uri); - } - - return null; - } - - public void deleteDownloadUri(String uri) { - synchronized (this) { - activeDownloads.remove(uri); - completedDownloads.remove(uri); - - if (downloadIdOutpointsMap.containsKey(uri)) { - downloadIdOutpointsMap.remove(uri); - } - if (downloadIdNotificationIdMap.containsKey(uri)) { - removeDownloadNotification(uri); - } - } - } - - private void removeDownloadNotification(String id) { - int notificationId = downloadIdNotificationIdMap.get(id); - if (downloadIdNotificationIdMap.containsKey(id)) { - downloadIdNotificationIdMap.remove(id); - } - if (builders.containsKey(notificationId)) { - builders.remove(notificationId); - } - - NotificationManagerCompat notificationManager = NotificationManagerCompat.from(context); - NotificationCompat.Builder builder = builders.get(notificationId); - notificationManager.cancel(notificationId); - - checkGroupDownloadIcon(notificationManager); - if (builders.values().size() == 0) { - notificationManager.cancel(DOWNLOAD_NOTIFICATION_GROUP_ID); - groupCreated = false; - } - } - - private int getNotificationId(String id) { - if (downloadIdNotificationIdMap.containsKey(id)) { - return downloadIdNotificationIdMap.get(id); - } - - int notificationId = generateNotificationId(); - if (MainActivity.downloadNotificationIds != null && - !MainActivity.downloadNotificationIds.contains(notificationId)) { - MainActivity.downloadNotificationIds.add(notificationId); - } - downloadIdNotificationIdMap.put(id, notificationId); - return notificationId; - } - - private void checkGroupDownloadIcon(NotificationManagerCompat notificationManager) { - if (groupCreated && groupBuilder != null && downloadIdNotificationIdMap.size() == 0) { - groupBuilder.setSmallIcon(android.R.drawable.stat_sys_download_done); - notificationManager.notify(DOWNLOAD_NOTIFICATION_GROUP_ID, groupBuilder.build()); - } - } - - private static String formatBytes(double bytes) - { - if (bytes < 1048576) { // < 1MB - return String.format("%s KB", DECIMAL_FORMAT.format(bytes / 1024.0)); - } - - if (bytes < 1073741824) { // < 1GB - return String.format("%s MB", DECIMAL_FORMAT.format(bytes / (1024.0 * 1024.0))); - } - - return String.format("%s GB", DECIMAL_FORMAT.format(bytes / (1024.0 * 1024.0 * 1024.0))); - } - - private static String truncateFilename(String filename, int alternateMaxLength) { - int maxLength = alternateMaxLength > 0 ? alternateMaxLength : MAX_FILENAME_LENGTH; - if (filename.length() < maxLength) { - return filename; - } - - // Get the extension - int dotIndex = filename.lastIndexOf("."); - if (dotIndex > -1) { - String extension = filename.substring(dotIndex); - return String.format("%s...%s", filename.substring(0, maxLength - extension.length() - 4), extension); - } - - return String.format("%s...", filename.substring(0, maxLength - 3)); - } - - private static String truncateFilename(String filename) { - return truncateFilename(filename, 0); - } -} diff --git a/app/src/main/java/io/lbry/browser/FileViewActivity.java b/app/src/main/java/io/lbry/browser/FileViewActivity.java new file mode 100644 index 00000000..29cef886 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/FileViewActivity.java @@ -0,0 +1,468 @@ +package io.lbry.browser; + +import android.app.PictureInPictureParams; +import android.content.BroadcastReceiver; +import android.content.Context; +import android.content.Intent; +import android.content.IntentFilter; +import android.content.res.Configuration; +import android.net.Uri; +import android.os.AsyncTask; +import android.os.Build; +import android.os.Bundle; +import android.text.format.DateUtils; +import android.view.View; +import android.widget.ImageView; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.appcompat.app.AppCompatActivity; +import androidx.core.widget.NestedScrollView; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import com.google.android.exoplayer2.Player; +import com.google.android.exoplayer2.SimpleExoPlayer; +import com.google.android.exoplayer2.extractor.DefaultExtractorsFactory; +import com.google.android.exoplayer2.source.MediaSource; +import com.google.android.exoplayer2.source.ProgressiveMediaSource; +import com.google.android.exoplayer2.ui.PlayerControlView; +import com.google.android.exoplayer2.ui.PlayerView; +import com.google.android.exoplayer2.upstream.DefaultDataSourceFactory; +import com.google.android.exoplayer2.util.Util; +import com.google.android.flexbox.FlexboxLayoutManager; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.adapter.ClaimListAdapter; +import io.lbry.browser.adapter.TagListAdapter; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.ClaimCacheKey; +import io.lbry.browser.model.File; +import io.lbry.browser.model.Tag; +import io.lbry.browser.tasks.ClaimSearchTask; +import io.lbry.browser.tasks.FileListTask; +import io.lbry.browser.tasks.LighthouseSearchTask; +import io.lbry.browser.tasks.ResolveTask; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class FileViewActivity extends AppCompatActivity { + + public static FileViewActivity instance = null; + private static final int RELATED_CONTENT_SIZE = 16; + + private SimpleExoPlayer player; + private boolean loadFilePending; + private boolean resolving; + private Claim claim; + private ClaimListAdapter relatedContentAdapter; + private File file; + private BroadcastReceiver sdkReadyReceiver; + private Player.EventListener fileViewPlayerListener; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + + String claimId = null; + String url = null; + Intent intent = getIntent(); + if (intent != null) { + claimId = intent.getStringExtra("claimId"); + url = intent.getStringExtra("url"); + } + if (Helper.isNullOrEmpty(url)) { + // This activity should not be opened without a url set + finish(); + return; + } + + instance = this; + ClaimCacheKey key = new ClaimCacheKey(); + key.setClaimId(claimId); + if (url.contains("#")) { + key.setPermanentUrl(url); // use the same url for the key so that we can match the key for any value that's the same + key.setCanonicalUrl(url); + key.setShortUrl(url); + } + if (Lbry.claimCache.containsKey(key)) { + claim = Lbry.claimCache.get(key); + checkAndResetNowPlayingClaim(); + file = claim.getFile(); + if (file == null) { + loadFile(); + } + } + setContentView(R.layout.activity_file_view); + + if (claim == null) { + resolveUrl(url); + } + + registerSdkReadyReceiver(); + + fileViewPlayerListener = new Player.EventListener() { + @Override + public void onPlayerStateChanged(boolean playWhenReady, int playbackState) { + /*if (playbackState == Player.STATE_READY) { + MainActivity.setNowPlayingClaim(claim, FileViewActivity.this); + }*/ + } + }; + + initUi(); + renderClaim(); + } + + private void checkAndResetNowPlayingClaim() { + if (MainActivity.nowPlayingClaim != null && + !MainActivity.nowPlayingClaim.getClaimId().equalsIgnoreCase(claim.getClaimId())) { + MainActivity.clearNowPlayingClaim(this); + } + } + + protected void onNewIntent(Intent intent) { + super.onNewIntent(intent); + MainActivity.startingFileViewActivity = false; + if (intent != null) { + String newClaimId = intent.getStringExtra("claimId"); + String newUrl = intent.getStringExtra("url"); + + String oldClaimId = claim != null ? claim.getClaimId() : null; + if (!Helper.isNullOrEmpty(newClaimId)) { + if (newClaimId.equalsIgnoreCase(oldClaimId)) { + // it's the same claim, so we do nothing + if (MainActivity.appPlayer != null) { + PlayerView view = findViewById(R.id.file_view_exoplayer_view); + view.setPlayer(null); + view.setPlayer(MainActivity.appPlayer); + } + return; + } + + ClaimCacheKey key = new ClaimCacheKey(); + key.setClaimId(newClaimId); + if (!Helper.isNullOrEmpty(newUrl) && newUrl.contains("#")) { + key.setPermanentUrl(newUrl); + key.setCanonicalUrl(newUrl); + key.setShortUrl(newUrl); + } + if (Lbry.claimCache.containsKey(key)) { + claim = Lbry.claimCache.get(key); + checkAndResetNowPlayingClaim(); + file = claim.getFile(); + if (file == null) { + loadFile(); + } + renderClaim(); + } else { + findViewById(R.id.file_view_claim_display_area).setVisibility(View.INVISIBLE); + resolveUrl(newUrl); + } + } + } + } + + private void registerSdkReadyReceiver() { + IntentFilter filter = new IntentFilter(); + filter.addAction(MainActivity.ACTION_SDK_READY); + sdkReadyReceiver = new BroadcastReceiver() { + @Override + public void onReceive(Context context, Intent intent) { + // authenticate after we receive the sdk ready event + if (loadFilePending) { + loadFile(); + } + } + }; + registerReceiver(sdkReadyReceiver, filter); + } + + private String getStreamingUrl() { + if (file != null && !Helper.isNullOrEmpty(file.getStreamingUrl())) { + return file.getStreamingUrl(); + } + + return buildLbryTvStreamingUrl(); + } + + private String buildLbryTvStreamingUrl() { + return String.format("https://player.lbry.tv/content/claims/%s/%s/stream", claim.getName(), claim.getClaimId()); + } + + private void loadFile() { + if (!Lbry.SDK_READY) { + // make use of the lbry.tv streaming URL + loadFilePending = true; + return; + } + + loadFilePending = false; + // TODO: Check if it's paid content and then wait for the user to explicity request the file + String claimId = claim.getClaimId(); + FileListTask task = new FileListTask(claimId, null, new FileListTask.FileListResultHandler() { + @Override + public void onSuccess(List files) { + if (files.size() > 0) { + file = files.get(0); + claim.setFile(file); + } + } + + @Override + public void onError(Exception error) { + + } + }); + } + + protected void onResume() { + super.onResume(); + MainActivity.startingFileViewActivity = false; + } + + private void resolveUrl(String url) { + resolving = true; + View loadingView = findViewById(R.id.file_view_loading_container); + ResolveTask task = new ResolveTask(url, Lbry.LBRY_TV_CONNECTION_STRING, loadingView, new ResolveTask.ResolveResultHandler() { + @Override + public void onSuccess(List claims) { + if (claims.size() > 0) { + claim = claims.get(0); + checkAndResetNowPlayingClaim(); + loadFile(); + renderClaim(); + } + } + + @Override + public void onError(Exception error) { + resolving = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void initUi() { + findViewById(R.id.file_view_title_area).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + ImageView descIndicator = findViewById(R.id.file_view_desc_toggle_arrow); + View descriptionArea = findViewById(R.id.file_view_description_area); + if (descriptionArea.getVisibility() != View.VISIBLE) { + descriptionArea.setVisibility(View.VISIBLE); + descIndicator.setImageResource(R.drawable.ic_arrow_dropup); + } else { + descriptionArea.setVisibility(View.GONE); + descIndicator.setImageResource(R.drawable.ic_arrow_dropdown); + } + } + }); + + RecyclerView relatedContentList = findViewById(R.id.file_view_related_content_list); + relatedContentList.setNestedScrollingEnabled(false); + LinearLayoutManager llm = new LinearLayoutManager(this); + relatedContentList.setLayoutManager(llm); + } + + private void renderClaim() { + if (claim == null) { + return; + } + + ((NestedScrollView) findViewById(R.id.file_view_scroll_view)).scrollTo(0, 0); + findViewById(R.id.file_view_claim_display_area).setVisibility(View.VISIBLE); + + ImageView descIndicator = findViewById(R.id.file_view_desc_toggle_arrow); + descIndicator.setImageResource(R.drawable.ic_arrow_dropdown); + + findViewById(R.id.file_view_description_area).setVisibility(View.GONE); + ((TextView) findViewById(R.id.file_view_title)).setText(claim.getTitle()); + ((TextView) findViewById(R.id.file_view_description)).setText(claim.getDescription()); + ((TextView) findViewById(R.id.file_view_publisher_name)).setText( + Helper.isNullOrEmpty(claim.getPublisherName()) ? getString(R.string.anonymous) : claim.getPublisherName()); + + RecyclerView descTagsList = findViewById(R.id.file_view_tag_list); + FlexboxLayoutManager flm = new FlexboxLayoutManager(this); + descTagsList.setLayoutManager(flm); + + List tags = claim.getTagObjects(); + TagListAdapter tagListAdapter = new TagListAdapter(tags, this); + tagListAdapter.setClickListener(new TagListAdapter.TagClickListener() { + @Override + public void onTagClicked(Tag tag) { + Intent intent = new Intent(MainActivity.ACTION_OPEN_ALL_CONTENT_TAG); + intent.putExtra("tag", tag.getName()); + sendBroadcast(intent); + moveTaskToBack(true); + } + }); + descTagsList.setAdapter(tagListAdapter); + findViewById(R.id.file_view_tag_area).setVisibility(tags.size() > 0 ? View.VISIBLE : View.GONE); + + Claim.GenericMetadata metadata = claim.getValue(); + if (metadata instanceof Claim.StreamMetadata) { + Claim.StreamMetadata streamMetadata = (Claim.StreamMetadata) metadata; + long publishTime = streamMetadata.getReleaseTime() > 0 ? streamMetadata.getReleaseTime() * 1000 : claim.getTimestamp() * 1000; + ((TextView) findViewById(R.id.file_view_publish_time)).setText(DateUtils.getRelativeTimeSpanString( + publishTime, System.currentTimeMillis(), 0, DateUtils.FORMAT_ABBREV_RELATIVE)); + + // Check the metadata type + String mediaType = streamMetadata.getSource().getMediaType(); + // Use Exoplayer view if it's video / audio + if (mediaType.startsWith("audio") || mediaType.startsWith("video")) { + showExoplayerView(); + playMedia(); + } else if (mediaType.startsWith("text")) { + + } else if (mediaType.startsWith("image")) { + + } else { + // unsupported type + showUnsupportedView(); + } + } + + loadRelatedContent(); + } + + private void showUnsupportedView() { + findViewById(R.id.file_view_exoplayer_container).setVisibility(View.GONE); + + findViewById(R.id.file_view_unsupported_container).setVisibility(View.VISIBLE); + } + + private void showExoplayerView() { + findViewById(R.id.file_view_unsupported_container).setVisibility(View.GONE); + + findViewById(R.id.file_view_exoplayer_container).setVisibility(View.VISIBLE); + } + + private void playMedia() { + boolean newPlayerCreated = false; + if (MainActivity.appPlayer == null) { + MainActivity.appPlayer = new SimpleExoPlayer.Builder(this).build(); + MainActivity.appPlayer.setPlayWhenReady(true); + MainActivity.appPlayer.addListener(fileViewPlayerListener); + + newPlayerCreated = true; + } + + PlayerView view = findViewById(R.id.file_view_exoplayer_view); + PlayerControlView controlView = findViewById(R.id.file_view_exoplayer_control_view); + view.setPlayer(MainActivity.appPlayer); + controlView.setPlayer(MainActivity.appPlayer); + + if (MainActivity.nowPlayingClaim != null && + MainActivity.nowPlayingClaim.getClaimId().equalsIgnoreCase(claim.getClaimId()) && + !newPlayerCreated) { + // if the claim is already playing, we don't need to reload the media source + return; + } + + MainActivity.setNowPlayingClaim(claim, FileViewActivity.this); + String userAgent = Util.getUserAgent(this, getString(R.string.app_name)); + MediaSource mediaSource = new ProgressiveMediaSource.Factory( + new DefaultDataSourceFactory(this, userAgent), + new DefaultExtractorsFactory() + ).createMediaSource(Uri.parse(getStreamingUrl())); + MainActivity.appPlayer.prepare(mediaSource, true, true); + } + + private void loadRelatedContent() { + // reset the list view + ((RecyclerView) findViewById(R.id.file_view_related_content_list)).setAdapter(null); + + String title = claim.getTitle(); + String claimId = claim.getClaimId(); + ProgressBar relatedLoading = findViewById(R.id.file_view_related_content_progress); + LighthouseSearchTask relatedTask = new LighthouseSearchTask(title, RELATED_CONTENT_SIZE, 0, false, claimId, relatedLoading, new ClaimSearchTask.ClaimSearchResultHandler() { + @Override + public void onSuccess(List claims, boolean hasReachedEnd) { + List filteredClaims = new ArrayList<>(); + for (Claim c : claims) { + if (!c.getClaimId().equalsIgnoreCase(claim.getClaimId())) { + filteredClaims.add(c); + } + } + + relatedContentAdapter = new ClaimListAdapter(filteredClaims, FileViewActivity.this); + relatedContentAdapter.setListener(new ClaimListAdapter.ClaimListItemListener() { + @Override + public void onClaimClicked(Claim claim) { + Intent intent = new Intent(FileViewActivity.this, FileViewActivity.class); + intent.putExtra("claimId", claim.getClaimId()); + intent.putExtra("url", claim.getPermanentUrl()); + MainActivity.startingFileViewActivity = true; + startActivity(intent); + } + }); + + RecyclerView relatedContentList = findViewById(R.id.file_view_related_content_list); + relatedContentList.setAdapter(relatedContentAdapter); + relatedContentAdapter.notifyDataSetChanged(); + } + + @Override + public void onError(Exception error) { + + } + }); + relatedTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void onBackPressed() { + MainActivity.mainActive = true; + Intent intent = new Intent(this, MainActivity.class); + startActivity(intent); + finish(); + } + + protected void onUserLeaveHint() { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O && !MainActivity.mainActive) { + PictureInPictureParams params = new PictureInPictureParams.Builder().build(); + enterPictureInPictureMode(params); + } + } + + protected void onStop() { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { + if (isInPictureInPictureMode() && MainActivity.appPlayer != null) { + MainActivity.appPlayer.setPlayWhenReady(false); + } + } + super.onStop(); + } + + protected void onDestroy() { + Helper.unregisterReceiver(sdkReadyReceiver, this); + if (MainActivity.appPlayer != null && fileViewPlayerListener != null) { + MainActivity.appPlayer.removeListener(fileViewPlayerListener); + } + instance = null; + super.onDestroy(); + } + + private void renderPictureInPictureMode() { + findViewById(R.id.file_view_scroll_view).setVisibility(View.GONE); + findViewById(R.id.file_view_exoplayer_control_view).setVisibility(View.GONE); + } + private void renderFullMode() { + findViewById(R.id.file_view_scroll_view).setVisibility(View.VISIBLE); + + PlayerControlView controlView = findViewById(R.id.file_view_exoplayer_control_view); + controlView.setPlayer(null); + controlView.setPlayer(MainActivity.appPlayer); + } + + @Override + public void onPictureInPictureModeChanged(boolean isInPictureInPictureMode, Configuration newConfig) { + if (isInPictureInPictureMode) { + renderPictureInPictureMode(); + } else { + renderFullMode(); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/FirstRunActivity.java b/app/src/main/java/io/lbry/browser/FirstRunActivity.java new file mode 100644 index 00000000..0aa9a0d2 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/FirstRunActivity.java @@ -0,0 +1,128 @@ +package io.lbry.browser; + +import android.content.BroadcastReceiver; +import android.content.Context; +import android.content.Intent; +import android.content.IntentFilter; +import android.content.SharedPreferences; +import android.os.AsyncTask; +import android.os.Bundle; +import android.text.Html; +import android.text.method.LinkMovementMethod; +import android.view.View; +import android.widget.TextView; +import android.widget.Toast; + +import androidx.appcompat.app.AppCompatActivity; +import androidx.core.text.HtmlCompat; +import androidx.preference.PreferenceManager; + +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.Lbryio; + +public class FirstRunActivity extends AppCompatActivity { + + private BroadcastReceiver sdkReadyReceiver; + private BroadcastReceiver authReceiver; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setContentView(R.layout.activity_first_run); + + TextView welcomeTos = findViewById(R.id.welcome_text_view_tos); + welcomeTos.setMovementMethod(LinkMovementMethod.getInstance()); + welcomeTos.setText(HtmlCompat.fromHtml(getString(R.string.welcome_tos), HtmlCompat.FROM_HTML_MODE_LEGACY)); + + findViewById(R.id.welcome_link_use_lbry).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + finishFirstRun(); + } + }); + + registerAuthReceiver(); + if (!Lbry.SDK_READY) { + findViewById(R.id.welcome_wait_container).setVisibility(View.VISIBLE); + IntentFilter filter = new IntentFilter(); + filter.addAction(MainActivity.ACTION_SDK_READY); + sdkReadyReceiver = new BroadcastReceiver() { + @Override + public void onReceive(Context context, Intent intent) { + // authenticate after we receive the sdk ready event + authenticate(); + } + }; + registerReceiver(sdkReadyReceiver, filter); + } else { + authenticate(); + } + } + + private void registerAuthReceiver() { + IntentFilter filter = new IntentFilter(); + filter.addAction(MainActivity.ACTION_USER_AUTHENTICATION_SUCCESS); + filter.addAction(MainActivity.ACTION_USER_AUTHENTICATION_FAILED); + authReceiver = new BroadcastReceiver() { + @Override + public void onReceive(Context context, Intent intent) { + if (MainActivity.ACTION_USER_AUTHENTICATION_SUCCESS.equals(intent.getAction())) { + handleAuthenticationSuccess(); + } else { + handleAuthenticationFailed(); + } + } + }; + registerReceiver(authReceiver, filter); + } + + private void handleAuthenticationSuccess() { + + // first_auth completed event + + findViewById(R.id.welcome_wait_container).setVisibility(View.GONE); + findViewById(R.id.welcome_display).setVisibility(View.VISIBLE); + findViewById(R.id.welcome_link_use_lbry).setVisibility(View.VISIBLE); + } + + private void handleAuthenticationFailed() { + Toast.makeText(this, "Authentication failed.", Toast.LENGTH_LONG).show(); + } + + private void authenticate() { + new AuthenticateTask(this).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void finishFirstRun() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + sp.edit().putBoolean(MainActivity.PREFERENCE_KEY_FIRST_RUN_COMPLETED, true).apply(); + + // first_run_completed event + + finish(); + } + + @Override + public void onBackPressed() { + return; + } + + @Override + protected void onDestroy() { + Helper.unregisterReceiver(authReceiver, this); + Helper.unregisterReceiver(sdkReadyReceiver, this); + super.onDestroy(); + } + + private static class AuthenticateTask extends AsyncTask { + private Context context; + public AuthenticateTask(Context context) { + this.context = context; + } + protected Void doInBackground(Void... params) { + Lbryio.authenticate(context); + return null; + } + } +} diff --git a/app/src/main/java/io/lbry/browser/LbrynetMessagingService.java b/app/src/main/java/io/lbry/browser/LbrynetMessagingService.java deleted file mode 100644 index f5c4c4a8..00000000 --- a/app/src/main/java/io/lbry/browser/LbrynetMessagingService.java +++ /dev/null @@ -1,189 +0,0 @@ -package io.lbry.browser; - -import android.app.NotificationChannel; -import android.app.NotificationManager; -import android.app.PendingIntent; -import android.content.Context; -import android.content.Intent; -import android.content.SharedPreferences; -import android.media.RingtoneManager; -import android.net.Uri; -import android.os.Build; -import android.os.Bundle; -import androidx.core.app.NotificationCompat; -import androidx.core.content.ContextCompat; -import android.util.Log; - -import com.google.firebase.analytics.FirebaseAnalytics; -import com.google.firebase.messaging.FirebaseMessagingService; -import com.google.firebase.messaging.RemoteMessage; - -import io.lbry.lbrysdk.LbrynetService; -import io.lbry.browser.reactmodules.UtilityModule; - -import java.io.UnsupportedEncodingException; -import java.net.URLEncoder; -import java.util.ArrayList; -import java.util.List; -import java.util.Map; - -public class LbrynetMessagingService extends FirebaseMessagingService { - - private static final String TAG = "LbrynetMessagingService"; - - private static final String NOTIFICATION_CHANNEL_ID = "io.lbry.browser.LBRY_ENGAGEMENT_CHANNEL"; - - private static final String TYPE_SUBSCRIPTION = "subscription"; - - private static final String TYPE_REWARD = "reward"; - - private static final String TYPE_INTERESTS = "interests"; - - private static final String TYPE_CREATOR = "creator"; - - private FirebaseAnalytics firebaseAnalytics; - - @Override - public void onMessageReceived(RemoteMessage remoteMessage) { - Log.d(TAG, "From: " + remoteMessage.getFrom()); - if (firebaseAnalytics == null) { - firebaseAnalytics = FirebaseAnalytics.getInstance(this); - } - - Map payload = remoteMessage.getData(); - if (payload != null) { - String type = payload.get("type"); - String url = payload.get("target"); - String title = payload.get("title"); - String body = payload.get("body"); - String name = payload.get("name"); // notification name - String contentTitle = payload.get("content_title"); - String channelUrl = payload.get("channel_url"); - //String publishTime = payload.get("publish_time"); - String publishTime = null; - - if (type != null && getEnabledTypes().indexOf(type) > -1 && body != null && body.trim().length() > 0) { - // only log the receive event for valid notifications received - if (firebaseAnalytics != null) { - Bundle bundle = new Bundle(); - bundle.putString("name", name); - firebaseAnalytics.logEvent("lbry_notification_receive", bundle); - } - - sendNotification(title, body, type, url, name, contentTitle, channelUrl, publishTime); - } - } - } - - @Override - public void onNewToken(String token) { - Log.d(TAG, "Refreshed token: " + token); - - // If you want to send messages to this application instance or - // manage this apps subscriptions on the server side, send the - // Instance ID token to your app server. - sendRegistrationToServer(token); - } - - /** - * Persist token to third-party servers. - * - * Modify this method to associate the user's FCM InstanceID token with any server-side account - * maintained by your application. - * - * @param token The new token. - */ - private void sendRegistrationToServer(String token) { - // TODO: Implement this method to send token to your app server. - } - - /** - * Create and show a simple notification containing the received FCM message. - * - * @param messageBody FCM message body received. - */ - private void sendNotification(String title, String messageBody, String type, String url, String name, - String contentTitle, String channelUrl, String publishTime) { - //Intent intent = new Intent(this, MainActivity.class); - //intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); - if (url == null) { - if (TYPE_REWARD.equals(type)) { - url = "lbry://?rewards"; - } else { - // default to home page - url = "lbry://?discover"; - } - } else { - if (!MainActivity.isServiceRunning(this, LbrynetService.class) && - contentTitle != null && - channelUrl != null && - !url.startsWith("lbry://?") /* not a special url */ - ) { - // only enter lite mode when contentTitle and channelUrl are set (and the service isn't running yet) - // cold start - url = url + ((url.indexOf("?") > -1) ? "&liteMode=1" : "?liteMode=1"); - try { - if (contentTitle != null) { - url = url + "&contentTitle=" + URLEncoder.encode(contentTitle, "UTF-8"); - } - if (channelUrl != null) { - url = url + "&channelUrl=" + URLEncoder.encode(channelUrl, "UTF-8"); - } - if (publishTime != null) { - url = url + "&publishTime=" + URLEncoder.encode(publishTime, "UTF-8"); - } - } catch (UnsupportedEncodingException ex) { - // shouldn't happen - } - } - } - - Intent launchIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(url)); - launchIntent.putExtra("notification_name", name); - launchIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP); - PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, launchIntent, PendingIntent.FLAG_ONE_SHOT); - - Uri defaultSoundUri = RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION); - NotificationCompat.Builder notificationBuilder = - new NotificationCompat.Builder(this, NOTIFICATION_CHANNEL_ID) - .setColor(ContextCompat.getColor(this, R.color.lbryGreen)) - .setSmallIcon(R.drawable.ic_lbry) - .setContentTitle(title) - .setContentText(messageBody) - .setAutoCancel(true) - .setSound(defaultSoundUri) - .setContentIntent(pendingIntent); - - NotificationManager notificationManager = - (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE); - - // Since android Oreo notification channel is needed. - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { - NotificationChannel channel = new NotificationChannel( - NOTIFICATION_CHANNEL_ID, "LBRY Engagement", NotificationManager.IMPORTANCE_DEFAULT); - notificationManager.createNotificationChannel(channel); - } - - notificationManager.notify(9898, notificationBuilder.build()); - } - - public List getEnabledTypes() { - SharedPreferences sp = getSharedPreferences(MainActivity.SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - List enabledTypes = new ArrayList(); - - if (sp.getBoolean(UtilityModule.RECEIVE_SUBSCRIPTION_NOTIFICATIONS, true)) { - enabledTypes.add(TYPE_SUBSCRIPTION); - } - if (sp.getBoolean(UtilityModule.RECEIVE_REWARD_NOTIFICATIONS, true)) { - enabledTypes.add(TYPE_REWARD); - } - if (sp.getBoolean(UtilityModule.RECEIVE_INTERESTS_NOTIFICATIONS, true)) { - enabledTypes.add(TYPE_INTERESTS); - } - if (sp.getBoolean(UtilityModule.RECEIVE_CREATOR_NOTIFICATIONS, true)) { - enabledTypes.add(TYPE_CREATOR); - } - - return enabledTypes; - } -} \ No newline at end of file diff --git a/app/src/main/java/io/lbry/browser/LocalFileProvider.java b/app/src/main/java/io/lbry/browser/LocalFileProvider.java deleted file mode 100644 index 87964c27..00000000 --- a/app/src/main/java/io/lbry/browser/LocalFileProvider.java +++ /dev/null @@ -1,7 +0,0 @@ -package io.lbry.browser; - -import androidx.core.content.FileProvider; - -public class LocalFileProvider extends FileProvider { - -} diff --git a/app/src/main/java/io/lbry/browser/MainActivity.java b/app/src/main/java/io/lbry/browser/MainActivity.java index f5b0d6f0..babf6108 100644 --- a/app/src/main/java/io/lbry/browser/MainActivity.java +++ b/app/src/main/java/io/lbry/browser/MainActivity.java @@ -1,266 +1,1114 @@ package io.lbry.browser; -import android.annotation.SuppressLint; -import android.annotation.TargetApi; +import android.app.ActivityManager; import android.app.Notification; import android.app.NotificationManager; import android.app.PendingIntent; -import android.os.AsyncTask; -import android.os.Build; -import android.os.Bundle; -import android.os.Environment; -import android.app.Activity; -import android.app.ActivityManager; +import android.app.PictureInPictureParams; import android.content.BroadcastReceiver; -import android.content.ContentUris; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; -import android.content.pm.PackageManager; import android.content.SharedPreferences; -import android.database.Cursor; -import android.Manifest; -import android.net.Uri; +import android.content.res.Configuration; +import android.content.res.TypedArray; +import android.graphics.drawable.Drawable; +import android.os.AsyncTask; +import android.os.Build; +import android.os.Bundle; import android.os.Handler; -import android.provider.DocumentsContract; -import android.provider.MediaStore; -import android.provider.Settings; -import androidx.core.app.ActivityCompat; +import android.text.Editable; +import android.text.TextWatcher; +import android.util.Base64; +import android.util.Log; +import android.view.View; +import android.view.Menu; +import android.view.inputmethod.InputMethodManager; +import android.widget.EditText; +import android.widget.TextView; +import android.widget.Toast; + +import com.google.android.exoplayer2.SimpleExoPlayer; +import com.google.android.exoplayer2.ui.PlayerView; +import com.google.android.material.snackbar.Snackbar; + +import androidx.annotation.Nullable; +import androidx.appcompat.app.ActionBar; +import androidx.appcompat.app.ActionBarDrawerToggle; +import androidx.appcompat.app.AppCompatDelegate; import androidx.core.app.NotificationCompat; import androidx.core.app.NotificationManagerCompat; import androidx.core.content.ContextCompat; -import androidx.fragment.app.FragmentActivity; -import android.telephony.SmsMessage; -import android.widget.Toast; +import androidx.core.content.res.ResourcesCompat; +import androidx.core.graphics.drawable.DrawableCompat; +import androidx.core.view.GravityCompat; +import androidx.fragment.app.Fragment; +import androidx.fragment.app.FragmentManager; +import androidx.fragment.app.FragmentTransaction; +import androidx.navigation.ui.AppBarConfiguration; +import androidx.preference.PreferenceManager; +import androidx.drawerlayout.widget.DrawerLayout; +import androidx.appcompat.app.AppCompatActivity; +import androidx.appcompat.widget.Toolbar; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; -import com.azendoo.reactnativesnackbar.SnackbarPackage; -import com.brentvatne.react.ReactVideoPackage; -import com.dylanvann.fastimage.FastImageViewPackage; -import com.facebook.react.common.LifecycleState; -import com.facebook.react.modules.core.DefaultHardwareBackBtnHandler; -import com.facebook.react.ReactRootView; -import com.facebook.react.ReactInstanceManager; -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.ReactContext; -import com.facebook.react.bridge.WritableArray; -import com.facebook.react.bridge.WritableMap; -import com.facebook.react.modules.core.DeviceEventManagerModule; -import com.facebook.react.modules.core.PermissionAwareActivity; -import com.facebook.react.modules.core.PermissionListener; -import com.facebook.react.shell.MainReactPackage; -import com.facebook.soloader.SoLoader; -import com.google.firebase.analytics.FirebaseAnalytics; -import com.reactnativecommunity.asyncstorage.AsyncStoragePackage; -import com.reactnativecommunity.webview.RNCWebViewPackage; -import com.rnfs.RNFSPackage; -import com.swmansion.gesturehandler.react.RNGestureHandlerEnabledRootView; -import com.swmansion.gesturehandler.react.RNGestureHandlerPackage; -import com.swmansion.reanimated.ReanimatedPackage; -import com.RNFetchBlob.RNFetchBlobPackage; +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; -import io.lbry.browser.reactmodules.UtilityModule; -import io.lbry.browser.reactpackages.LbryReactPackage; -import io.lbry.browser.reactmodules.BackgroundMediaModule; +import java.io.BufferedReader; +import java.io.FileInputStream; +import java.io.InputStreamReader; +import java.net.ConnectException; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.concurrent.Executors; +import java.util.concurrent.ScheduledExecutorService; +import java.util.concurrent.TimeUnit; + +import io.lbry.browser.adapter.NavigationMenuAdapter; +import io.lbry.browser.adapter.UrlSuggestionListAdapter; +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.listener.WalletBalanceListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.ClaimCacheKey; +import io.lbry.browser.model.NavMenuItem; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.model.WalletSync; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.tasks.LighthouseAutoCompleteTask; +import io.lbry.browser.tasks.ResolveTask; +import io.lbry.browser.tasks.wallet.DefaultSyncTaskHandler; +import io.lbry.browser.tasks.wallet.SyncGetTask; +import io.lbry.browser.tasks.wallet.SyncTaskHandler; +import io.lbry.browser.tasks.wallet.WalletBalanceTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.ui.channel.ChannelFragment; +import io.lbry.browser.ui.following.FollowingFragment; +import io.lbry.browser.ui.search.SearchFragment; +import io.lbry.browser.ui.settings.SettingsFragment; +import io.lbry.browser.ui.allcontent.AllContentFragment; +import io.lbry.browser.ui.wallet.WalletFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; import io.lbry.lbrysdk.LbrynetService; import io.lbry.lbrysdk.ServiceHelper; import io.lbry.lbrysdk.Utils; +import lombok.Getter; -import java.io.File; -import java.net.ConnectException; -import java.util.ArrayList; -import java.util.Date; -import java.util.Iterator; -import java.util.List; +public class MainActivity extends AppCompatActivity implements SdkStatusListener { -import org.json.JSONObject; -import org.json.JSONArray; -import org.json.JSONException; -import org.reactnative.camera.RNCameraPackage; + public static SimpleExoPlayer appPlayer; + public static Claim nowPlayingClaim; + public static boolean startingFileViewActivity = false; + public static boolean mainActive = false; + private boolean enteringPIPMode = false; -public class MainActivity extends FragmentActivity implements DefaultHardwareBackBtnHandler, PermissionAwareActivity { + private Map openNavFragments; + private static final Map fragmentClassNavIdMap = new HashMap<>(); + static { + fragmentClassNavIdMap.put(FollowingFragment.class, NavMenuItem.ID_ITEM_FOLLOWING); + fragmentClassNavIdMap.put(WalletFragment.class, NavMenuItem.ID_ITEM_WALLET); + fragmentClassNavIdMap.put(SettingsFragment.class, NavMenuItem.ID_ITEM_SETTINGS); + fragmentClassNavIdMap.put(AllContentFragment.class, NavMenuItem.ID_ITEM_ALL_CONTENT); - private static Activity currentActivity = null; - private static final int OVERLAY_PERMISSION_REQ_CODE = 101; - private static final int STORAGE_PERMISSION_REQ_CODE = 201; - private static final int PHONE_STATE_PERMISSION_REQ_CODE = 202; - private static final int RECEIVE_SMS_PERMISSION_REQ_CODE = 203; - public static final int DOCUMENT_PICKER_RESULT_CODE = 301; - public static final String SHARED_PREFERENCES_NAME = "LBRY"; - public static final String SALT_KEY = "salt"; - public static final String DEVICE_ID_KEY = "deviceId"; - public static final String SOURCE_NOTIFICATION_ID_KEY = "sourceNotificationId"; - public static final String SETTING_KEEP_DAEMON_RUNNING = "keepDaemonRunning"; - public static List downloadNotificationIds = new ArrayList(); + // Internal (sub-)pages + fragmentClassNavIdMap.put(ChannelFragment.class, NavMenuItem.ID_ITEM_FOLLOWING); + fragmentClassNavIdMap.put(SearchFragment.class, NavMenuItem.ID_ITEM_FOLLOWING); + } - private BroadcastReceiver notificationsReceiver; - private BroadcastReceiver smsReceiver; + public static final int REQUEST_SIMPLE_SIGN_IN = 2001; + public static final int REQUEST_WALLET_SYNC_SIGN_IN = 2002; + + // broadcast action names + public static final String ACTION_SDK_READY = "io.lbry.browser.Broadcast.SdkReady"; + public static final String ACTION_AUTH_TOKEN_GENERATED = "io.lbry.browser.Broadcast.AuthTokenGenerated"; + public static final String ACTION_USER_AUTHENTICATION_SUCCESS = "io.lbry.browser.Broadcast.UserAuthenticationSuccess"; + public static final String ACTION_USER_SIGN_IN_SUCCESS = "io.lbry.browser.Broadcast.UserSignInSuccess"; + public static final String ACTION_USER_AUTHENTICATION_FAILED = "io.lbry.browser.Broadcast.UserAuthenticationFailed"; + public static final String ACTION_NOW_PLAYING_CLAIM_UPDATED = "io.lbry.browser.Broadcast.NowPlayingClaimUpdated"; + public static final String ACTION_NOW_PLAYING_CLAIM_CLEARED = "io.lbry.browser.Broadcast.NowPlayingClaimCleared"; + public static final String ACTION_OPEN_ALL_CONTENT_TAG = "io.lbry.browser.Broadcast.OpenAllContentTag"; + + // preference keys + public static final String PREFERENCE_KEY_DARK_MODE = "io.lbry.browser.preference.userinterface.DarkMode"; + public static final String PREFERENCE_KEY_NOTIFICATION_URL_SUGGESTIONS = "io.lbry.browser.preference.userinterface.UrlSuggestions"; + public static final String PREFERENCE_KEY_NOTIFICATION_SUBSCRIPTIONS = "io.lbry.browser.preference.notifications.Subscriptions"; + public static final String PREFERENCE_KEY_NOTIFICATION_REWARDS = "io.lbry.browser.preference.notifications.Rewards"; + public static final String PREFERENCE_KEY_NOTIFICATION_CONTENT_INTERESTS = "io.lbry.browser.preference.notifications.ContentInterests"; + public static final String PREFERENCE_KEY_KEEP_SDK_BACKGROUND = "io.lbry.browser.preference.other.KeepSdkInBackground"; + public static final String PREFERENCE_KEY_PARTICIPATE_DATA_NETWORK = "io.lbry.browser.preference.other.ParticipateInDataNetwork"; + + // Internal flags / setting preferences + public static final String PREFERENCE_KEY_INTERNAL_SKIP_WALLET_ACCOUNT = "io.lbry.browser.preference.internal.WalletSkipAccount"; + public static final String PREFERENCE_KEY_INTERNAL_WALLET_SYNC_ENABLED = "io.lbry.browser.preference.internal.WalletSyncEnabled"; + public static final String PREFERENCE_KEY_INTERNAL_WALLET_RECEIVE_ADDRESS = "io.lbry.browser.preference.internal.WalletReceiveAddress"; + + private final int CHECK_SDK_READY_INTERVAL = 1000; + + public static final String PREFERENCE_KEY_FIRST_RUN_COMPLETED = "io.lbry.browser.Preference.FirstRunCompleted"; + public static final String PREFERENCE_KEY_AUTH_TOKEN = "io.lbry.browser.Preference.AuthToken"; + + public static final String SECURE_VALUE_KEY_SAVED_PASSWORD = "io.lbry.browser.PX"; + + private static final String TAG = "io.lbry.browser.Main"; + + private NavigationMenuAdapter navMenuAdapter; + private UrlSuggestionListAdapter urlSuggestionListAdapter; + + // broadcast receivers private BroadcastReceiver serviceActionsReceiver; - private BroadcastReceiver downloadEventReceiver; - private FirebaseAnalytics firebaseAnalytics; - private ReactRootView mReactRootView; - private ReactInstanceManager mReactInstanceManager; + private BroadcastReceiver requestsReceiver; + private BroadcastReceiver userActionsReceiver; - /** - * Flag which indicates whether or not the service is running. Will be updated in the - * onResume method. - */ + private boolean userAuthenticated = false; + + private boolean appStarted; private boolean serviceRunning; private CheckSdkReadyTask checkSdkReadyTask; private boolean receivedStopService; - private PermissionListener permissionListener; - public static boolean lbrySdkReady; + private AppBarConfiguration mAppBarConfiguration; + private ActionBarDrawerToggle toggle; + @Getter + private DatabaseHelper dbHelper; + private int selectedMenuItemId = -1; + private List sdkStatusListeners; + private List walletBalanceListeners; + @Getter + private ScheduledExecutorService scheduler = Executors.newSingleThreadScheduledExecutor(); + private boolean walletBalanceUpdateScheduled; + private String pendingAllContentTag; + private String pendingChannelUrl; + private boolean pendingFollowingReload; + + private final List supportedMenuItemIds = Arrays.asList( + NavMenuItem.ID_ITEM_FOLLOWING, NavMenuItem.ID_ITEM_ALL_CONTENT, NavMenuItem.ID_ITEM_WALLET, NavMenuItem.ID_ITEM_SETTINGS + ); - protected String getMainComponentName() { - return "LBRYApp"; - } - - public static LaunchTiming CurrentLaunchTiming; - @Override protected void onCreate(Bundle savedInstanceState) { - CurrentLaunchTiming = new LaunchTiming(new Date()); + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + boolean darkMode = sp.getBoolean(PREFERENCE_KEY_DARK_MODE, false); + AppCompatDelegate.setDefaultNightMode(darkMode ? AppCompatDelegate.MODE_NIGHT_YES : AppCompatDelegate.MODE_NIGHT_NO); + + initKeyStore(); + loadAuthToken(); + + dbHelper = new DatabaseHelper(this); + if (!darkMode) { + getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR); + } + super.onCreate(savedInstanceState); - currentActivity = this; + setContentView(R.layout.activity_main); + Toolbar toolbar = findViewById(R.id.toolbar); + setSupportActionBar(toolbar); - SoLoader.init(this, false); - - // Register the stop service receiver (so that we close the activity if the user requests the service to stop) + // register receivers + registerRequestsReceiver(); registerServiceActionsReceiver(); + registerUserActionsReceiver(); - // Register SMS receiver for handling verification texts - registerSmsReceiver(); + // setup uri bar + setupUriBar(); - // Register the receiver to emit download events - registerDownloadEventReceiver(); + // other + openNavFragments = new HashMap<>(); + sdkStatusListeners = new ArrayList<>(); + walletBalanceListeners = new ArrayList<>(); - // Start the sdk service if it is not started - // Check the dht setting - SharedPreferences sp = getSharedPreferences(MainActivity.SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - LbrynetService.setDHTEnabled(sp.getBoolean(UtilityModule.DHT_ENABLED, false)); - serviceRunning = isServiceRunning(this, LbrynetService.class); - if (!serviceRunning) { - CurrentLaunchTiming.setColdStart(true); - ServiceHelper.start(this, "", LbrynetService.class, "lbrynetservice"); - } - checkSdkReady(); + sdkStatusListeners.add(this); - checkNotificationOpenIntent(getIntent()); + FragmentManager fragmentManager = getSupportFragmentManager(); + fragmentManager.addOnBackStackChangedListener(backStackChangedListener); - mReactRootView = new RNGestureHandlerEnabledRootView(this); - mReactInstanceManager = ReactInstanceManager.builder() - .setApplication(getApplication()) - .setCurrentActivity(this) - .setBundleAssetName("index.android.bundle") - .setJSMainModulePath("index") - .addPackage(new MainReactPackage()) - .addPackage(new AsyncStoragePackage()) - .addPackage(new FastImageViewPackage()) - .addPackage(new RNCWebViewPackage()) - .addPackage(new ReactVideoPackage()) - .addPackage(new ReanimatedPackage()) - .addPackage(new RNCameraPackage()) - .addPackage(new RNFetchBlobPackage()) - .addPackage(new RNFSPackage()) - .addPackage(new RNGestureHandlerPackage()) - .addPackage(new SnackbarPackage()) - .addPackage(new LbryReactPackage()) - .setUseDeveloperSupport(BuildConfig.DEBUG) - .setInitialLifecycleState(LifecycleState.RESUMED) - .build(); - mReactRootView.startReactApplication(mReactInstanceManager, "LBRYApp", null); - - registerNotificationsReceiver(); - - setContentView(mReactRootView); - } - - private void checkSdkReady() { - if (!lbrySdkReady) { - new Handler().postDelayed(new Runnable() { - @Override - public void run() { - if (checkSdkReadyTask != null && checkSdkReadyTask.getStatus() != AsyncTask.Status.FINISHED) { - // task already running - return; - } - checkSdkReadyTask = new CheckSdkReadyTask(); - checkSdkReadyTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); - } - }, 1000); - } - } - - private void checkNotificationOpenIntent(Intent intent) { - if (intent != null) { - String notificationName = intent.getStringExtra("notification_name"); - if (notificationName != null) { - logNotificationOpen(notificationName); - } - } - } - - private void logNotificationOpen(String name) { - if (firebaseAnalytics == null) { - firebaseAnalytics = FirebaseAnalytics.getInstance(this); - } - - Bundle bundle = new Bundle(); - bundle.putString("name", name); - firebaseAnalytics.logEvent("lbry_notification_open", bundle); - } - - private void registerDownloadEventReceiver() { - IntentFilter intentFilter = new IntentFilter(); - intentFilter.addAction(DownloadManager.ACTION_DOWNLOAD_EVENT); - downloadEventReceiver = new BroadcastReceiver() { - @Override - public void onReceive(Context context, Intent intent) { - String downloadAction = intent.getStringExtra("action"); - String uri = intent.getStringExtra("uri"); - String outpoint = intent.getStringExtra("outpoint"); - String fileInfoJson = intent.getStringExtra("file_info"); - - - if (uri == null || outpoint == null || (fileInfoJson == null && !"abort".equals(downloadAction))) { - return; - } - - String eventName = null; - WritableMap params = Arguments.createMap(); - params.putString("uri", uri); - params.putString("outpoint", outpoint); - - ReactContext reactContext = mReactInstanceManager.getCurrentReactContext(); - if ("abort".equals(downloadAction)) { - eventName = "onDownloadAborted"; - if (reactContext != null) { - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class).emit(eventName, params); - } - return; - } - - try { - JSONObject json = new JSONObject(fileInfoJson); - WritableMap fileInfo = JSONObjectToMap(json); - params.putMap("fileInfo", fileInfo); - - if (DownloadManager.ACTION_UPDATE.equals(downloadAction)) { - double progress = intent.getDoubleExtra("progress", 0); - params.putDouble("progress", progress); - eventName = "onDownloadUpdated"; - } else { - eventName = (DownloadManager.ACTION_START.equals(downloadAction)) ? "onDownloadStarted" : "onDownloadCompleted"; - } - - if (reactContext != null) { - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class).emit(eventName, params); - } - } catch (JSONException ex) { - // pass + DrawerLayout drawer = findViewById(R.id.drawer_layout); + toggle = new ActionBarDrawerToggle( + this, drawer, toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close) { + public void onDrawerClosed(View view) { + super.onDrawerClosed(view); + if (shouldOpenUserSelectedMenuItem) { + openSelectedMenuItem(); + shouldOpenUserSelectedMenuItem = false; } } }; - registerReceiver(downloadEventReceiver, intentFilter); + drawer.addDrawerListener(toggle); + toggle.syncState(); + toggle.setToolbarNavigationClickListener((view) -> { + if (toggle != null && !toggle.isDrawerIndicatorEnabled()) { + FragmentManager manager = getSupportFragmentManager(); + if (manager != null) { + manager.popBackStack(); + setSelectedNavMenuItemForFragment(getCurrentFragment()); + } + } + }); + + findViewById(R.id.global_now_playing_close).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + stopExoplayer(); + nowPlayingClaim = null; + findViewById(R.id.global_now_playing_card).setVisibility(View.GONE); + } + }); + + findViewById(R.id.global_now_playing_card).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (nowPlayingClaim != null) { + Intent intent = new Intent(MainActivity.this, FileViewActivity.class); + intent.putExtra("claimId", nowPlayingClaim.getClaimId()); + intent.putExtra("url", nowPlayingClaim.getPermanentUrl()); + startingFileViewActivity = true; + startActivity(intent); + } + } + }); + + // display custom navigation menu + LinearLayoutManager llm = new LinearLayoutManager(this); + RecyclerView navItemsView = findViewById(R.id.nav_view_items); + navItemsView.setLayoutManager(llm); + navMenuAdapter = new NavigationMenuAdapter(flattenNavMenu(buildNavMenu(this)), this); + navMenuAdapter.setListener(new NavigationMenuAdapter.NavigationMenuItemClickListener() { + @Override + public void onNavigationMenuItemClicked(NavMenuItem menuItem) { + if (navMenuAdapter.getCurrentItemId() == menuItem.getId() && menuItem.getId() != NavMenuItem.ID_ITEM_ALL_CONTENT) { + // already open + navMenuAdapter.setCurrentItem(menuItem); + closeDrawer(); + return; + } + + if (!supportedMenuItemIds.contains(menuItem.getId())) { + Snackbar.make(navItemsView, R.string.not_yet_implemented, Snackbar.LENGTH_LONG).show(); + } else { + navMenuAdapter.setCurrentItem(menuItem); + shouldOpenUserSelectedMenuItem = true; + selectedMenuItemId = menuItem.getId(); + } + closeDrawer(); + } + }); + navItemsView.setAdapter(navMenuAdapter); + + findViewById(R.id.sign_in_button).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + walletSyncSignIn(); + } + }); + } + + private boolean shouldOpenUserSelectedMenuItem; + + public void addSdkStatusListener(SdkStatusListener listener) { + if (!sdkStatusListeners.contains(listener)) { + sdkStatusListeners.add(listener); + } + } + + public void removeSdkStatusListener(SdkStatusListener listener) { + sdkStatusListeners.remove(listener); + } + + public void addWalletBalanceListener(WalletBalanceListener listener) { + if (!walletBalanceListeners.contains(listener)) { + walletBalanceListeners.add(listener); + } + } + + public void removeWalletBalanceListener(WalletBalanceListener listener) { + walletBalanceListeners.remove(listener); + } + + private void openSelectedMenuItem() { + switch (selectedMenuItemId) { + // TODO: reverse map lookup for class? + case NavMenuItem.ID_ITEM_FOLLOWING: + openFragment(FollowingFragment.class, true, NavMenuItem.ID_ITEM_FOLLOWING); + break; + case NavMenuItem.ID_ITEM_ALL_CONTENT: + openFragment(AllContentFragment.class, true, NavMenuItem.ID_ITEM_ALL_CONTENT); + break; + case NavMenuItem.ID_ITEM_WALLET: + openFragment(WalletFragment.class, true, NavMenuItem.ID_ITEM_WALLET); + break; + + case NavMenuItem.ID_ITEM_SETTINGS: + openFragment(SettingsFragment.class, true, NavMenuItem.ID_ITEM_SETTINGS); + break; + } + } + + public void openChannelClaim(Claim claim) { + Map params = new HashMap<>(); + params.put("url", claim.getPermanentUrl()); + params.put("claim", getCachedClaimForUrl(claim.getPermanentUrl())); + openFragment(ChannelFragment.class, true, NavMenuItem.ID_ITEM_FOLLOWING, params); + setWunderbarValue(claim.getShortUrl()); + } + + public void openChannelUrl(String url) { + Map params = new HashMap<>(); + params.put("url", url); + params.put("claim", getCachedClaimForUrl(url)); + openFragment(ChannelFragment.class, true, NavMenuItem.ID_ITEM_FOLLOWING, params); + setWunderbarValue(url); + } + + private Claim getCachedClaimForUrl(String url) { + ClaimCacheKey key = new ClaimCacheKey(); + key.setCanonicalUrl(url); + key.setPermanentUrl(url); + key.setShortUrl(url); + return Lbry.claimCache.containsKey(key) ? Lbry.claimCache.get(key) : null; + } + + public void setWunderbarValue(String value) { + EditText wunderbar = findViewById(R.id.wunderbar); + wunderbar.setText(value); + wunderbar.setSelection(0); + } + + private void openAllContentFragmentWithTag(String tag) { + Map params = new HashMap<>(); + params.put("singleTag", tag); + openFragment(AllContentFragment.class, true, NavMenuItem.ID_ITEM_ALL_CONTENT, params); + } + + public static void openFileUrl(String url, Context context) { + Intent intent = new Intent(context, FileViewActivity.class); + intent.putExtra("url", url); + startingFileViewActivity = true; + context.startActivity(intent); + } + + public static void openFileClaim(Claim claim, Context context) { + Intent intent = new Intent(context, FileViewActivity.class); + intent.putExtra("claimId", claim.getClaimId()); + intent.putExtra("url", claim.getPermanentUrl()); + startingFileViewActivity = true; + context.startActivity(intent); + } + + private FragmentManager.OnBackStackChangedListener backStackChangedListener = new FragmentManager.OnBackStackChangedListener() { + @Override + public void onBackStackChanged() { + FragmentManager manager = getSupportFragmentManager(); + if (manager != null) { + Fragment currentFragment = getCurrentFragment(); + + } + } + }; + + public void setSelectedMenuItemForFragment(Fragment fragment) { + if (fragment != null) { + Class fragmentClass = fragment.getClass(); + if (fragmentClassNavIdMap.containsKey(fragmentClass)) { + navMenuAdapter.setCurrentItem(fragmentClassNavIdMap.get(fragmentClass)); + } + } + } + + private void renderPictureInPictureMode() { + findViewById(R.id.content_main).setVisibility(View.GONE); + findViewById(R.id.global_now_playing_card).setVisibility(View.GONE); + getSupportActionBar().hide(); + + PlayerView pipPlayer = findViewById(R.id.pip_player); + pipPlayer.setVisibility(View.VISIBLE); + pipPlayer.setPlayer(appPlayer); + } + private void renderFullMode() { + getSupportActionBar().show(); + findViewById(R.id.content_main).setVisibility(View.VISIBLE); + findViewById(R.id.global_now_playing_card).setVisibility(View.VISIBLE); + + PlayerView pipPlayer = findViewById(R.id.pip_player); + pipPlayer.setVisibility(View.INVISIBLE); + pipPlayer.setPlayer(null); + } + + @Override + protected void onDestroy() { + unregisterReceivers(); + NotificationManagerCompat notificationManager = NotificationManagerCompat.from(this); + if (receivedStopService || !isServiceRunning(this, LbrynetService.class)) { + notificationManager.cancelAll(); + } + if (dbHelper != null) { + dbHelper.close(); + } + stopExoplayer(); + super.onDestroy(); + } + + private static void stopExoplayer() { + if (appPlayer != null) { + appPlayer.stop(true); + appPlayer.release(); + appPlayer = null; + } + } + + @Override + public boolean onCreateOptionsMenu(Menu menu) { + // Inflate the menu; this adds items to the action bar if it is present. + //getMenuInflater().inflate(R.menu.main, menu); + return true; + } + + public void updateWalletBalance() { + WalletBalanceTask task = new WalletBalanceTask(new WalletBalanceTask.WalletBalanceHandler() { + @Override + public void onSuccess(WalletBalance walletBalance) { + for (WalletBalanceListener listener : walletBalanceListeners) { + if (listener != null) { + listener.onWalletBalanceUpdated(walletBalance); + } + } + Lbry.walletBalance = walletBalance; + } + + @Override + public void onError(Exception error) { + + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + @Override + protected void onResume() { + super.onResume(); + mainActive = true; + + checkFirstRun(); + checkNowPlaying(); + + // check (and start) the LBRY SDK service + serviceRunning = isServiceRunning(this, LbrynetService.class); + if (!serviceRunning) { + ServiceHelper.start(this, "", LbrynetService.class, "lbrynetservice"); + } + checkSdkReady(); + showSignedInUser(); + + if (!Helper.isNullOrEmpty(pendingAllContentTag)) { + openAllContentFragmentWithTag(pendingAllContentTag); + pendingAllContentTag = null; + } + if (pendingFollowingReload) { + loadFollowingContent(); + pendingFollowingReload = false; + } + } + + @Override + protected void onPause() { + mainActive = false; + super.onPause(); + } + + private void toggleUrlSuggestions(boolean visible) { + View container = findViewById(R.id.url_suggestions_container); + View closeIcon = findViewById(R.id.wunderbar_close); + container.setVisibility(visible ? View.VISIBLE : View.GONE); + closeIcon.setVisibility(visible ? View.VISIBLE : View.GONE); + } + + private void setupUriBar() { + findViewById(R.id.wunderbar_close).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + findViewById(R.id.wunderbar).clearFocus(); + InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE); + imm.hideSoftInputFromWindow(view.getWindowToken(), 0); + } + }); + findViewById(R.id.wunderbar).setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + toggleUrlSuggestions(hasFocus); + } + }); + + ((EditText) findViewById(R.id.wunderbar)).addTextChangedListener(new TextWatcher() { + @Override + public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) { + + } + + @Override + public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) { + if (charSequence != null) { + handleUriInputChanged(charSequence.toString().trim()); + } + } + + @Override + public void afterTextChanged(Editable editable) { + + } + }); + + urlSuggestionListAdapter = new UrlSuggestionListAdapter(this); + urlSuggestionListAdapter.setListener(new UrlSuggestionListAdapter.UrlSuggestionClickListener() { + @Override + public void onUrlSuggestionClicked(UrlSuggestion urlSuggestion) { + switch (urlSuggestion.getType()) { + case UrlSuggestion.TYPE_CHANNEL: + // open channel page + break; + case UrlSuggestion.TYPE_FILE: + Context context = MainActivity.this; + if (urlSuggestion.getClaim() != null) { + openFileClaim(urlSuggestion.getClaim(), context); + } else { + openFileUrl(urlSuggestion.getUri().toString(), context); + } + break; + case UrlSuggestion.TYPE_SEARCH: + launchSearch(urlSuggestion.getText()); + break; + case UrlSuggestion.TYPE_TAG: + // open tag page + break; + } + findViewById(R.id.wunderbar).clearFocus(); + //findViewById(R.id.url_suggestions_container).setVisibility(View.GONE); + } + }); + + RecyclerView urlSuggestionList = findViewById(R.id.url_suggestions); + LinearLayoutManager llm = new LinearLayoutManager(this); + urlSuggestionList.setLayoutManager(llm); + urlSuggestionList.setAdapter(urlSuggestionListAdapter); + } + + private void launchSearch(String text) { + Fragment currentFragment = getCurrentFragment(); + if (currentFragment instanceof SearchFragment) { + ((SearchFragment) currentFragment).search(text, 0); + } else { + try { + SearchFragment fragment = SearchFragment.class.newInstance(); + fragment.setCurrentQuery(text); + openFragment(fragment, true); + } catch (Exception ex) { + // pass + } + } + } + + private void resolveUrlSuggestions(List urls) { + ResolveTask task = new ResolveTask(urls, Lbry.LBRY_TV_CONNECTION_STRING, null, new ResolveTask.ResolveResultHandler() { + @Override + public void onSuccess(List claims) { + for (int i = 0; i < claims.size(); i++) { + // build a simple url from the claim for matching + Claim claim = claims.get(i); + if (Helper.isNullOrEmpty(claim.getName())) { + continue; + } + + LbryUri simpleUrl = new LbryUri(); + if (claim.getName().startsWith("@")) { + // channel + simpleUrl.setChannelName(claim.getName()); + } else { + simpleUrl.setStreamName(claim.getName()); + } + + urlSuggestionListAdapter.setClaimForUrl(simpleUrl, claim); + } + urlSuggestionListAdapter.notifyDataSetChanged(); + } + + @Override + public void onError(Exception error) { + + } + }); + task.executeOnExecutor(AsyncTask.SERIAL_EXECUTOR); + } + + private void handleUriInputChanged(String text) { + // build the default suggestions + urlSuggestionListAdapter.clear(); + if (Helper.isNullOrEmpty(text)) { + return; + } + + List defaultSuggestions = buildDefaultSuggestions(text); + urlSuggestionListAdapter.addUrlSuggestions(defaultSuggestions); + + LighthouseAutoCompleteTask task = new LighthouseAutoCompleteTask(text, null, new LighthouseAutoCompleteTask.AutoCompleteResultHandler() { + @Override + public void onSuccess(List suggestions) { + urlSuggestionListAdapter.addUrlSuggestions(suggestions); + + List urls = urlSuggestionListAdapter.getItemUrls(); + resolveUrlSuggestions(urls); + } + + @Override + public void onError(Exception error) { + + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private List buildDefaultSuggestions(String text) { + List suggestions = new ArrayList(); + + // First item is always search + UrlSuggestion searchSuggestion = new UrlSuggestion(UrlSuggestion.TYPE_SEARCH, text); + suggestions.add(searchSuggestion); + + if (!text.matches(LbryUri.REGEX_INVALID_URI)) { + boolean isChannel = text.startsWith("@"); + if (!isChannel) { + LbryUri uri = new LbryUri(); + uri.setStreamName(text); + UrlSuggestion fileSuggestion = new UrlSuggestion(UrlSuggestion.TYPE_FILE, text); + fileSuggestion.setUri(uri); + suggestions.add(fileSuggestion); + } + + if (text.indexOf(' ') == -1) { + // channels and tags should not contain spaces + if (isChannel) { + LbryUri uri = new LbryUri(); + uri.setChannelName(text); + UrlSuggestion suggestion = new UrlSuggestion(UrlSuggestion.TYPE_CHANNEL, text); + suggestion.setUri(uri); + suggestions.add(suggestion); + } else { + UrlSuggestion suggestion = new UrlSuggestion(UrlSuggestion.TYPE_TAG, text); + suggestions.add(suggestion); + } + } + } + + return suggestions; + } + + private void checkNowPlaying() { + if (nowPlayingClaim != null) { + findViewById(R.id.global_now_playing_card).setVisibility(View.VISIBLE); + ((TextView) findViewById(R.id.global_now_playing_title)).setText(nowPlayingClaim.getTitle()); + ((TextView) findViewById(R.id.global_now_playing_channel_title)).setText(nowPlayingClaim.getPublisherTitle()); + } + if (appPlayer != null) { + PlayerView playerView = findViewById(R.id.global_now_playing_player_view); + playerView.setPlayer(null); + playerView.setPlayer(appPlayer); + playerView.setUseController(false); + } + } + + private void initKeyStore() { + try { + Lbry.KEYSTORE = Utils.initKeyStore(this); + } catch (Exception ex) { + // This shouldn't happen, but in case it does. + Toast.makeText(this, "The keystore could not be initialized. The app requires a secure keystore to run properly.", Toast.LENGTH_LONG).show(); + finish(); + } + } + + private void checkFirstRun() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + boolean firstRunCompleted = sp.getBoolean(PREFERENCE_KEY_FIRST_RUN_COMPLETED, false); + if (!firstRunCompleted) { + startActivity(new Intent(this, FirstRunActivity.class)); + } else if (!appStarted) { + // first run completed, startup + startup(); + } + } + + public static boolean isServiceRunning(Context context, Class serviceClass) { + ActivityManager manager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE); + for (ActivityManager.RunningServiceInfo serviceInfo : manager.getRunningServices(Integer.MAX_VALUE)) { + if (serviceClass.getName().equals(serviceInfo.service.getClassName())) { + return true; + } + } + + return false; + } + + private void loadAuthToken() { + // Check if an auth token is present and then set it for Lbryio + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + String encryptedAuthToken = sp.getString(PREFERENCE_KEY_AUTH_TOKEN, null); + if (!Helper.isNullOrEmpty(encryptedAuthToken)) { + try { + Lbryio.AUTH_TOKEN = new String(Utils.decrypt( + Base64.decode(encryptedAuthToken, Base64.NO_WRAP), this, Lbry.KEYSTORE), "UTF8"); + } catch (Exception ex) { + // pass. A new auth token would have to be generated if the old one cannot be decrypted + android.util.Log.e(TAG, "Could not decrypt existing auth token.", ex); + } + } + } + + private void checkSdkReady() { + if (!Lbry.SDK_READY) { + new Handler().postDelayed(() -> { + if (checkSdkReadyTask != null && checkSdkReadyTask.getStatus() != AsyncTask.Status.FINISHED) { + // task already running + return; + } + checkSdkReadyTask = new CheckSdkReadyTask(MainActivity.this, sdkStatusListeners); + checkSdkReadyTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + }, CHECK_SDK_READY_INTERVAL); + } else { + scheduleWalletBalanceUpdate(); + } + } + + public void onSdkReady() { + if (Lbryio.isSignedIn()) { + checkSyncedWallet(); + } + scheduleWalletBalanceUpdate(); + } + + private void scheduleWalletBalanceUpdate() { + if (scheduler != null && !walletBalanceUpdateScheduled) { + scheduler.scheduleAtFixedRate(new Runnable() { + @Override + public void run() { + updateWalletBalance(); + } + }, 0, 5, TimeUnit.SECONDS); + walletBalanceUpdateScheduled = true; + } + } + + private void registerRequestsReceiver() { + IntentFilter intentFilter = new IntentFilter(); + intentFilter.addAction(ACTION_AUTH_TOKEN_GENERATED); + intentFilter.addAction(ACTION_OPEN_ALL_CONTENT_TAG); + intentFilter.addAction(ACTION_USER_SIGN_IN_SUCCESS); + requestsReceiver = new BroadcastReceiver() { + @Override + public void onReceive(Context context, Intent intent) { + String action = intent.getAction(); + if (ACTION_AUTH_TOKEN_GENERATED.equalsIgnoreCase(action)) { + handleAuthTokenGenerated(intent); + } else if (ACTION_OPEN_ALL_CONTENT_TAG.equalsIgnoreCase(action)) { + handleOpenContentTag(intent); + } else if (ACTION_USER_SIGN_IN_SUCCESS.equalsIgnoreCase(action)) { + handleUserSignInSuccess(intent); + } + } + + private void handleAuthTokenGenerated(Intent intent) { + // store the value + String encryptedAuthToken = intent.getStringExtra("authToken"); + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(MainActivity.this); + sp.edit().putString(PREFERENCE_KEY_AUTH_TOKEN, encryptedAuthToken).apply(); + } + + private void handleOpenContentTag(Intent intent) { + String tag = intent.getStringExtra("tag"); + if (!Helper.isNullOrEmpty(tag)) { + pendingAllContentTag = tag; + } + } + private void handleUserSignInSuccess(Intent intent) { + pendingFollowingReload = true; + } + private void handleOpenChannelUrl(String url) { + pendingChannelUrl = url; + } + }; + registerReceiver(requestsReceiver, intentFilter); + } + + private void loadFollowingContent() { + for (Fragment fragment : openNavFragments.values()) { + if (fragment instanceof FollowingFragment) { + ((FollowingFragment) fragment).loadFollowing(); + } + } + } + + private void registerUserActionsReceiver() { + IntentFilter intentFilter = new IntentFilter(); + intentFilter.addAction(ACTION_NOW_PLAYING_CLAIM_UPDATED); + intentFilter.addAction(ACTION_NOW_PLAYING_CLAIM_CLEARED); + userActionsReceiver = new BroadcastReceiver() { + @Override + public void onReceive(Context context, Intent intent) { + String action = intent.getAction(); + if (ACTION_NOW_PLAYING_CLAIM_UPDATED.equals(action)) { + handleNowPlayingClaimUpdated(); + } else if (ACTION_NOW_PLAYING_CLAIM_CLEARED.equals(action)) { + handleNowPlayingClaimCleared(); + } + } + + private void handleNowPlayingClaimUpdated() { + if (nowPlayingClaim != null) { + ((TextView) findViewById(R.id.global_now_playing_title)).setText(nowPlayingClaim.getTitle()); + ((TextView) findViewById(R.id.global_now_playing_channel_title)).setText(nowPlayingClaim.getPublisherTitle()); + } + } + + private void handleNowPlayingClaimCleared() { + findViewById(R.id.global_now_playing_card).setVisibility(View.GONE); + ((TextView) findViewById(R.id.global_now_playing_title)).setText(null); + ((TextView) findViewById(R.id.global_now_playing_channel_title)).setText(null); + } + }; + registerReceiver(userActionsReceiver, intentFilter); + } + + @Override + public void onBackPressed() { + DrawerLayout drawer = findViewById(R.id.drawer_layout); + if (drawer.isDrawerOpen(GravityCompat.START)) { + drawer.closeDrawer(GravityCompat.START); + } else { + // check fragment and nav history + FragmentManager manager = getSupportFragmentManager(); + int backCount = getSupportFragmentManager().getBackStackEntryCount(); + if (backCount > 0) { + // we can pop the stack + manager.popBackStack(); + setSelectedNavMenuItemForFragment(getCurrentFragment()); + } else if (!enterPIPMode()) { + // we're at the top of the stack + moveTaskToBack(true); + return; + } + } + } + + public void simpleSignIn() { + Intent intent = new Intent(this, VerificationActivity.class); + intent.putExtra("flow", VerificationActivity.VERIFICATION_FLOW_SIGN_IN); + startActivityForResult(intent, REQUEST_SIMPLE_SIGN_IN); + } + + public void walletSyncSignIn() { + Intent intent = new Intent(this, VerificationActivity.class); + intent.putExtra("flow", VerificationActivity.VERIFICATION_FLOW_WALLET); + startActivityForResult(intent, REQUEST_WALLET_SYNC_SIGN_IN); + } + + @Override + protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) { + super.onActivityResult(requestCode, resultCode, data); + if (resultCode == RESULT_OK) { + // user signed in + showSignedInUser(); + + if (requestCode == REQUEST_WALLET_SYNC_SIGN_IN) { + for (Fragment fragment : openNavFragments.values()) { + if (fragment instanceof WalletFragment) { + ((WalletFragment) fragment).onWalletSyncEnabled(); + } + } + } + } + } + + private void showSignedInUser() { + if (Lbryio.isSignedIn()) { + findViewById(R.id.sign_in_button).setVisibility(View.GONE); + findViewById(R.id.signed_in_email_container).setVisibility(View.VISIBLE); + ((TextView) findViewById(R.id.signed_in_email)).setText(Lbryio.getSignedInEmail()); + findViewById(R.id.sign_in_header_divider).setBackgroundColor(getResources().getColor(R.color.lightDivider)); + } + } + + private Fragment getCurrentFragment() { + int backCount = getSupportFragmentManager().getBackStackEntryCount(); + if (backCount > 0) { + try { + Fragment fragment = getSupportFragmentManager().getFragments().get(backCount - 1); + return fragment; + } catch (IndexOutOfBoundsException ex) { + return null; + } + } + return null; + } + + public void hideActionBar() { + ActionBar actionBar = getSupportActionBar(); + if (actionBar != null) { + actionBar.hide(); + } + } + public void showActionBar() { + ActionBar actionBar = getSupportActionBar(); + if (actionBar != null) { + actionBar.show(); + } + } + + private void startup() { + final Context context = this; + // perform some tasks before launching + (new AsyncTask() { + protected void onPreExecute() { + hideActionBar(); + lockDrawer(); + findViewById(R.id.splash_view).setVisibility(View.VISIBLE); + } + protected Boolean doInBackground(Void... params) { + BufferedReader reader = null; + try { + // Load the installation id from the file system + String lbrynetDir = String.format("%s/%s", Utils.getAppInternalStorageDir(context), "lbrynet"); + String installIdPath = String.format("%s/install_id", lbrynetDir); + reader = new BufferedReader(new InputStreamReader(new FileInputStream(installIdPath))); + String installId = reader.readLine(); + if (Helper.isNullOrEmpty(installId)) { + // no install_id found (first run didn't start the sdk successfully?) + return false; + } + + // load the exchange rate + if (Lbryio.LBCUSDRate == 0) { + Lbryio.loadExchangeRate(); + } + + Lbry.INSTALLATION_ID = installId; + if (Lbryio.currentUser == null) { + Lbryio.authenticate(context); + } + Lbryio.newInstall(context); + + // (light) fetch subscriptions + if (Lbryio.cacheSubscriptions.size() == 0) { + List subscriptions = new ArrayList<>(); + List subUrls = new ArrayList<>(); + JSONArray array = (JSONArray) Lbryio.parseResponse(Lbryio.call("subscription", "list", context)); + if (array != null) { + for (int i = 0; i < array.length(); i++) { + JSONObject item = array.getJSONObject(i); + String claimId = item.getString("claim_id"); + String channelName = item.getString("channel_name"); + + LbryUri url = new LbryUri(); + url.setChannelName(channelName); + url.setClaimId(claimId); + subscriptions.add(new Subscription(channelName, url.toString())); + subUrls.add(url.toString()); + } + Lbryio.cacheSubscriptions = subscriptions; + + // resolve subscriptions + if (subUrls.size() > 0 && Lbryio.cacheResolvedSubscriptions.size() != Lbryio.cacheSubscriptions.size()) { + List resolvedSubs = Lbry.resolve(subUrls, Lbry.LBRY_TV_CONNECTION_STRING); + Lbryio.cacheResolvedSubscriptions = resolvedSubs; + } + } + } + } catch (Exception ex) { + // nope + android.util.Log.e(TAG, String.format("App startup failed: %s", ex.getMessage()), ex); + return false; + } finally { + Helper.closeCloseable(reader); + } + + return true; + } + protected void onPostExecute(Boolean startupSuccessful) { + if (!startupSuccessful) { + Toast.makeText(context, R.string.startup_failed, Toast.LENGTH_LONG).show(); + finish(); + return; + } + + findViewById(R.id.splash_view).setVisibility(View.GONE); + unlockDrawer(); + showActionBar(); + + if (navMenuAdapter != null) { + navMenuAdapter.setCurrentItem(NavMenuItem.ID_ITEM_FOLLOWING); + } + + loadLastFragment(); + showSignedInUser(); + + appStarted = true; + } + }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void loadLastFragment() { + Fragment fragment = getCurrentFragment(); + + if (fragment != null) { + openFragment(fragment, true); + } else { + openFragment(FollowingFragment.class, false, NavMenuItem.ID_ITEM_FOLLOWING); + } + } + + private void setSelectedNavMenuItemForFragment(Fragment fragment) { + if (fragment == null) { + // assume the first fragment is selected + navMenuAdapter.setCurrentItem(NavMenuItem.ID_ITEM_FOLLOWING); + return; + } + + Class fragmentClass = fragment.getClass(); + if (fragmentClassNavIdMap.containsKey(fragmentClass)) { + navMenuAdapter.setCurrentItem(fragmentClassNavIdMap.get(fragmentClass)); + } + } + + @Override + protected void onUserLeaveHint() { + enterPIPMode(); + } + + protected boolean enterPIPMode() { + if (enteringPIPMode) { + return true; + } + + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O && + appPlayer != null && + FileViewActivity.instance == null && + !startingFileViewActivity) { + enteringPIPMode = true; + + getSupportActionBar().hide(); + findViewById(R.id.global_now_playing_card).setVisibility(View.GONE); + findViewById(R.id.pip_player).setVisibility(View.VISIBLE); + + PictureInPictureParams params = new PictureInPictureParams.Builder().build(); + enterPictureInPictureMode(params); + return true; + } + + return false; } private void registerServiceActionsReceiver() { @@ -290,6 +1138,12 @@ public class MainActivity extends FragmentActivity implements DefaultHardwareBac registerReceiver(serviceActionsReceiver, intentFilter); } + private void unregisterReceivers() { + Helper.unregisterReceiver(requestsReceiver, this); + Helper.unregisterReceiver(serviceActionsReceiver, this); + Helper.unregisterReceiver(userActionsReceiver, this); + } + private Notification buildServiceNotification() { NotificationCompat.Builder builder = new NotificationCompat.Builder(this, LbrynetService.NOTIFICATION_CHANNEL_ID); Intent contextIntent = new Intent(this, MainActivity.class); @@ -312,556 +1166,73 @@ public class MainActivity extends FragmentActivity implements DefaultHardwareBac return notification; } - private void registerNotificationsReceiver() { - // Background media receiver - IntentFilter filter = new IntentFilter(); - filter.addAction(BackgroundMediaModule.ACTION_PLAY); - filter.addAction(BackgroundMediaModule.ACTION_PAUSE); - notificationsReceiver = new BroadcastReceiver() { - @Override - public void onReceive(Context context, Intent intent) { - String action = intent.getAction(); - ReactContext reactContext = mReactInstanceManager.getCurrentReactContext(); - if (reactContext != null) { - if (BackgroundMediaModule.ACTION_PLAY.equals(action)) { - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) - .emit("onBackgroundPlayPressed", null); - } - if (BackgroundMediaModule.ACTION_PAUSE.equals(action)) { - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) - .emit("onBackgroundPausePressed", null); - } - } - } - }; - registerReceiver(notificationsReceiver, filter); + private static List buildNavMenu(Context context) { + NavMenuItem findContentGroup = new NavMenuItem(NavMenuItem.ID_GROUP_FIND_CONTENT, R.string.find_content, true, context); + NavMenuItem yourContentGroup = new NavMenuItem(NavMenuItem.ID_GROUP_YOUR_CONTENT, R.string.your_content, true, context); + NavMenuItem walletGroup = new NavMenuItem(NavMenuItem.ID_GROUP_WALLET, R.string.wallet, true, context); + NavMenuItem otherGroup = new NavMenuItem(NavMenuItem.ID_GROUP_OTHER, 0, true, context); + + findContentGroup.setItems(Arrays.asList( + new NavMenuItem(NavMenuItem.ID_ITEM_FOLLOWING, R.string.fa_heart, R.string.following, "Following", context), + new NavMenuItem(NavMenuItem.ID_ITEM_EDITORS_CHOICE, R.string.fa_star, R.string.editors_choice, "EditorsChoice", context), + new NavMenuItem(NavMenuItem.ID_ITEM_YOUR_TAGS, R.string.fa_hashtag, R.string.your_tags, "YourTags", context), + new NavMenuItem(NavMenuItem.ID_ITEM_ALL_CONTENT, R.string.fa_globe_americas, R.string.all_content, "AllContent", context) + )); + + yourContentGroup.setItems(Arrays.asList( + new NavMenuItem(NavMenuItem.ID_ITEM_NEW_PUBLISH, R.string.fa_upload, R.string.new_publish, "NewPublish", context), + new NavMenuItem(NavMenuItem.ID_ITEM_CHANNELS, R.string.fa_at, R.string.channels, "Channels", context), + new NavMenuItem(NavMenuItem.ID_ITEM_LIBRARY, R.string.fa_download, R.string.library, "Library", context), + new NavMenuItem(NavMenuItem.ID_ITEM_PUBLISHES, R.string.fa_cloud_upload, R.string.publishes, "Publishes", context) + )); + + walletGroup.setItems(Arrays.asList( + new NavMenuItem(NavMenuItem.ID_ITEM_WALLET, R.string.fa_wallet, R.string.wallet, "Wallet", context), + new NavMenuItem(NavMenuItem.ID_ITEM_REWARDS, R.string.fa_award, R.string.rewards, "Rewards", context), + new NavMenuItem(NavMenuItem.ID_ITEM_INVITES, R.string.fa_user_friends, R.string.invites, "Invites", context) + )); + + otherGroup.setItems(Arrays.asList( + new NavMenuItem(NavMenuItem.ID_ITEM_SETTINGS, R.string.fa_cog, R.string.settings, "Settings", context), + new NavMenuItem(NavMenuItem.ID_ITEM_ABOUT, R.string.fa_mobile_alt, R.string.about, "About", context) + )); + + return Arrays.asList(findContentGroup, yourContentGroup, walletGroup, otherGroup); } - public void registerSmsReceiver() { - if (!hasPermission(Manifest.permission.RECEIVE_SMS, this)) { - // don't create the receiver if we don't have the read sms permission - return; - } - - IntentFilter smsFilter = new IntentFilter(); - smsFilter.addAction("android.provider.Telephony.SMS_RECEIVED"); - smsReceiver = new BroadcastReceiver() { - @Override - public void onReceive(Context context, Intent intent) { - // Get the message - Bundle bundle = intent.getExtras(); - if (bundle != null) { - Object[] pdus = (Object[]) bundle.get("pdus"); - if (pdus != null && pdus.length > 0) { - SmsMessage sms = SmsMessage.createFromPdu((byte[]) pdus[0]); - String text = sms.getMessageBody(); - if (text == null || text.trim().length() == 0) { - return; - } - - // Retrieve verification code from the text message if it contains - // the strings "lbry", "verification code" and the colon (following the expected format) - text = text.toLowerCase(); - if (text.indexOf("lbry") > -1 && text.indexOf("verification code") > -1 && text.indexOf(":") > -1) { - String code = text.substring(text.lastIndexOf(":") + 1).trim(); - ReactContext reactContext = mReactInstanceManager.getCurrentReactContext(); - if (reactContext != null) { - WritableMap params = Arguments.createMap(); - params.putString("code", code); - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) - .emit("onVerificationCodeReceived", params); - } - } - } - } - } - }; - registerReceiver(smsReceiver, smsFilter); - } - - @Override - protected void onActivityResult(int requestCode, int resultCode, Intent data) { - if (requestCode == OVERLAY_PERMISSION_REQ_CODE) { - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { - if (!Settings.canDrawOverlays(this)) { - // SYSTEM_ALERT_WINDOW permission not granted... + // Flatten the structure into a single list for the RecyclerView + private static List flattenNavMenu(List navMenuItems) { + List flatMenu = new ArrayList<>(); + for (NavMenuItem item : navMenuItems) { + flatMenu.add(item); + if (item.getItems() != null) { + for (NavMenuItem subItem : item.getItems()) { + flatMenu.add(subItem); } } } - if (requestCode == DOCUMENT_PICKER_RESULT_CODE) { - ReactContext reactContext = mReactInstanceManager.getCurrentReactContext(); - if (reactContext != null) { - if (resultCode == RESULT_OK) { - Uri fileUri = data.getData(); - String filePath = getRealPathFromURI_API19(this, fileUri); - WritableMap params = Arguments.createMap(); - params.putString("path", filePath); - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) - .emit("onDocumentPickerFilePicked", params); - } else if (resultCode == RESULT_CANCELED) { - // user canceled or request failed - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) - .emit("onDocumentPickerCanceled", null); - } - } - } + return flatMenu; } - public static Activity getActivity() { - Activity activity = new Activity(); - activity = currentActivity; - return activity; + public static void setNowPlayingClaim(Claim claim, Context context) { + nowPlayingClaim = claim; + context.sendBroadcast(new Intent(ACTION_NOW_PLAYING_CLAIM_UPDATED)); } - @Override - public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) { - ReactContext reactContext = mReactInstanceManager.getCurrentReactContext(); - switch (requestCode) { - case STORAGE_PERMISSION_REQ_CODE: - if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { - if (BuildConfig.DEBUG && Build.VERSION.SDK_INT >= 23 && !Settings.canDrawOverlays(this)) { - Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, - Uri.parse("package:" + getPackageName())); - startActivityForResult(intent, OVERLAY_PERMISSION_REQ_CODE); - } - if (reactContext != null) { - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) - .emit("onStoragePermissionGranted", null); - } - } else { - // Permission not granted - /*Toast.makeText(this, - "LBRY requires access to your device storage to be able to download files and media." + - " Please enable the storage permission and restart the app.", Toast.LENGTH_LONG).show();*/ - if (reactContext != null) { - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) - .emit("onStoragePermissionRefused", null); - } - } - break; - - case PHONE_STATE_PERMISSION_REQ_CODE: - if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { - // Permission granted. Emit an onPhoneStatePermissionGranted event - if (reactContext != null) { - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) - .emit("onPhoneStatePermissionGranted", null); - } - } else { - // Permission not granted. Simply show a message. - Toast.makeText(this, - "No permission granted to read your device state. Rewards cannot be claimed.", Toast.LENGTH_LONG).show(); - } - break; - - case RECEIVE_SMS_PERMISSION_REQ_CODE: - if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { - // Permission granted. Emit an onPhoneStatePermissionGranted event - if (reactContext != null) { - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) - .emit("onReceiveSmsPermissionGranted", null); - } - - // register the receiver - if (smsReceiver == null) { - registerSmsReceiver(); - } - } else { - // Permission not granted. Simply show a message. - Toast.makeText(this, - "No permission granted to receive your SMS messages. You may have to enter the verification code manually.", - Toast.LENGTH_LONG).show(); - } - break; - } - - if (permissionListener != null) { - permissionListener.onRequestPermissionsResult(requestCode, permissions, grantResults); - } + public static void clearNowPlayingClaim(Context context) { + nowPlayingClaim = null; + context.sendBroadcast(new Intent(ACTION_NOW_PLAYING_CLAIM_CLEARED)); } - @Override - public void invokeDefaultOnBackPressed() { - super.onBackPressed(); - } + private static class CheckSdkReadyTask extends AsyncTask { + private Context context; + private List listeners; - @Override - protected void onPause() { - super.onPause(); - - if (mReactInstanceManager != null) { - mReactInstanceManager.onHostPause(this); + public CheckSdkReadyTask(Context context, List listeners) { + this.context = context; + this.listeners = new ArrayList<>(listeners); } - } - - @Override - protected void onResume() { - super.onResume(); - - SharedPreferences sp = getSharedPreferences(SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - LbrynetService.setDHTEnabled(sp.getBoolean(UtilityModule.DHT_ENABLED, false)); - - serviceRunning = isServiceRunning(this, LbrynetService.class); - if (!serviceRunning) { - ServiceHelper.start(this, "", LbrynetService.class, "lbrynetservice"); - } - checkSdkReady(); - - if (mReactInstanceManager != null) { - mReactInstanceManager.onHostResume(this, this); - } - } - - @Override - protected void onDestroy() { - // check service running setting and end it here - SharedPreferences sp = getSharedPreferences(SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - boolean shouldKeepDaemonRunning = sp.getBoolean(SETTING_KEEP_DAEMON_RUNNING, true); - if (!shouldKeepDaemonRunning) { - serviceRunning = isServiceRunning(this, LbrynetService.class); - if (serviceRunning) { - ServiceHelper.stop(this, LbrynetService.class); - } - } - - if (notificationsReceiver != null) { - unregisterReceiver(notificationsReceiver); - notificationsReceiver = null; - } - - if (smsReceiver != null) { - unregisterReceiver(smsReceiver); - smsReceiver = null; - } - - if (downloadEventReceiver != null) { - unregisterReceiver(downloadEventReceiver); - downloadEventReceiver = null; - } - - if (serviceActionsReceiver != null) { - unregisterReceiver(serviceActionsReceiver); - serviceActionsReceiver = null; - } - - NotificationManagerCompat notificationManager = NotificationManagerCompat.from(this); - notificationManager.cancel(BackgroundMediaModule.NOTIFICATION_ID); - notificationManager.cancel(DownloadManager.DOWNLOAD_NOTIFICATION_GROUP_ID); - if (downloadNotificationIds != null) { - for (int i = 0; i < downloadNotificationIds.size(); i++) { - notificationManager.cancel(downloadNotificationIds.get(i)); - } - } - if (receivedStopService || !isServiceRunning(this, LbrynetService.class)) { - notificationManager.cancelAll(); - } - super.onDestroy(); - - if (mReactInstanceManager != null) { - mReactInstanceManager.onHostDestroy(this); - } - } - - @Override - public void onBackPressed() { - if (mReactInstanceManager != null) { - mReactInstanceManager.onBackPressed(); - } else { - super.onBackPressed(); - } - } - - @TargetApi(Build.VERSION_CODES.M) - public void requestPermissions(String[] permissions, int requestCode, PermissionListener listener) { - permissionListener = listener; - ActivityCompat.requestPermissions(this, permissions, requestCode); - } - - @Override - public void onNewIntent(Intent intent) { - if (mReactInstanceManager != null) { - mReactInstanceManager.onNewIntent(intent); - } - - if (intent != null) { - int sourceNotificationId = intent.getIntExtra(SOURCE_NOTIFICATION_ID_KEY, -1); - if (sourceNotificationId > -1) { - NotificationManagerCompat notificationManager = NotificationManagerCompat.from(this); - notificationManager.cancel(sourceNotificationId); - } - - checkNotificationOpenIntent(intent); - } - - super.onNewIntent(intent); - } - - private static void checkPermission(String permission, int requestCode, String rationale, Context context, boolean forceRequest) { - if (ContextCompat.checkSelfPermission(context, permission) != PackageManager.PERMISSION_GRANTED) { - // Should we show an explanation? - if (!forceRequest && ActivityCompat.shouldShowRequestPermissionRationale((Activity) context, permission)) { - Toast.makeText(context, rationale, Toast.LENGTH_LONG).show(); - } else { - ActivityCompat.requestPermissions((Activity) context, new String[] { permission }, requestCode); - } - } - } - - private static void checkPermission(String permission, int requestCode, String rationale, Context context) { - checkPermission(permission, requestCode, rationale, context, false); - } - - public static boolean hasPermission(String permission, Context context) { - return (ContextCompat.checkSelfPermission(context, permission) == PackageManager.PERMISSION_GRANTED); - } - - public static void checkPhoneStatePermission(Context context) { - // Request read phone state permission - checkPermission(Manifest.permission.READ_PHONE_STATE, - PHONE_STATE_PERMISSION_REQ_CODE, - "LBRY requires optional access to be able to identify your device for rewards. " + - "You cannot claim rewards without this permission.", - context, - true); - } - - public static void checkReceiveSmsPermission(Context context) { - // Request read phone state permission - checkPermission(Manifest.permission.RECEIVE_SMS, - RECEIVE_SMS_PERMISSION_REQ_CODE, - "LBRY requires access to be able to read a verification text message for rewards.", - context, - true); - } - - public static void checkStoragePermission(Context context) { - // Request read phone state permission - checkPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, - STORAGE_PERMISSION_REQ_CODE, - "LBRY requires access to your device storage to be able to download files and media.", - context, - true); - } - - public static boolean isServiceRunning(Context context, Class serviceClass) { - ActivityManager manager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE); - for (ActivityManager.RunningServiceInfo serviceInfo : manager.getRunningServices(Integer.MAX_VALUE)) { - if (serviceClass.getName().equals(serviceInfo.service.getClassName())) { - return true; - } - } - - return false; - } - - private static WritableMap JSONObjectToMap(JSONObject jsonObject) throws JSONException { - WritableMap map = Arguments.createMap(); - Iterator keys = jsonObject.keys(); - while(keys.hasNext()) { - String key = keys.next(); - Object value = jsonObject.get(key); - if (value instanceof JSONArray) { - map.putArray(key, JSONArrayToList((JSONArray) value)); - } else if (value instanceof JSONObject) { - map.putMap(key, JSONObjectToMap((JSONObject) value)); - } else if (value instanceof Boolean) { - map.putBoolean(key, (Boolean) value); - } else if (value instanceof Integer) { - map.putInt(key, (Integer) value); - } else if (value instanceof Double) { - map.putDouble(key, (Double) value); - } else if (value instanceof String) { - map.putString(key, (String) value); - } else { - map.putString(key, value.toString()); - } - } - - return map; - } - - private static WritableArray JSONArrayToList(JSONArray jsonArray) throws JSONException { - WritableArray array = Arguments.createArray(); - for(int i = 0; i < jsonArray.length(); i++) { - Object value = jsonArray.get(i); - if (value instanceof JSONArray) { - array.pushArray(JSONArrayToList((JSONArray) value)); - } else if (value instanceof JSONObject) { - array.pushMap(JSONObjectToMap((JSONObject) value)); - } else if (value instanceof Boolean) { - array.pushBoolean((Boolean) value); - } else if (value instanceof Integer) { - array.pushInt((Integer) value); - } else if (value instanceof Double) { - array.pushDouble((Double) value); - } else if (value instanceof String) { - array.pushString((String) value); - } else { - array.pushString(value.toString()); - } - } - - return array; - } - - /** - * https://gist.github.com/HBiSoft/15899990b8cd0723c3a894c1636550a8 - */ - @SuppressLint("NewApi") - public static String getRealPathFromURI_API19(final Context context, final Uri uri) { - final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT; - - // DocumentProvider - if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) { - // ExternalStorageProvider - if (isExternalStorageDocument(uri)) { - final String docId = DocumentsContract.getDocumentId(uri); - final String[] split = docId.split(":"); - final String type = split[0]; - - // This is for checking Main Memory - if ("primary".equalsIgnoreCase(type)) { - if (split.length > 1) { - return Environment.getExternalStorageDirectory() + "/" + split[1]; - } else { - return Environment.getExternalStorageDirectory() + "/"; - } - // This is for checking SD Card - } else { - return "storage" + "/" + docId.replace(":", "/"); - } - - } - // DownloadsProvider - else if (isDownloadsDocument(uri)) { - String fileName = getFilePath(context, uri); - if (fileName != null) { - return Environment.getExternalStorageDirectory().toString() + "/Download/" + fileName; - } - - String id = DocumentsContract.getDocumentId(uri); - if (id.startsWith("raw:")) { - id = id.replaceFirst("raw:", ""); - File file = new File(id); - if (file.exists()) - return id; - } - - final Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"), Long.valueOf(id)); - return getDataColumn(context, contentUri, null, null); - } - // MediaProvider - else if (isMediaDocument(uri)) { - final String docId = DocumentsContract.getDocumentId(uri); - final String[] split = docId.split(":"); - final String type = split[0]; - - Uri contentUri = null; - if ("image".equals(type)) { - contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI; - } else if ("video".equals(type)) { - contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI; - } else if ("audio".equals(type)) { - contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI; - } - - final String selection = "_id=?"; - final String[] selectionArgs = new String[]{ - split[1] - }; - - return getDataColumn(context, contentUri, selection, selectionArgs); - } - } - // MediaStore (and general) - else if ("content".equalsIgnoreCase(uri.getScheme())) { - - // Return the remote address - if (isGooglePhotosUri(uri)) - return uri.getLastPathSegment(); - - return getDataColumn(context, uri, null, null); - } - // File - else if ("file".equalsIgnoreCase(uri.getScheme())) { - return uri.getPath(); - } - - return null; - } - - public static String getDataColumn(Context context, Uri uri, String selection, String[] selectionArgs) { - Cursor cursor = null; - final String column = "_data"; - final String[] projection = { - column - }; - - try { - cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs, null); - if (cursor != null && cursor.moveToFirst()) { - final int index = cursor.getColumnIndexOrThrow(column); - return cursor.getString(index); - } - } finally { - if (cursor != null) - cursor.close(); - } - return null; - } - - - public static String getFilePath(Context context, Uri uri) { - Cursor cursor = null; - final String[] projection = { MediaStore.MediaColumns.DISPLAY_NAME }; - - try { - cursor = context.getContentResolver().query(uri, projection, null, null, null); - if (cursor != null && cursor.moveToFirst()) { - final int index = cursor.getColumnIndexOrThrow(MediaStore.MediaColumns.DISPLAY_NAME); - return cursor.getString(index); - } - } finally { - if (cursor != null) - cursor.close(); - } - return null; - } - - /** - * @param uri The Uri to check. - * @return Whether the Uri authority is ExternalStorageProvider. - */ - public static boolean isExternalStorageDocument(Uri uri) { - return "com.android.externalstorage.documents".equals(uri.getAuthority()); - } - - /** - * @param uri The Uri to check. - * @return Whether the Uri authority is DownloadsProvider. - */ - public static boolean isDownloadsDocument(Uri uri) { - return "com.android.providers.downloads.documents".equals(uri.getAuthority()); - } - - /** - * @param uri The Uri to check. - * @return Whether the Uri authority is MediaProvider. - */ - public static boolean isMediaDocument(Uri uri) { - return "com.android.providers.media.documents".equals(uri.getAuthority()); - } - - /** - * @param uri The Uri to check. - * @return Whether the Uri authority is Google Photos. - */ - public static boolean isGooglePhotosUri(Uri uri) { - return "com.google.android.apps.photos.content".equals(uri.getAuthority()); - } - - private class CheckSdkReadyTask extends AsyncTask { public Boolean doInBackground(Void... params) { boolean sdkReady = false; @@ -870,20 +1241,13 @@ public class MainActivity extends FragmentActivity implements DefaultHardwareBac if (response != null) { JSONObject result = new JSONObject(response); JSONObject status = result.getJSONObject("result"); - - // send status response for splash page updates - WritableMap sdkStatus = JSONObjectToMap(status); - ReactContext reactContext = mReactInstanceManager.getCurrentReactContext(); - if (reactContext != null) { - WritableMap evtParams = Arguments.createMap(); - evtParams.putMap("status", sdkStatus); - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class).emit("onSdkStatusResponse", evtParams); + if (!Lbry.IS_STATUS_PARSED) { + Lbry.parseStatus(status.toString()); } + // TODO: Broadcast startup status changes JSONObject startupStatus = status.getJSONObject("startup_status"); - sdkReady = startupStatus.has("stream_manager") && startupStatus.has("wallet") && - startupStatus.getBoolean("stream_manager") && startupStatus.getBoolean("wallet") && - (status.getJSONObject("wallet").getLong("blocks_behind") <= 0); + sdkReady = startupStatus.getBoolean("stream_manager") && startupStatus.getBoolean("wallet"); } } catch (ConnectException ex) { // pass @@ -894,37 +1258,167 @@ public class MainActivity extends FragmentActivity implements DefaultHardwareBac return sdkReady; } protected void onPostExecute(Boolean sdkReady) { - lbrySdkReady = sdkReady; - ReactContext reactContext = mReactInstanceManager.getCurrentReactContext(); - if (sdkReady && reactContext != null) { - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class).emit("onSdkReady", null); + Lbry.SDK_READY = sdkReady; + if (context != null) { + if (sdkReady) { + context.sendBroadcast(new Intent(ACTION_SDK_READY)); + + // update listeners + for (SdkStatusListener listener : listeners) { + if (listener != null) { + listener.onSdkReady(); + } + } + } else if (context instanceof MainActivity) { + ((MainActivity) context).checkSdkReady(); + } + } + } + } + + public void showNavigationBackIcon() { + ActionBar actionBar = getSupportActionBar(); + if (actionBar != null) { + actionBar.setDisplayHomeAsUpEnabled(true); + } + if (toggle != null) { + TypedArray a = getTheme().obtainStyledAttributes(R.style.AppTheme, new int[] {R.attr.homeAsUpIndicator}); + int attributeResourceId = a.getResourceId(0, 0); + Drawable drawable = ResourcesCompat.getDrawable(getResources(), attributeResourceId, null); + DrawableCompat.setTint(drawable, ContextCompat.getColor(this, R.color.actionBarForeground)); + + toggle.setDrawerIndicatorEnabled(false); + toggle.setHomeAsUpIndicator(drawable); + } + } + + private void closeDrawer() { + DrawerLayout drawer = findViewById(R.id.drawer_layout); + drawer.closeDrawer(GravityCompat.START); + } + + public void lockDrawer() { + DrawerLayout drawer = findViewById(R.id.drawer_layout); + drawer.setDrawerLockMode(DrawerLayout.LOCK_MODE_LOCKED_CLOSED); + } + + public void unlockDrawer() { + DrawerLayout drawer = findViewById(R.id.drawer_layout); + drawer.setDrawerLockMode(DrawerLayout.LOCK_MODE_UNLOCKED); + } + + public void restoreToggle() { + ActionBar actionBar = getSupportActionBar(); + if (actionBar != null) { + actionBar.setDisplayHomeAsUpEnabled(false); + actionBar.setTitle(null); + } + if (toggle != null) { + toggle.setDrawerIndicatorEnabled(true); + } + unlockDrawer(); + showSearchBar(); + } + + public void hideSearchBar() { + findViewById(R.id.wunderbar_container).setVisibility(View.GONE); + } + + public void showSearchBar() { + findViewById(R.id.wunderbar_container).setVisibility(View.VISIBLE); + } + + @Override + public void onPictureInPictureModeChanged(boolean isInPictureInPictureMode, Configuration newConfig) { + enteringPIPMode = false; + if (isInPictureInPictureMode) { + // Hide the full-screen UI (controls, etc.) while in picture-in-picture mode. + renderPictureInPictureMode(); + } else { + // Restore the full-screen UI. + renderFullMode(); + } + } + + protected void onStop() { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { + if (!MainActivity.startingFileViewActivity && appPlayer != null && isInPictureInPictureMode()) { + appPlayer.setPlayWhenReady(false); + } + } + super.onStop(); + } + + public void openFragment(Fragment fragment, boolean allowNavigateBack) { + Fragment currentFragment = getCurrentFragment(); + if (currentFragment != null && currentFragment.equals(fragment)) { + return; + } + + try { + FragmentManager manager = getSupportFragmentManager(); + FragmentTransaction transaction = manager.beginTransaction().replace(R.id.content_main, fragment); + if (allowNavigateBack) { + transaction.addToBackStack(null); + } + transaction.commit(); + } catch (Exception ex) { + // pass + } + } + + public void openFragment(Class fragmentClass, boolean allowNavigateBack, int navItemId) { + openFragment(fragmentClass, allowNavigateBack, navItemId, null); + } + + private static String buildNavFragmentKey(Class fragmentClass, int navItemId) { + return String.format("%s-%d", fragmentClass.getName(), navItemId); + } + + public void openFragment(Class fragmentClass, boolean allowNavigateBack, int navItemId, Map params) { + try { + String key = buildNavFragmentKey(fragmentClass, navItemId); + Fragment fragment = openNavFragments.containsKey(key) ? openNavFragments.get(key) : (Fragment) fragmentClass.newInstance(); + if (fragment instanceof BaseFragment) { + ((BaseFragment) fragment).setParams(params); + } + Fragment currentFragment = getCurrentFragment(); + if (currentFragment != null && currentFragment.equals(fragment)) { + return; } - if (!sdkReady) { - checkSdkReady(); + fragment.setRetainInstance(true); + FragmentManager manager = getSupportFragmentManager(); + FragmentTransaction transaction = manager.beginTransaction().replace(R.id.content_main, fragment); + if (allowNavigateBack) { + transaction.addToBackStack(null); } + transaction.commit(); + + if (navItemId > -1) { + openNavFragments.put(key, fragment); + } + } catch (Exception ex) { + // pass } } - - public static class LaunchTiming { - private Date start; - private boolean coldStart; - - public LaunchTiming(Date start) { - this.start = start; - } - - public Date getStart() { - return start; - } - public void setStart(Date start) { - this.start = start; - } - public boolean isColdStart() { - return coldStart; - } - public void setColdStart(boolean coldStart) { - this.coldStart = coldStart; - } + + private void checkSyncedWallet() { + String password = Utils.getSecureValue(SECURE_VALUE_KEY_SAVED_PASSWORD, this, Lbry.KEYSTORE); + // Just check if the current user has a synced wallet, no need to do anything else here + SyncGetTask task = new SyncGetTask(password, false, null, new DefaultSyncTaskHandler() { + @Override + public void onSyncGetSuccess(WalletSync walletSync) { + Lbryio.userHasSyncedWallet = true; + Lbryio.setLastWalletSync(walletSync); + Lbryio.setLastRemoteHash(walletSync.getHash()); + } + + @Override + public void onSyncGetWalletNotFound() { } + @Override + public void onSyncGetError(Exception error) { } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); } } diff --git a/app/src/main/java/io/lbry/browser/VerificationActivity.java b/app/src/main/java/io/lbry/browser/VerificationActivity.java new file mode 100644 index 00000000..29342786 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/VerificationActivity.java @@ -0,0 +1,152 @@ +package io.lbry.browser; + +import android.content.Intent; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.View; + +import androidx.fragment.app.FragmentActivity; +import androidx.viewpager.widget.ViewPager; +import androidx.viewpager2.widget.ViewPager2; + +import com.google.android.material.snackbar.Snackbar; + +import java.util.Arrays; + +import io.lbry.browser.adapter.VerificationPagerAdapter; +import io.lbry.browser.listener.SignInListener; +import io.lbry.browser.listener.WalletSyncListener; +import io.lbry.browser.model.lbryinc.User; +import io.lbry.browser.tasks.FetchCurrentUserTask; +import io.lbry.browser.utils.Lbryio; + +public class VerificationActivity extends FragmentActivity implements SignInListener, WalletSyncListener { + + public static final int VERIFICATION_FLOW_SIGN_IN = 1; + public static final int VERIFICATION_FLOW_REWARDS = 2; + public static final int VERIFICATION_FLOW_WALLET = 3; + + private String email; + private boolean signedIn; + private int flow; + + @Override + public void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + + signedIn = Lbryio.isSignedIn(); + Intent intent = getIntent(); + if (intent != null) { + flow = intent.getIntExtra("flow", -1); + if (flow == -1 || (flow == VERIFICATION_FLOW_SIGN_IN && signedIn)) { + // no flow specified (or user is already signed in), just exit + setResult(signedIn ? RESULT_OK : RESULT_CANCELED); + finish(); + return; + } + } + + if (!Arrays.asList(VERIFICATION_FLOW_SIGN_IN, VERIFICATION_FLOW_REWARDS, VERIFICATION_FLOW_WALLET).contains(flow)) { + // invalid flow specified + setResult(RESULT_CANCELED); + finish(); + return; + } + + setContentView(R.layout.activity_verification); + ViewPager2 viewPager = findViewById(R.id.verification_pager); + viewPager.setUserInputEnabled(false); + viewPager.setSaveEnabled(false); + viewPager.setAdapter(new VerificationPagerAdapter(this)); + + if (Lbryio.isSignedIn() && flow == VERIFICATION_FLOW_WALLET) { + viewPager.setCurrentItem(1); + } + + findViewById(R.id.verification_close_button).setVisibility(View.VISIBLE); + findViewById(R.id.verification_close_button).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + setResult(RESULT_CANCELED); + finish(); + } + }); + } + + @Override + public void onBackPressed() { + // ignore back press + return; + } + + public void onEmailAdded(String email) { + this.email = email; + findViewById(R.id.verification_close_button).setVisibility(View.GONE); + } + public void onEmailEdit() { + findViewById(R.id.verification_close_button).setVisibility(View.VISIBLE); + } + public void onEmailVerified() { + Snackbar.make(findViewById(R.id.verification_pager), R.string.sign_in_successful, Snackbar.LENGTH_LONG).show(); + sendBroadcast(new Intent(MainActivity.ACTION_USER_SIGN_IN_SUCCESS)); + + if (flow == VERIFICATION_FLOW_SIGN_IN) { + final Intent resultIntent = new Intent(); + resultIntent.putExtra("flow", VERIFICATION_FLOW_SIGN_IN); + resultIntent.putExtra("email", email); + + // only sign in required, don't do anything else + FetchCurrentUserTask task = new FetchCurrentUserTask(new FetchCurrentUserTask.FetchUserTaskHandler() { + @Override + public void onSuccess(User user) { + Lbryio.currentUser = user; + setResult(RESULT_OK, resultIntent); + finish(); + } + + @Override + public void onError(Exception error) { + setResult(RESULT_CANCELED); + finish(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } else { + // change pager view depending on flow + FetchCurrentUserTask task = new FetchCurrentUserTask(new FetchCurrentUserTask.FetchUserTaskHandler() { + @Override + public void onSuccess(User user) { Lbryio.currentUser = user; } + @Override + public void onError(Exception error) { } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + + ViewPager2 viewPager = findViewById(R.id.verification_pager); + // for rewards, (show phone verification if not done, or manual verification if required) + + // for wallet sync, if password unlock is required, show password entry page + viewPager.setCurrentItem(1); + } + } + + @Override + public void onWalletSyncProcessing() { + findViewById(R.id.verification_close_button).setVisibility(View.GONE); + } + @Override + public void onWalletSyncWaitingForInput() { + findViewById(R.id.verification_close_button).setVisibility(View.VISIBLE); + } + + @Override + public void onWalletSyncEnabled() { + findViewById(R.id.verification_close_button).setVisibility(View.VISIBLE); + setResult(RESULT_OK); + finish(); + } + + @Override + public void onWalletSyncFailed(Exception error) { + findViewById(R.id.verification_close_button).setVisibility(View.VISIBLE); + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/ChannelFilterListAdapter.java b/app/src/main/java/io/lbry/browser/adapter/ChannelFilterListAdapter.java new file mode 100644 index 00000000..d463d97c --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/ChannelFilterListAdapter.java @@ -0,0 +1,119 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ImageView; +import android.widget.TextView; + +import androidx.recyclerview.widget.RecyclerView; + +import com.bumptech.glide.Glide; +import com.bumptech.glide.request.RequestOptions; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.Claim; +import io.lbry.browser.listener.ChannelItemSelectionListener; +import io.lbry.browser.utils.Helper; +import lombok.Getter; +import lombok.Setter; + +public class ChannelFilterListAdapter extends RecyclerView.Adapter { + private Context context; + private List items; + @Getter + private Claim selectedItem; + @Setter + private ChannelItemSelectionListener listener; + + public ChannelFilterListAdapter(Context context) { + this.context = context; + this.items = new ArrayList<>(); + + // Always list the placeholder as the first item + Claim claim = new Claim(); + claim.setPlaceholder(true); + items.add(claim); + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected View mediaContainer; + protected View alphaContainer; + protected TextView allTextView; + protected ImageView thumbnailView; + protected TextView alphaView; + protected TextView titleView; + public ViewHolder(View v) { + super(v); + mediaContainer = v.findViewById(R.id.channel_filter_media_container); + alphaContainer = v.findViewById(R.id.channel_filter_no_thumbnail); + alphaView = v.findViewById(R.id.channel_filter_alpha_view); + thumbnailView = v.findViewById(R.id.channel_filter_thumbnail); + titleView = v.findViewById(R.id.channel_filter_title); + allTextView = v.findViewById(R.id.channel_filter_all); + } + } + + public int getItemCount() { + return items != null ? items.size() : 0; + } + + public boolean isClaimSelected(Claim claim) { + return claim.equals(selectedItem); + } + + public void addClaims(List claims) { + for (Claim claim : claims) { + if (!items.contains(claim)) { + items.add(claim); + } + } + notifyDataSetChanged(); + } + + @Override + public ChannelFilterListAdapter.ViewHolder onCreateViewHolder(ViewGroup root, int viewType) { + View v = LayoutInflater.from(context).inflate(R.layout.list_item_channel_filter, root, false); + return new ChannelFilterListAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(ChannelFilterListAdapter.ViewHolder vh, int position) { + Claim claim = items.get(position); + vh.alphaView.setVisibility(claim.isPlaceholder() ? View.GONE : View.VISIBLE); + vh.titleView.setVisibility(claim.isPlaceholder() ? View.GONE : View.VISIBLE); + vh.allTextView.setVisibility(claim.isPlaceholder() ? View.VISIBLE : View.GONE); + + vh.titleView.setText(Helper.isNullOrEmpty(claim.getTitle()) ? claim.getName() : claim.getTitle()); + String thumbnailUrl = claim.getThumbnailUrl(); + if (!Helper.isNullOrEmpty(thumbnailUrl) && context != null) { + Glide.with(context.getApplicationContext()).load(thumbnailUrl).apply(RequestOptions.circleCropTransform()).into(vh.thumbnailView); + } + vh.alphaContainer.setVisibility(claim.isPlaceholder() || Helper.isNullOrEmpty(thumbnailUrl) ? View.VISIBLE : View.GONE); + vh.thumbnailView.setVisibility(claim.isPlaceholder() || Helper.isNullOrEmpty(thumbnailUrl) ? View.GONE : View.VISIBLE); + vh.alphaView.setText(claim.isPlaceholder() ? null : claim.getName().substring(1, 2)); + + int bgColor = Helper.generateRandomColorForValue(claim.getClaimId()); + Helper.setIconViewBackgroundColor(vh.alphaContainer, bgColor, claim.isPlaceholder(), context); + + vh.itemView.setSelected(isClaimSelected(claim)); + vh.itemView.setOnClickListener(view -> { + if (claim.isPlaceholder()) { + selectedItem = null; + if (listener != null) { + listener.onChannelSelectionCleared(); + } + } else if (!claim.equals(selectedItem)) { + selectedItem = claim; + if (listener != null) { + listener.onChannelItemSelected(claim); + } + } + notifyDataSetChanged(); + }); + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/ClaimListAdapter.java b/app/src/main/java/io/lbry/browser/adapter/ClaimListAdapter.java new file mode 100644 index 00000000..ed87c566 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/ClaimListAdapter.java @@ -0,0 +1,217 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.text.format.DateUtils; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ImageView; +import android.widget.TextView; + +import androidx.recyclerview.widget.RecyclerView; + +import com.bumptech.glide.Glide; +import com.bumptech.glide.request.RequestOptions; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.Claim; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryUri; +import lombok.Setter; + +public class ClaimListAdapter extends RecyclerView.Adapter { + private static final int VIEW_TYPE_STREAM = 1; + private static final int VIEW_TYPE_CHANNEL = 2; + private static final int VIEW_TYPE_FEATURED = 3; // featured search result + + private Context context; + private List items; + private List selectedItems; + @Setter + private ClaimListItemListener listener; + + public ClaimListAdapter(List items, Context context) { + this.context = context; + this.items = new ArrayList<>(items); + this.selectedItems = new ArrayList<>(); + } + + public List getSelectedItems() { + return this.selectedItems; + } + public void clearSelectedItems() { + this.selectedItems.clear(); + } + public boolean isClaimSelected(Claim claim) { + return selectedItems.contains(claim); + } + + public Claim getFeaturedItem() { + for (Claim claim : items) { + if (claim.isFeatured()) { + return claim; + } + } + return null; + } + + public void clearItems() { + clearSelectedItems(); + this.items.clear(); + notifyDataSetChanged(); + } + + public void addFeaturedItem(Claim claim) { + items.add(0, claim); + notifyDataSetChanged(); + } + + public void addItems(List claims) { + for (Claim claim : claims) { + if (!items.contains(claim)) { + items.add(claim); + } + } + notifyDataSetChanged(); + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected ImageView thumbnailView; + protected View noThumbnailView; + protected TextView alphaView; + protected TextView vanityUrlView; + protected TextView durationView; + protected TextView titleView; + protected TextView publisherView; + protected TextView publishTimeView; + public ViewHolder(View v) { + super(v); + alphaView = v.findViewById(R.id.claim_thumbnail_alpha); + noThumbnailView = v.findViewById(R.id.claim_no_thumbnail); + thumbnailView = v.findViewById(R.id.claim_thumbnail); + vanityUrlView = v.findViewById(R.id.claim_vanity_url); + durationView = v.findViewById(R.id.claim_duration); + titleView = v.findViewById(R.id.claim_title); + publisherView = v.findViewById(R.id.claim_publisher); + publishTimeView = v.findViewById(R.id.claim_publish_time); + } + } + + @Override + public int getItemCount() { + return items != null ? items.size() : 0; + } + + @Override + public int getItemViewType(int position) { + if (items.get(position).isFeatured()) { + return VIEW_TYPE_FEATURED; + } + + return Claim.TYPE_CHANNEL.equalsIgnoreCase(items.get(position).getValueType()) ? VIEW_TYPE_CHANNEL : VIEW_TYPE_STREAM; + } + + @Override + public ClaimListAdapter.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { + int viewResourceId = -1; + switch (viewType) { + case VIEW_TYPE_FEATURED: viewResourceId = R.layout.list_item_featured_search_result; break; + case VIEW_TYPE_CHANNEL: viewResourceId = R.layout.list_item_channel; break; + case VIEW_TYPE_STREAM: default: viewResourceId = R.layout.list_item_stream; break; + } + + View v = LayoutInflater.from(context).inflate(viewResourceId, parent, false); + return new ClaimListAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(ClaimListAdapter.ViewHolder vh, int position) { + int type = getItemViewType(position); + Claim item = items.get(position); + Claim.GenericMetadata metadata = item.getValue(); + Claim signingChannel = item.getSigningChannel(); + Claim.StreamMetadata streamMetadata = null; + if (metadata instanceof Claim.StreamMetadata) { + streamMetadata = (Claim.StreamMetadata) metadata; + } + String thumbnailUrl = item.getThumbnailUrl(); + long publishTime = (streamMetadata != null && streamMetadata.getReleaseTime() > 0) ? streamMetadata.getReleaseTime() * 1000 : item.getTimestamp() * 1000; + int bgColor = Helper.generateRandomColorForValue(item.getClaimId()); + if (bgColor == 0) { + bgColor = Helper.generateRandomColorForValue(item.getName()); + } + + vh.itemView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (listener != null) { + listener.onClaimClicked(item); + } + } + }); + + vh.publisherView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (listener != null && signingChannel != null) { + listener.onClaimClicked(signingChannel); + } + } + }); + + + vh.titleView.setText(Helper.isNullOrEmpty(item.getTitle()) ? item.getName() : item.getTitle()); + if (type == VIEW_TYPE_FEATURED) { + LbryUri vanityUrl = new LbryUri(); + vanityUrl.setClaimName(item.getName()); + vh.vanityUrlView.setText(vanityUrl.toString()); + } + + vh.noThumbnailView.setVisibility(Helper.isNullOrEmpty(thumbnailUrl) ? View.VISIBLE : View.GONE); + Helper.setIconViewBackgroundColor(vh.noThumbnailView, bgColor, false, context); + + if (type == VIEW_TYPE_FEATURED && item.isUnresolved()) { + vh.durationView.setVisibility(View.GONE); + vh.titleView.setText("Nothing here. Publish something!"); + vh.alphaView.setText(item.getName().substring(0, Math.min(5, item.getName().length() - 1))); + } else { + if (Claim.TYPE_STREAM.equalsIgnoreCase(item.getValueType())) { + long duration = item.getDuration(); + if (!Helper.isNullOrEmpty(thumbnailUrl)) { + Glide.with(context.getApplicationContext()). + load(thumbnailUrl). + centerCrop(). + placeholder(R.drawable.bg_thumbnail_placeholder). + into(vh.thumbnailView); + } + + vh.alphaView.setText(item.getName().substring(0, Math.min(5, item.getName().length() - 1))); + vh.publisherView.setText(signingChannel != null ? signingChannel.getName() : context.getString(R.string.anonymous)); + vh.publishTimeView.setText(DateUtils.getRelativeTimeSpanString( + publishTime, System.currentTimeMillis(), 0, DateUtils.FORMAT_ABBREV_RELATIVE)); + vh.durationView.setVisibility(duration > 0 ? View.VISIBLE : View.GONE); + vh.durationView.setText(Helper.formatDuration(duration)); + } else if (Claim.TYPE_CHANNEL.equalsIgnoreCase(item.getValueType())) { + if (!Helper.isNullOrEmpty(thumbnailUrl)) { + Glide.with(context.getApplicationContext()). + load(thumbnailUrl). + centerCrop(). + placeholder(R.drawable.bg_thumbnail_placeholder). + apply(RequestOptions.circleCropTransform()). + into(vh.thumbnailView); + } + vh.alphaView.setText(item.getName().substring(1, 2).toUpperCase()); + vh.publisherView.setText(item.getName()); + vh.publishTimeView.setText(DateUtils.getRelativeTimeSpanString( + publishTime, System.currentTimeMillis(), 0, DateUtils.FORMAT_ABBREV_RELATIVE)); + } + } + } + + public interface ClaimListItemListener { + void onClaimClicked(Claim claim); + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/NavigationMenuAdapter.java b/app/src/main/java/io/lbry/browser/adapter/NavigationMenuAdapter.java new file mode 100644 index 00000000..2f1de9be --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/NavigationMenuAdapter.java @@ -0,0 +1,105 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.view.LayoutInflater; +import android.view.MenuItem; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ImageView; +import android.widget.TextView; + +import androidx.recyclerview.widget.RecyclerView; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.NavMenuItem; +import io.lbry.browser.ui.controls.SolidIconView; +import lombok.Getter; +import lombok.Setter; + +public class NavigationMenuAdapter extends RecyclerView.Adapter { + private static final int TYPE_GROUP = 1; + private static final int TYPE_ITEM = 2; + + private Context context; + private List menuItems; + private NavMenuItem currentItem; + @Setter + private NavigationMenuItemClickListener listener; + + public NavigationMenuAdapter(List menuItems, Context context) { + this.menuItems = new ArrayList<>(menuItems); + this.context = context; + } + + public void setCurrentItem(int id) { + for (NavMenuItem item : menuItems) { + if (item.getId() == id) { + this.currentItem = item; + break; + } + } + notifyDataSetChanged(); + } + + public void setCurrentItem(NavMenuItem currentItem) { + this.currentItem = currentItem; + notifyDataSetChanged(); + } + + public int getCurrentItemId() { + return currentItem != null ? currentItem.getId() : -1; + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected SolidIconView iconView; + protected TextView titleView; + public ViewHolder(View v) { + super(v); + titleView = v.findViewById(R.id.nav_menu_title); + iconView = v.findViewById(R.id.nav_menu_item_icon); + } + } + + @Override + public int getItemCount() { + return menuItems != null ? menuItems.size() : 0; + } + + @Override + public int getItemViewType(int position) { + return menuItems.get(position).isGroup() ? TYPE_GROUP : TYPE_ITEM; + } + + @Override + public NavigationMenuAdapter.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { + View v = LayoutInflater.from(context).inflate(viewType == TYPE_GROUP ? + R.layout.list_item_nav_menu_group : R.layout.list_item_nav_menu_item, parent, false); + return new NavigationMenuAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(ViewHolder vh, int position) { + int type = getItemViewType(position); + NavMenuItem item = menuItems.get(position); + vh.titleView.setText(item.getTitle()); + if (type == TYPE_ITEM && vh.iconView != null) { + vh.iconView.setText(item.getIcon()); + } + vh.itemView.setSelected(item.equals(currentItem)); + vh.itemView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (listener != null) { + listener.onNavigationMenuItemClicked(item); + } + } + }); + } + + public interface NavigationMenuItemClickListener { + void onNavigationMenuItemClicked(NavMenuItem menuItem); + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/SuggestedChannelGridAdapter.java b/app/src/main/java/io/lbry/browser/adapter/SuggestedChannelGridAdapter.java new file mode 100644 index 00000000..5a68f561 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/SuggestedChannelGridAdapter.java @@ -0,0 +1,120 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ImageView; +import android.widget.TextView; + +import androidx.recyclerview.widget.RecyclerView; + +import com.bumptech.glide.Glide; +import com.bumptech.glide.request.RequestOptions; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.Claim; +import io.lbry.browser.listener.ChannelItemSelectionListener; +import io.lbry.browser.utils.Helper; +import lombok.Setter; + +public class SuggestedChannelGridAdapter extends RecyclerView.Adapter { + private Context context; + private List items; + private List selectedItems; + @Setter + private ChannelItemSelectionListener listener; + + public SuggestedChannelGridAdapter(List items, Context context) { + this.items = new ArrayList<>(items); + this.selectedItems = new ArrayList<>(); + this.context = context; + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected ImageView thumbnailView; + protected TextView alphaView; + protected TextView titleView; + protected TextView tagView; + public ViewHolder(View v) { + super(v); + alphaView = v.findViewById(R.id.suggested_channel_alpha_view); + thumbnailView = v.findViewById(R.id.suggested_channel_thumbnail); + titleView = v.findViewById(R.id.suggested_channel_title); + tagView = v.findViewById(R.id.suggested_channel_tag); + } + } + + public int getItemCount() { + return items != null ? items.size() : 0; + } + + public int getSelectedCount() { return selectedItems.size(); } + + public List getSelectedItems() { + return this.selectedItems; + } + public void clearSelectedItems() { + this.selectedItems.clear(); + } + public boolean isClaimSelected(Claim claim) { + return selectedItems.contains(claim); + } + + public void addClaims(List claims) { + for (Claim claim : claims) { + if (!items.contains(claim)) { + items.add(claim); + } + } + notifyDataSetChanged(); + } + + @Override + public SuggestedChannelGridAdapter.ViewHolder onCreateViewHolder(ViewGroup root, int viewType) { + View v = LayoutInflater.from(context).inflate(R.layout.list_item_suggested_channel, root, false); + return new SuggestedChannelGridAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(SuggestedChannelGridAdapter.ViewHolder vh, int position) { + Claim claim = items.get(position); + + String thumbnailUrl = claim.getThumbnailUrl(); + if (!Helper.isNullOrEmpty(thumbnailUrl)) { + vh.alphaView.setVisibility(View.GONE); + vh.thumbnailView.setVisibility(View.VISIBLE); + Glide.with(context.getApplicationContext()).load(thumbnailUrl).apply(RequestOptions.circleCropTransform()).into(vh.thumbnailView); + } else { + vh.alphaView.setVisibility(View.VISIBLE); + vh.thumbnailView.setVisibility(View.GONE); + } + + vh.alphaView.setText(claim.getFirstCharacter()); + vh.titleView.setText(Helper.isNullOrEmpty(claim.getTitle()) ? claim.getName() : claim.getTitle()); + + String firstTag = claim.getFirstTag(); + vh.tagView.setVisibility(Helper.isNullOrEmpty(firstTag) ? View.INVISIBLE : View.VISIBLE); + vh.tagView.setBackgroundResource(R.drawable.bg_tag); + vh.tagView.setText(firstTag); + vh.itemView.setSelected(isClaimSelected(claim)); + + vh.itemView.setOnClickListener(view -> { + if (selectedItems.contains(claim)) { + selectedItems.remove(claim); + if (listener != null) { + listener.onChannelItemDeselected(claim); + } + } else { + selectedItems.add(claim); + if (listener != null) { + listener.onChannelItemSelected(claim); + } + } + notifyDataSetChanged(); + }); + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/TagListAdapter.java b/app/src/main/java/io/lbry/browser/adapter/TagListAdapter.java new file mode 100644 index 00000000..5a7a4faa --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/TagListAdapter.java @@ -0,0 +1,74 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.TextView; + +import androidx.recyclerview.widget.RecyclerView; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.Tag; +import lombok.Setter; + +public class TagListAdapter extends RecyclerView.Adapter { + private Context context; + private List items; + @Setter + private TagClickListener clickListener; + + public TagListAdapter(List tags, Context context) { + this.context = context; + this.items = new ArrayList<>(tags); + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected TextView nameView; + public ViewHolder(View v) { + super(v); + nameView = v.findViewById(R.id.tag_name); + } + } + + public int getItemCount() { + return items != null ? items.size() : 0; + } + + public void addTags(List tags) { + for (Tag tag : tags) { + if (!items.contains(tag)) { + items.add(tag); + } + } + notifyDataSetChanged(); + } + + @Override + public TagListAdapter.ViewHolder onCreateViewHolder(ViewGroup root, int viewType) { + View v = LayoutInflater.from(context).inflate(R.layout.list_item_tag, root, false); + return new TagListAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(TagListAdapter.ViewHolder vh, int position) { + Tag tag = items.get(position); + vh.nameView.setText(tag.getName().toLowerCase()); + vh.itemView.setBackgroundResource(tag.isMature() ? R.drawable.bg_tag_mature : R.drawable.bg_tag); + vh.itemView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (clickListener != null) { + clickListener.onTagClicked(tag); + } + } + }); + } + + public interface TagClickListener { + void onTagClicked(Tag tag); + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/TransactionListAdapter.java b/app/src/main/java/io/lbry/browser/adapter/TransactionListAdapter.java new file mode 100644 index 00000000..ff4bd478 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/TransactionListAdapter.java @@ -0,0 +1,126 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.content.Intent; +import android.net.Uri; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ImageView; +import android.widget.TextView; + +import androidx.recyclerview.widget.RecyclerView; + +import java.text.DecimalFormat; +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.Date; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.Transaction; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryUri; +import lombok.Setter; + +public class TransactionListAdapter extends RecyclerView.Adapter { + + private static final String EXPLORER_TX_PREFIX = "https://explorer.lbry.com/tx"; + private static final DecimalFormat TX_LIST_AMOUNT_FORMAT = new DecimalFormat("#,##0.0000"); + private static final SimpleDateFormat TX_LIST_DATE_FORMAT = new SimpleDateFormat("MMM d"); + + private Context context; + private List items; + @Setter + private TransactionClickListener listener; + + public TransactionListAdapter(List transactions, Context context) { + this.context = context; + this.items = new ArrayList<>(transactions); + } + + public void clear() { + items.clear(); + notifyDataSetChanged(); + } + + public List getItems() { + return new ArrayList<>(items); + } + + public void addTransactions(List transactions) { + for (Transaction tx : transactions) { + if (!items.contains(tx)) { + items.add(tx); + } + } + notifyDataSetChanged(); + } + + public int getItemCount() { + return items != null ? items.size() : 0; + } + + @Override + public TransactionListAdapter.ViewHolder onCreateViewHolder(ViewGroup root, int viewType) { + View v = LayoutInflater.from(context).inflate(R.layout.list_item_transaction, root, false); + return new TransactionListAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(TransactionListAdapter.ViewHolder vh, int position) { + Transaction item = items.get(position); + vh.descView.setText(item.getDescriptionStringId()); + vh.amountView.setText(TX_LIST_AMOUNT_FORMAT.format(item.getValue().doubleValue())); + vh.claimView.setText(item.getClaim()); + vh.feeView.setText(context.getString(R.string.tx_list_fee, TX_LIST_AMOUNT_FORMAT.format(item.getFee().doubleValue()))); + vh.txidLinkView.setText(item.getTxid().substring(0, 8)); + vh.dateView.setText(TX_LIST_DATE_FORMAT.format(item.getTxDate())); + + vh.infoFeeContainer.setVisibility(!Helper.isNullOrEmpty(item.getClaim()) || Math.abs(item.getFee().doubleValue()) > 0 ? + View.VISIBLE : View.GONE); + + vh.txidLinkView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (context != null) { + Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(String.format("%s/%s", EXPLORER_TX_PREFIX, item.getTxid()))); + context.startActivity(intent); + } + } + }); + + vh.itemView.setOnClickListener(view -> { + if (listener != null) { + listener.onTransactionClicked(item); + } + }); + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected TextView descView; + protected TextView amountView; + protected TextView claimView; + protected TextView feeView; + protected TextView txidLinkView; + protected TextView dateView; + protected View infoFeeContainer; + + public ViewHolder(View v) { + super(v); + descView = v.findViewById(R.id.transaction_desc); + amountView = v.findViewById(R.id.transaction_amount); + claimView = v.findViewById(R.id.transaction_claim); + feeView = v.findViewById(R.id.transaction_fee); + txidLinkView = v.findViewById(R.id.transaction_id_link); + dateView = v.findViewById(R.id.transaction_date); + infoFeeContainer = v.findViewById(R.id.transaction_info_fee_container); + } + } + + public interface TransactionClickListener { + void onTransactionClicked(Transaction transaction); + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/UrlSuggestionListAdapter.java b/app/src/main/java/io/lbry/browser/adapter/UrlSuggestionListAdapter.java new file mode 100644 index 00000000..4b164be9 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/UrlSuggestionListAdapter.java @@ -0,0 +1,138 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.TextView; + +import androidx.recyclerview.widget.RecyclerView; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.ui.controls.SolidIconView; +import io.lbry.browser.utils.LbryUri; +import lombok.Setter; + +public class UrlSuggestionListAdapter extends RecyclerView.Adapter { + private Context context; + private List items; + @Setter + private UrlSuggestionClickListener listener; + + public UrlSuggestionListAdapter(Context context) { + this.context = context; + this.items = new ArrayList<>(); + } + + public void clear() { + items.clear(); + notifyDataSetChanged(); + } + + public List getItems() { + return new ArrayList<>(items); + } + + public List getItemUrls() { + List uris = new ArrayList<>(); + for (int i = 0; i < items.size(); i++) { + LbryUri uri = items.get(i).getUri(); + if (uri != null) { + uris.add(uri.toString()); + } + } + return uris; + } + + public void setClaimForUrl(LbryUri url, Claim claim) { + for (int i = 0; i < items.size(); i++) { + LbryUri thisUrl = items.get(i).getUri(); + if (thisUrl != null && thisUrl.equals(url)) { + items.get(i).setClaim(claim); + } + } + } + + public void addUrlSuggestions(List urlSuggestions) { + for (UrlSuggestion urlSuggestion : urlSuggestions) { + if (!items.contains(urlSuggestion)) { + items.add(urlSuggestion); + } + } + notifyDataSetChanged(); + } + + public int getItemCount() { + return items != null ? items.size() : 0; + } + + @Override + public UrlSuggestionListAdapter.ViewHolder onCreateViewHolder(ViewGroup root, int viewType) { + View v = LayoutInflater.from(context).inflate(R.layout.list_item_url_suggestion, root, false); + return new UrlSuggestionListAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(UrlSuggestionListAdapter.ViewHolder vh, int position) { + UrlSuggestion item = items.get(position); + + String fullTitle, desc; + int iconStringId; + switch (item.getType()) { + case UrlSuggestion.TYPE_CHANNEL: + iconStringId = R.string.fa_at; + fullTitle = item.getTitle(); + desc = item.getClaim() != null ? item.getClaim().getTitle() : + String.format(context.getString(R.string.view_channel_url_desc), item.getText()); + break; + case UrlSuggestion.TYPE_TAG: + iconStringId = R.string.fa_hashtag; + fullTitle = String.format(context.getString(R.string.tag_url_title), item.getText()); + desc = String.format(context.getString(R.string.explore_tag_url_desc), item.getText()); + break; + case UrlSuggestion.TYPE_SEARCH: + iconStringId = R.string.fa_search; + fullTitle = String.format(context.getString(R.string.search_url_title), item.getText()); + desc = String.format(context.getString(R.string.search_url_desc), item.getText()); + break; + case UrlSuggestion.TYPE_FILE: + default: + iconStringId = R.string.fa_file; + fullTitle = item.getTitle(); + desc = item.getClaim() != null ? item.getClaim().getTitle() : + String.format(context.getString(R.string.view_file_url_desc), item.getText()); + break; + } + + vh.iconView.setText(iconStringId); + vh.titleView.setText(fullTitle); + vh.descView.setText(desc); + + vh.itemView.setOnClickListener(view -> { + if (listener != null) { + listener.onUrlSuggestionClicked(item); + } + }); + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected SolidIconView iconView; + protected TextView titleView; + protected TextView descView; + public ViewHolder(View v) { + super(v); + iconView = v.findViewById(R.id.url_suggestion_icon); + titleView = v.findViewById(R.id.url_suggestion_title); + descView = v.findViewById(R.id.url_suggestion_description); + } + } + + public interface UrlSuggestionClickListener { + void onUrlSuggestionClicked(UrlSuggestion urlSuggestion); + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/VerificationPagerAdapter.java b/app/src/main/java/io/lbry/browser/adapter/VerificationPagerAdapter.java new file mode 100644 index 00000000..3591654f --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/VerificationPagerAdapter.java @@ -0,0 +1,52 @@ +package io.lbry.browser.adapter; + +import androidx.fragment.app.Fragment; +import androidx.fragment.app.FragmentActivity; +import androidx.viewpager2.adapter.FragmentStateAdapter; + +import io.lbry.browser.listener.SignInListener; +import io.lbry.browser.listener.WalletSyncListener; +import io.lbry.browser.ui.verification.EmailVerificationFragment; +import io.lbry.browser.ui.verification.WalletVerificationFragment; +import lombok.SneakyThrows; + +/** + * 4 fragments + * - Email collect / verify (sign in) + * - Phone number collect / verify (rewards) + * - Wallet password + * - Manual verification page + */ +public class VerificationPagerAdapter extends FragmentStateAdapter { + private FragmentActivity activity; + + public VerificationPagerAdapter(FragmentActivity activity) { + super(activity); + this.activity = activity; + } + + @SneakyThrows + @Override + public Fragment createFragment(int position) { + switch (position) { + case 0: + default: + EmailVerificationFragment evFragment = EmailVerificationFragment.class.newInstance(); + if (activity instanceof SignInListener) { + evFragment.setListener((SignInListener) activity); + } + return evFragment; + case 1: + WalletVerificationFragment wvFragment = WalletVerificationFragment.class.newInstance(); + if (activity instanceof WalletSyncListener) { + wvFragment.setListener((WalletSyncListener) activity); + } + return wvFragment; + } + } + + @Override + public int getItemCount() { + return 2; + } +} diff --git a/app/src/main/java/io/lbry/browser/data/DatabaseHelper.java b/app/src/main/java/io/lbry/browser/data/DatabaseHelper.java new file mode 100644 index 00000000..cca1c4d0 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/data/DatabaseHelper.java @@ -0,0 +1,74 @@ +package io.lbry.browser.data; + +import android.content.Context; +import android.database.Cursor; +import android.database.sqlite.SQLiteDatabase; +import android.database.sqlite.SQLiteOpenHelper; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.utils.Helper; + +public class DatabaseHelper extends SQLiteOpenHelper { + public static final int DATABASE_VERSION = 1; + public static final String DATABASE_NAME = "LbryApp.db"; + + private static final String[] SQL_CREATE_TABLES = { + // local subscription store + "CREATE TABLE subscriptions (url TEXT PRIMARY KEY NOT NULL, channel_name TEXT NOT NULL)", + + // local claim cache store for quick load / refresh + + }; + private static final String[] SQL_CREATE_INDEXES = { + "CREATE UNIQUE INDEX idx_subscription_url ON subscriptions (url)" + }; + + private static final String SQL_INSERT_SUBSCRIPTION = "REPLACE INTO subscriptions (channel_name, url) VALUES (?, ?)"; + private static final String SQL_DELETE_SUBSCRIPTION = "DELETE FROM subscriptions WHERE url = ?"; + private static final String SQL_GET_SUBSCRIPTIONS = "SELECT channel_name, url FROM subscriptions"; + + public DatabaseHelper(Context context) { + super(context, String.format("%s/%s", context.getFilesDir().getAbsolutePath(), DATABASE_NAME), null, DATABASE_VERSION); + } + public void onCreate(SQLiteDatabase db) { + for (String sql : SQL_CREATE_TABLES) { + db.execSQL(sql); + } + for (String sql : SQL_CREATE_INDEXES) { + db.execSQL(sql); + } + } + public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { + + } + public void onDowngrade(SQLiteDatabase db, int oldVersion, int newVersion) { + + } + + public static void createOrUpdateSubscription(Subscription subscription, SQLiteDatabase db) { + db.execSQL(SQL_INSERT_SUBSCRIPTION, new Object[] { subscription.getChannelName(), subscription.getUrl() }); + } + public static void deleteSubscription(Subscription subscription, SQLiteDatabase db) { + db.execSQL(SQL_DELETE_SUBSCRIPTION, new Object[] { subscription.getUrl() }); + } + public static List getSubscriptions(SQLiteDatabase db) { + List subscriptions = new ArrayList<>(); + Cursor cursor = null; + try { + cursor = db.rawQuery(SQL_GET_SUBSCRIPTIONS, null); + while (cursor.moveToNext()) { + Subscription subscription = new Subscription(); + subscription.setChannelName(cursor.getString(0)); + subscription.setUrl(cursor.getString(1)); + subscriptions.add(subscription); + } + } finally { + Helper.closeCursor(cursor); + } + return subscriptions; + } + +} diff --git a/app/src/main/java/io/lbry/browser/dialog/ContentFromDialogFragment.java b/app/src/main/java/io/lbry/browser/dialog/ContentFromDialogFragment.java new file mode 100644 index 00000000..d81102b7 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/dialog/ContentFromDialogFragment.java @@ -0,0 +1,112 @@ +package io.lbry.browser.dialog; + +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; + +import com.google.android.material.bottomsheet.BottomSheetDialogFragment; + +import io.lbry.browser.R; +import lombok.Setter; + +public class ContentFromDialogFragment extends BottomSheetDialogFragment { + public static final String TAG = "ContentFromDialog"; + public static final int ITEM_FROM_PAST_24_HOURS = 1; + public static final int ITEM_FROM_PAST_WEEK = 2; + public static final int ITEM_FROM_PAST_MONTH = 3; + public static final int ITEM_FROM_PAST_YEAR = 4; + public static final int ITEM_FROM_ALL_TIME = 5; + + @Setter + private ContentFromListener contentFromListener; + private int currentFromItem; + + public static ContentFromDialogFragment newInstance() { + return new ContentFromDialogFragment(); + } + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View view = inflater.inflate(R.layout.dialog_content_from, container,false); + + ContentFromItemClickListener clickListener = new ContentFromItemClickListener(this, contentFromListener); + view.findViewById(R.id.content_from_past_24_hours_item).setOnClickListener(clickListener); + view.findViewById(R.id.content_from_past_week_item).setOnClickListener(clickListener); + view.findViewById(R.id.content_from_past_month_item).setOnClickListener(clickListener); + view.findViewById(R.id.content_from_past_year_item).setOnClickListener(clickListener); + view.findViewById(R.id.content_from_all_time_item).setOnClickListener(clickListener); + checkSelectedFromItem(currentFromItem, view); + + return view; + } + + public static void checkSelectedFromItem(int fromItem, View parent) { + int checkViewId = -1; + switch (fromItem) { + case ITEM_FROM_PAST_24_HOURS: checkViewId = R.id.content_from_past_24_hours_item_selected; break; + case ITEM_FROM_PAST_WEEK: checkViewId = R.id.content_from_past_week_item_selected; break; + case ITEM_FROM_PAST_MONTH: checkViewId = R.id.content_from_past_month_item_selected; break; + case ITEM_FROM_PAST_YEAR: checkViewId = R.id.content_from_past_year_item_selected; break; + case ITEM_FROM_ALL_TIME: checkViewId = R.id.content_from_all_time_item_selected; break; + } + if (parent != null && checkViewId > -1) { + parent.findViewById(checkViewId).setVisibility(View.VISIBLE); + } + } + + public void setCurrentFromItem(int fromItem) { + this.currentFromItem = fromItem; + } + + private static class ContentFromItemClickListener implements View.OnClickListener { + + private final int[] checkViewIds = { + R.id.content_from_past_24_hours_item, + R.id.content_from_past_week_item, + R.id.content_from_past_month_item, + R.id.content_from_past_year_item, + R.id.content_from_all_time_item + }; + private BottomSheetDialogFragment dialog; + private ContentFromListener listener; + + public ContentFromItemClickListener(BottomSheetDialogFragment dialog, ContentFromListener listener) { + this.dialog = dialog; + this.listener = listener; + } + + public void onClick(View view) { + int currentFromItem = -1; + + if (dialog != null) { + View dialogView = dialog.getView(); + if (dialogView != null) { + for (int id : checkViewIds) { + dialogView.findViewById(id).setVisibility(View.GONE); + } + } + } + + switch (view.getId()) { + case R.id.content_from_past_24_hours_item: currentFromItem = ITEM_FROM_PAST_24_HOURS; break; + case R.id.content_from_past_week_item: currentFromItem = ITEM_FROM_PAST_WEEK; break; + case R.id.content_from_past_month_item: currentFromItem = ITEM_FROM_PAST_MONTH; break; + case R.id.content_from_past_year_item: currentFromItem = ITEM_FROM_PAST_YEAR; break; + case R.id.content_from_all_time_item: currentFromItem = ITEM_FROM_ALL_TIME; break; + } + + checkSelectedFromItem(currentFromItem, view); + if (listener != null) { + listener.onContentFromItemSelected(currentFromItem); + } + + if (dialog != null) { + dialog.dismiss(); + } + } + } + + public interface ContentFromListener { + void onContentFromItemSelected(int contentFromItem); + } +} diff --git a/app/src/main/java/io/lbry/browser/dialog/ContentSortDialogFragment.java b/app/src/main/java/io/lbry/browser/dialog/ContentSortDialogFragment.java new file mode 100644 index 00000000..06ac5e57 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/dialog/ContentSortDialogFragment.java @@ -0,0 +1,100 @@ +package io.lbry.browser.dialog; + +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; + +import com.google.android.material.bottomsheet.BottomSheetDialogFragment; + +import io.lbry.browser.R; +import lombok.Setter; + +public class ContentSortDialogFragment extends BottomSheetDialogFragment { + public static final String TAG = "ContentSortDialog"; + public static final int ITEM_SORT_BY_TRENDING = 1; + public static final int ITEM_SORT_BY_NEW = 2; + public static final int ITEM_SORT_BY_TOP = 3; + + @Setter + private SortByListener sortByListener; + private int currentSortByItem; + + public static ContentSortDialogFragment newInstance() { + return new ContentSortDialogFragment(); + } + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View view = inflater.inflate(R.layout.dialog_content_sort, container,false); + + SortByItemClickListener clickListener = new SortByItemClickListener(this, sortByListener); + view.findViewById(R.id.sort_by_trending_item).setOnClickListener(clickListener); + view.findViewById(R.id.sort_by_new_item).setOnClickListener(clickListener); + view.findViewById(R.id.sort_by_top_item).setOnClickListener(clickListener); + checkSelectedSortByItem(currentSortByItem, view); + + return view; + } + + public static void checkSelectedSortByItem(int sortByItem, View parent) { + int checkViewId = -1; + switch (sortByItem) { + case ITEM_SORT_BY_TRENDING: checkViewId = R.id.sort_by_trending_item_selected; break; + case ITEM_SORT_BY_NEW: checkViewId = R.id.sort_by_new_item_selected; break; + case ITEM_SORT_BY_TOP: checkViewId = R.id.sort_by_top_item_selected; break; + } + if (parent != null && checkViewId > -1) { + parent.findViewById(checkViewId).setVisibility(View.VISIBLE); + } + } + + public void setCurrentSortByItem(int sortByItem) { + this.currentSortByItem = sortByItem; + } + + private static class SortByItemClickListener implements View.OnClickListener { + + private final int[] checkViewIds = { + R.id.sort_by_trending_item_selected, R.id.sort_by_new_item_selected, R.id.sort_by_top_item_selected + }; + private BottomSheetDialogFragment dialog; + private SortByListener listener; + + public SortByItemClickListener(BottomSheetDialogFragment dialog, SortByListener listener) { + this.dialog = dialog; + this.listener = listener; + } + + public void onClick(View view) { + int selectedSortByItem = -1; + + if (dialog != null) { + View dialogView = dialog.getView(); + if (dialogView != null) { + for (int id : checkViewIds) { + dialogView.findViewById(id).setVisibility(View.GONE); + } + } + } + + switch (view.getId()) { + case R.id.sort_by_trending_item: selectedSortByItem = ITEM_SORT_BY_TRENDING; break; + case R.id.sort_by_new_item: selectedSortByItem = ITEM_SORT_BY_NEW; break; + case R.id.sort_by_top_item: selectedSortByItem = ITEM_SORT_BY_TOP; break; + } + + checkSelectedSortByItem(selectedSortByItem, view); + if (listener != null) { + listener.onSortByItemSelected(selectedSortByItem); + } + + if (dialog != null) { + dialog.dismiss(); + } + } + } + + public interface SortByListener { + void onSortByItemSelected(int sortBy); + } +} diff --git a/app/src/main/java/io/lbry/browser/exceptions/ApiCallException.java b/app/src/main/java/io/lbry/browser/exceptions/ApiCallException.java new file mode 100644 index 00000000..652576c6 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/exceptions/ApiCallException.java @@ -0,0 +1,13 @@ +package io.lbry.browser.exceptions; + +public class ApiCallException extends Exception { + public ApiCallException() { + super(); + } + public ApiCallException(String message) { + super(message); + } + public ApiCallException(String message, Throwable cause) { + super(message, cause); + } +} diff --git a/app/src/main/java/io/lbry/browser/exceptions/LbryRequestException.java b/app/src/main/java/io/lbry/browser/exceptions/LbryRequestException.java new file mode 100644 index 00000000..a71a09e6 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/exceptions/LbryRequestException.java @@ -0,0 +1,13 @@ +package io.lbry.browser.exceptions; + +public class LbryRequestException extends Exception { + public LbryRequestException() { + super(); + } + public LbryRequestException(String message) { + super(message); + } + public LbryRequestException(String message, Throwable cause) { + super(message, cause); + } +} diff --git a/app/src/main/java/io/lbry/browser/exceptions/LbryResponseException.java b/app/src/main/java/io/lbry/browser/exceptions/LbryResponseException.java new file mode 100644 index 00000000..63a67e79 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/exceptions/LbryResponseException.java @@ -0,0 +1,13 @@ +package io.lbry.browser.exceptions; + +public class LbryResponseException extends Exception { + public LbryResponseException() { + super(); + } + public LbryResponseException(String message) { + super(message); + } + public LbryResponseException(String message, Throwable cause) { + super(message, cause); + } +} diff --git a/app/src/main/java/io/lbry/browser/exceptions/LbryUriException.java b/app/src/main/java/io/lbry/browser/exceptions/LbryUriException.java new file mode 100644 index 00000000..34fbcb22 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/exceptions/LbryUriException.java @@ -0,0 +1,13 @@ +package io.lbry.browser.exceptions; + +public class LbryUriException extends Exception { + public LbryUriException() { + super(); + } + public LbryUriException(String message) { + super(message); + } + public LbryUriException(String message, Throwable cause) { + super(message, cause); + } +} diff --git a/app/src/main/java/io/lbry/browser/exceptions/LbryioRequestException.java b/app/src/main/java/io/lbry/browser/exceptions/LbryioRequestException.java new file mode 100644 index 00000000..bb88a0e7 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/exceptions/LbryioRequestException.java @@ -0,0 +1,13 @@ +package io.lbry.browser.exceptions; + +public class LbryioRequestException extends Exception { + public LbryioRequestException() { + super(); + } + public LbryioRequestException(String message) { + super(message); + } + public LbryioRequestException(String message, Throwable cause) { + super(message, cause); + } +} diff --git a/app/src/main/java/io/lbry/browser/exceptions/LbryioResponseException.java b/app/src/main/java/io/lbry/browser/exceptions/LbryioResponseException.java new file mode 100644 index 00000000..a2a2b7b4 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/exceptions/LbryioResponseException.java @@ -0,0 +1,25 @@ +package io.lbry.browser.exceptions; + +import lombok.Getter; + +public class LbryioResponseException extends Exception { + @Getter + private int statusCode; + public LbryioResponseException() { + super(); + } + public LbryioResponseException(String message) { + super(message); + } + public LbryioResponseException(String message, Throwable cause) { + super(message, cause); + } + public LbryioResponseException(String message, int statusCode) { + super(message); + this.statusCode = statusCode; + } + public LbryioResponseException(String message, Throwable cause, int statusCode) { + super(message, cause); + this.statusCode = statusCode; + } +} diff --git a/app/src/main/java/io/lbry/browser/exceptions/WalletException.java b/app/src/main/java/io/lbry/browser/exceptions/WalletException.java new file mode 100644 index 00000000..b6dcbc40 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/exceptions/WalletException.java @@ -0,0 +1,13 @@ +package io.lbry.browser.exceptions; + +public class WalletException extends Exception { + public WalletException() { + super(); + } + public WalletException(String message) { + super(message); + } + public WalletException(String message, Throwable cause) { + super(message, cause); + } +} diff --git a/app/src/main/java/io/lbry/browser/listener/ChannelItemSelectionListener.java b/app/src/main/java/io/lbry/browser/listener/ChannelItemSelectionListener.java new file mode 100644 index 00000000..39e06c14 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/ChannelItemSelectionListener.java @@ -0,0 +1,9 @@ +package io.lbry.browser.listener; + +import io.lbry.browser.model.Claim; + +public interface ChannelItemSelectionListener { + void onChannelItemSelected(Claim claim); + void onChannelItemDeselected(Claim claim); + void onChannelSelectionCleared(); +} diff --git a/app/src/main/java/io/lbry/browser/listener/SdkStatusListener.java b/app/src/main/java/io/lbry/browser/listener/SdkStatusListener.java new file mode 100644 index 00000000..866ff382 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/SdkStatusListener.java @@ -0,0 +1,5 @@ +package io.lbry.browser.listener; + +public interface SdkStatusListener { + void onSdkReady(); +} diff --git a/app/src/main/java/io/lbry/browser/listener/SignInListener.java b/app/src/main/java/io/lbry/browser/listener/SignInListener.java new file mode 100644 index 00000000..0217a91f --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/SignInListener.java @@ -0,0 +1,7 @@ +package io.lbry.browser.listener; + +public interface SignInListener { + void onEmailAdded(String email); + void onEmailEdit(); + void onEmailVerified(); +} diff --git a/app/src/main/java/io/lbry/browser/listener/WalletBalanceListener.java b/app/src/main/java/io/lbry/browser/listener/WalletBalanceListener.java new file mode 100644 index 00000000..5e0ede06 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/WalletBalanceListener.java @@ -0,0 +1,7 @@ +package io.lbry.browser.listener; + +import io.lbry.browser.model.WalletBalance; + +public interface WalletBalanceListener { + void onWalletBalanceUpdated(WalletBalance walletBalance); +} diff --git a/app/src/main/java/io/lbry/browser/listener/WalletSyncListener.java b/app/src/main/java/io/lbry/browser/listener/WalletSyncListener.java new file mode 100644 index 00000000..fcee590b --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/WalletSyncListener.java @@ -0,0 +1,8 @@ +package io.lbry.browser.listener; + +public interface WalletSyncListener { + void onWalletSyncProcessing(); + void onWalletSyncWaitingForInput(); + void onWalletSyncEnabled(); + void onWalletSyncFailed(Exception error); +} diff --git a/app/src/main/java/io/lbry/browser/model/Claim.java b/app/src/main/java/io/lbry/browser/model/Claim.java new file mode 100644 index 00000000..549a656d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/Claim.java @@ -0,0 +1,338 @@ +package io.lbry.browser.model; + +import com.google.gson.FieldNamingPolicy; +import com.google.gson.Gson; +import com.google.gson.GsonBuilder; +import com.google.gson.reflect.TypeToken; + +import org.json.JSONException; +import org.json.JSONObject; + +import java.lang.reflect.Type; +import java.text.ParseException; +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.stream.Stream; + +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryUri; +import lombok.Data; +import lombok.EqualsAndHashCode; +import lombok.Getter; +import lombok.Setter; + +@Data +@EqualsAndHashCode(onlyExplicitlyIncluded = true) +public class Claim { + public static final String CLAIM_TYPE_CLAIM = "claim"; + public static final String CLAIM_TYPE_UPDATE = "update"; + public static final String CLAIM_TYPE_SUPPORT = "support"; + + public static final String TYPE_STREAM = "stream"; + public static final String TYPE_CHANNEL = "channel"; + + public static final String STREAM_TYPE_AUDIO = "audio"; + public static final String STREAM_TYPE_IMAGE = "image"; + public static final String STREAM_TYPE_VIDEO = "video"; + public static final String STREAM_TYPE_SOFTWARE = "software"; + + public static final String ORDER_BY_EFFECTIVE_AMOUNT = "effective_amount"; + public static final String ORDER_BY_RELEASE_TIME = "release_time"; + public static final String ORDER_BY_TRENDING_GROUP = "trending_group"; + public static final String ORDER_BY_TRENDING_MIXED = "trending_mixed"; + + public static final List CLAIM_TYPES = Arrays.asList(TYPE_CHANNEL, TYPE_STREAM); + public static final List STREAM_TYPES = Arrays.asList( + STREAM_TYPE_AUDIO, STREAM_TYPE_IMAGE, STREAM_TYPE_SOFTWARE, STREAM_TYPE_VIDEO + ); + + public static final String RELEASE_TIME_DATE_FORMAT = "yyyy-MM-dd'T'HH:mm:ss'Z'"; + + @EqualsAndHashCode.Include + private boolean placeholder; + private boolean featured; + private boolean unresolved; // used for featured + private String address; + private String amount; + private String canonicalUrl; + @EqualsAndHashCode.Include + private String claimId; + private int claimSequence; + private String claimOp; + private long confirmations; + private boolean decodedClaim; + private long timestamp; + private long height; + private boolean isMine; + private String name; + private String normalizedName; + private int nout; + private String permanentUrl; + private String shortUrl; + private String txid; + private String type; // claim | update | support + private String valueType; // stream | channel + private Claim signingChannel; + private String repostChannelUrl; + private boolean isChannelSignatureValid; + private GenericMetadata value; + private File file; // associated file if it exists + + public String getThumbnailUrl() { + if (value != null && value.getThumbnail() != null) { + return value.getThumbnail().getUrl(); + } + return null; + } + + public String getCoverUrl() { + if (TYPE_CHANNEL.equals(valueType) && value != null && value instanceof ChannelMetadata && ((ChannelMetadata) value).getCover() != null) { + return ((ChannelMetadata) value).getCover().getUrl(); + } + return null; + } + + public String getFirstCharacter() { + if (name != null) { + return name.startsWith("@") ? name.substring(1) : name; + } + return ""; + } + + public String getFirstTag() { + if (value != null && value.tags != null && value.tags.size() > 0) { + return value.tags.get(0); + } + return null; + } + + public String getDescription() { + return (value != null) ? value.getDescription() : null; + } + + public String getPublisherName() { + if (signingChannel != null) { + return signingChannel.getName(); + } + return "Anonymous"; + } + + public String getPublisherTitle() { + if (signingChannel != null) { + return Helper.isNullOrEmpty(signingChannel.getTitle()) ? signingChannel.getName() : signingChannel.getTitle(); + } + return "Anonymous"; + } + + public List getTags() { + return (value != null && value.getTags() != null) ? new ArrayList<>(value.getTags()) : new ArrayList<>(); + } + + public List getTagObjects() { + List tags = new ArrayList<>(); + if (value != null && value.getTags() != null) { + for (String value : value.getTags()) { + tags.add(new Tag(value)); + } + } + return tags; + } + + public String getTitle() { + return (value != null) ? value.getTitle() : null; + } + + public long getDuration() { + if (value instanceof StreamMetadata) { + StreamMetadata metadata = (StreamMetadata) value; + if (STREAM_TYPE_VIDEO.equalsIgnoreCase(metadata.getStreamType()) && metadata.getVideo() != null) { + return metadata.getVideo().getDuration(); + } else if (STREAM_TYPE_AUDIO.equalsIgnoreCase(metadata.getStreamType()) && metadata.getAudio() != null) { + return metadata.getAudio().getDuration(); + } + } + + return 0; + } + + public static Claim fromJSONObject(JSONObject claimObject) { + String claimJson = claimObject.toString(); + Type type = new TypeToken(){}.getType(); + Type streamMetadataType = new TypeToken(){}.getType(); + Type channelMetadataType = new TypeToken(){}.getType(); + + Gson gson = new GsonBuilder().setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES).create(); + Claim claim = gson.fromJson(claimJson, type); + + // Specific value parsing + try { + JSONObject value = claimObject.getJSONObject("value"); + String valueType = claim.getValueType(); + if (value != null) { + String valueJson = value.toString(); + if (TYPE_STREAM.equalsIgnoreCase(valueType)) { + claim.setValue(gson.fromJson(valueJson, streamMetadataType)); + } else if (TYPE_CHANNEL.equalsIgnoreCase(valueType)) { + claim.setValue(gson.fromJson(valueJson, channelMetadataType)); + } + } + } catch (JSONException ex) { + // pass + } + + return claim; + } + + public static Claim fromSearchJSONObject(JSONObject searchResultObject) { + Claim claim = new Claim(); + LbryUri claimUri = new LbryUri(); + try { + claim.setClaimId(searchResultObject.getString("claimId")); + claim.setName(searchResultObject.getString("name")); + + if (claim.getName().startsWith("@")) { + claimUri.setChannelClaimId(claim.getClaimId()); + claimUri.setChannelName(claim.getName()); + claim.setValueType(TYPE_CHANNEL); + } else { + claimUri.setStreamClaimId(claim.getClaimId()); + claimUri.setStreamName(claim.getName()); + claim.setValueType(TYPE_STREAM); + } + + int duration = searchResultObject.isNull("duration") ? 0 : searchResultObject.getInt("duration"); + long feeAmount = searchResultObject.isNull("fee") ? 0 : searchResultObject.getLong("fee"); + String releaseTimeString = !searchResultObject.isNull("release_time") ? searchResultObject.getString("release_time") : null; + long releaseTime = 0; + try { + releaseTime = Double.valueOf(new SimpleDateFormat(RELEASE_TIME_DATE_FORMAT).parse(releaseTimeString).getTime() / 1000.0).longValue(); + } catch (ParseException ex) { + // pass + } + + GenericMetadata metadata = (duration > 0 || releaseTime > 0 || feeAmount > 0) ? new StreamMetadata() : new GenericMetadata(); + metadata.setTitle(searchResultObject.getString("title")); + if (metadata instanceof StreamMetadata) { + StreamInfo streamInfo = new StreamInfo(); + if (duration > 0) { + // assume stream type video + ((StreamMetadata) metadata).setStreamType(STREAM_TYPE_VIDEO); + streamInfo.setDuration(duration); + } + + Fee fee = null; + if (feeAmount > 0) { + fee = new Fee(); + fee.setAmount(String.valueOf(feeAmount)); + } + + ((StreamMetadata) metadata).setFee(fee); + ((StreamMetadata) metadata).setVideo(streamInfo); + ((StreamMetadata) metadata).setReleaseTime(releaseTime); + } + claim.setValue(metadata); + + if (!searchResultObject.isNull("thumbnail_url")) { + Resource thumbnail = new Resource(); + thumbnail.setUrl(searchResultObject.getString("thumbnail_url")); + claim.getValue().setThumbnail(thumbnail); + } + + if (!searchResultObject.isNull("channel_claim_id") && !searchResultObject.isNull("channel")) { + Claim signingChannel = new Claim(); + signingChannel.setClaimId(searchResultObject.getString("channel_claim_id")); + signingChannel.setName(searchResultObject.getString("channel")); + + claimUri.setChannelClaimId(signingChannel.getClaimId()); + claimUri.setChannelName(signingChannel.getName()); + + claim.setSigningChannel(signingChannel); + } + } catch (JSONException ex) { + // pass + } + + claim.setPermanentUrl(claimUri.toString()); + + return claim; + } + + @Data + public static class Meta { + private long activationHeight; + private int claimsInChannel; + private int creationHeight; + private int creationTimestamp; + private String effectiveAmount; + private long expirationHeight; + private boolean isControlling; + private String supportAmount; + private int reposted; + private double trendingGlobal; + private double trendingGroup; + private double trendingLocal; + private double trendingMixed; + } + + @Data + public static class GenericMetadata { + private String title; + private String description; + private Resource thumbnail; + private List languages; + private List tags; + private List locations; + } + + @Data + @EqualsAndHashCode(callSuper = true) + public static class ChannelMetadata extends GenericMetadata { + private String publicKey; + private String publicKeyId; + private Resource cover; + private String email; + private String websiteUrl; + private List featured; + } + + @Data + @EqualsAndHashCode(callSuper = true) + public static class StreamMetadata extends GenericMetadata { + private String license; + private String licenseUrl; + private long releaseTime; + private String author; + private Fee fee; + private String streamType; // video | audio | image | software + private Source source; + private StreamInfo video; + private StreamInfo audio; + private StreamInfo image; + private StreamInfo software; + + @Data + public static class Source { + private String sdHash; + private String mediaType; + private String hash; + private String name; + private long size; + } + } + + // only support "url" for now + @Data + public static class Resource { + private String url; + } + + @Data + public static class StreamInfo { + private long duration; // video / audio + private long height; // video / image + private long width; // video / image + private String os; // software + } +} diff --git a/app/src/main/java/io/lbry/browser/model/ClaimCacheKey.java b/app/src/main/java/io/lbry/browser/model/ClaimCacheKey.java new file mode 100644 index 00000000..da9792be --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/ClaimCacheKey.java @@ -0,0 +1,86 @@ +package io.lbry.browser.model; + +import androidx.annotation.Nullable; + +import io.lbry.browser.utils.Helper; +import lombok.Getter; +import lombok.Setter; + +/** + * Class to represent a key to check equality with another object + */ +public class ClaimCacheKey { + @Getter + @Setter + private String claimId; + @Getter + @Setter + private String canonicalUrl; + @Getter + @Setter + private String permanentUrl; + @Getter + @Setter + private String shortUrl; + @Getter + @Setter + private String vanityUrl; + + public static ClaimCacheKey fromClaim(Claim claim) { + ClaimCacheKey key = new ClaimCacheKey(); + key.setClaimId(claim.getClaimId()); + key.setCanonicalUrl(claim.getCanonicalUrl()); + key.setPermanentUrl(claim.getPermanentUrl()); + key.setShortUrl(claim.getShortUrl()); + + return key; + } + + @Override + public boolean equals(@Nullable Object obj) { + if (obj == null || !(obj instanceof ClaimCacheKey)) { + return false; + } + + ClaimCacheKey key = (ClaimCacheKey) obj; + if (!Helper.isNullOrEmpty(claimId) && !Helper.isNullOrEmpty(key.getClaimId())) { + return claimId.equalsIgnoreCase(key.getClaimId()); + } + if (!Helper.isNullOrEmpty(permanentUrl) && !Helper.isNullOrEmpty(key.getPermanentUrl())) { + return permanentUrl.equalsIgnoreCase(key.getPermanentUrl()); + } + if (!Helper.isNullOrEmpty(canonicalUrl) && !Helper.isNullOrEmpty(key.getCanonicalUrl())) { + return canonicalUrl.equalsIgnoreCase(key.getCanonicalUrl()); + } + if (!Helper.isNullOrEmpty(shortUrl) && !Helper.isNullOrEmpty(key.getShortUrl())) { + return shortUrl.equalsIgnoreCase(key.getShortUrl()); + } + if (!Helper.isNullOrEmpty(vanityUrl) && !Helper.isNullOrEmpty(key.getVanityUrl())) { + return vanityUrl.equalsIgnoreCase(key.getVanityUrl()); + } + + return false; + } + + @Override + public int hashCode() { + if (!Helper.isNullOrEmpty(claimId)) { + return claimId.hashCode(); + } + if (!Helper.isNullOrEmpty(permanentUrl)) { + return permanentUrl.hashCode(); + } + if (!Helper.isNullOrEmpty(canonicalUrl)) { + return canonicalUrl.hashCode(); + } + if (!Helper.isNullOrEmpty(shortUrl)) { + return shortUrl.hashCode(); + } + if (!Helper.isNullOrEmpty(vanityUrl)) { + return vanityUrl.hashCode(); + } + + return super.hashCode(); + } +} + diff --git a/app/src/main/java/io/lbry/browser/model/Fee.java b/app/src/main/java/io/lbry/browser/model/Fee.java new file mode 100644 index 00000000..12fbc61d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/Fee.java @@ -0,0 +1,10 @@ +package io.lbry.browser.model; + +import lombok.Data; + +@Data +public class Fee { + private String amount; + private String currency; + private String address; +} diff --git a/app/src/main/java/io/lbry/browser/model/File.java b/app/src/main/java/io/lbry/browser/model/File.java new file mode 100644 index 00000000..ca68e86a --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/File.java @@ -0,0 +1,54 @@ +package io.lbry.browser.model; + +import com.google.gson.FieldNamingPolicy; +import com.google.gson.Gson; +import com.google.gson.GsonBuilder; +import com.google.gson.reflect.TypeToken; + +import org.json.JSONObject; + +import java.lang.reflect.Type; + +import lombok.Data; + +@Data +public class File { + private Claim.StreamMetadata metadata; + private long addedOn; + private int blobsCompleted; + private int blobsInStream; + private int blobsRemaining; + private String channelClaimId; + private String channelName; + private String claimId; + private String claimName; + private boolean completed; + private String downloadDirectory; + private String downloadPath; + private String fileName; + private String key; + private String mimeType; + private int nout; + private String outpoint; + private int pointsPaid; + private String protobuf; + private String sdHash; + private String status; + private boolean stopped; + private String streamHash; + private String streamName; + private String streamingUrl; + private String suggestedFileName; + private long totalBytes; + private long totalBytesLowerBound; + private String txid; + private long writtenBytes; + + public static File fromJSONObject(JSONObject fileObject) { + String fileJson = fileObject.toString(); + Type type = new TypeToken(){}.getType(); + Gson gson = new GsonBuilder().setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES).create(); + File file = gson.fromJson(fileJson, type); + return file; + } +} diff --git a/app/src/main/java/io/lbry/browser/model/Location.java b/app/src/main/java/io/lbry/browser/model/Location.java new file mode 100644 index 00000000..e1673424 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/Location.java @@ -0,0 +1,13 @@ +package io.lbry.browser.model; + +import lombok.Data; + +@Data +public class Location { + private double latitude; + private double longitude; + private String country; + private String state; + private String city; + private String code; +} diff --git a/app/src/main/java/io/lbry/browser/model/NavMenuItem.java b/app/src/main/java/io/lbry/browser/model/NavMenuItem.java new file mode 100644 index 00000000..02e4dd14 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/NavMenuItem.java @@ -0,0 +1,68 @@ +package io.lbry.browser.model; + +import android.content.Context; + +import androidx.core.content.res.ResourcesCompat; + +import java.util.ArrayList; +import java.util.List; + +import lombok.Data; + +@Data +public class NavMenuItem { + public static final int ID_GROUP_FIND_CONTENT = 100; + public static final int ID_GROUP_YOUR_CONTENT = 200; + public static final int ID_GROUP_WALLET = 300; + public static final int ID_GROUP_OTHER = 400; + + // Find Content + public static final int ID_ITEM_FOLLOWING = 101; + public static final int ID_ITEM_EDITORS_CHOICE = 102; + public static final int ID_ITEM_YOUR_TAGS = 103; + public static final int ID_ITEM_ALL_CONTENT = 104; + + // Your Content + public static final int ID_ITEM_CHANNELS = 201; + public static final int ID_ITEM_LIBRARY = 202; + public static final int ID_ITEM_PUBLISHES = 203; + public static final int ID_ITEM_NEW_PUBLISH = 204; + + // Wallet + public static final int ID_ITEM_WALLET = 301; + public static final int ID_ITEM_REWARDS = 302; + public static final int ID_ITEM_INVITES = 303; + + // Other + public static final int ID_ITEM_SETTINGS = 401; + public static final int ID_ITEM_ABOUT = 402; + + private Context context; + private int id; + private boolean group; + private int icon; + private String title; + private String name; // same as title, but only as en lang for events + private List items; + + public NavMenuItem(int id, int titleResourceId, boolean group, Context context) { + this.context = context; + this.id = id; + this.group = group; + + if (titleResourceId > 0) { + this.title = context.getString(titleResourceId); + } + if (group) { + this.items = new ArrayList<>(); + } + } + + public NavMenuItem(int id, int iconStringId, int titleResourceId, String name, Context context) { + this.context = context; + this.id = id; + this.icon = iconStringId; + this.title = context.getString(titleResourceId); + this.name = name; + } +} diff --git a/app/src/main/java/io/lbry/browser/model/Tag.java b/app/src/main/java/io/lbry/browser/model/Tag.java new file mode 100644 index 00000000..c89d3691 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/Tag.java @@ -0,0 +1,30 @@ +package io.lbry.browser.model; + +import io.lbry.browser.utils.Helper; +import lombok.Getter; +import lombok.Setter; + +public class Tag { + @Getter + @Setter + private String name; + + public Tag(String name) { + this.name = name; + } + + public String getLowercaseName() { + return name.toLowerCase(); + } + + public boolean isMature() { + return Helper.MATURE_TAG_NAMES.contains(name.toLowerCase()); + } + + public boolean equals(Object o) { + return (o instanceof Tag) && ((Tag) o).getName().equalsIgnoreCase(name); + } + public int hashCode() { + return name.hashCode(); + } +} diff --git a/app/src/main/java/io/lbry/browser/model/Transaction.java b/app/src/main/java/io/lbry/browser/model/Transaction.java new file mode 100644 index 00000000..deae9759 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/Transaction.java @@ -0,0 +1,92 @@ +package io.lbry.browser.model; + +import org.json.JSONException; +import org.json.JSONObject; + +import java.math.BigDecimal; +import java.util.Date; + +import io.lbry.browser.R; +import io.lbry.browser.utils.Helper; +import lombok.Data; + +@Data +public class Transaction { + private int confirmations; + private Date txDate; + private String date; + private String claim; + private String txid; + private BigDecimal value; + private BigDecimal fee; + private long timestamp; + private int descriptionStringId; + private TransactionInfo abandonInfo; + private TransactionInfo claimInfo; + private TransactionInfo supportInfo; + + public static Transaction fromJSONObject(JSONObject jsonObject) { + Transaction transaction = new Transaction(); + transaction.setConfirmations(Helper.getJSONInt("confirmations", -1, jsonObject)); + transaction.setDate(Helper.getJSONString("date", null, jsonObject)); + transaction.setTxid(Helper.getJSONString("txid", null, jsonObject)); + transaction.setValue(new BigDecimal(Helper.getJSONString("value", "0", jsonObject))); + transaction.setFee(new BigDecimal(Helper.getJSONString("fee", "0", jsonObject))); + transaction.setTimestamp(Helper.getJSONLong("timestamp", 0, jsonObject) * 1000); + transaction.setTxDate(new Date(transaction.getTimestamp())); + + int descStringId = -1; + TransactionInfo info = null; + try { + if (jsonObject.has("abandon_info")) { + info = TransactionInfo.fromJSONObject(jsonObject.getJSONObject("abandon_info")); + descStringId = R.string.abandon; + transaction.setAbandonInfo(info); + } else if (jsonObject.has("claim_info")) { + info = TransactionInfo.fromJSONObject(jsonObject.getJSONObject("claim_info")); + descStringId = info.getClaimName().startsWith("@") ? R.string.channel : R.string.publish; + transaction.setClaimInfo(info); + } else if (jsonObject.has("support_info")) { + info = TransactionInfo.fromJSONObject(jsonObject.getJSONObject("support_info")); + descStringId = info.isTip() ? R.string.tip : R.string.support; + transaction.setSupportInfo(info); + } + + if (info != null) { + transaction.setClaim(info.getClaimName()); + } + } catch (JSONException ex) { + // pass + } + + if (descStringId == -1) { + descStringId = transaction.getValue().signum() == -1 || transaction.getFee().signum() == -1 ? R.string.spend : R.string.receive; + } + transaction.setDescriptionStringId(descStringId); + + return transaction; + } + + @Data + public static class TransactionInfo { + private String address; + private BigDecimal amount; + private String claimId; + private String claimName; + private boolean isTip; + private int nout; + + public static TransactionInfo fromJSONObject(JSONObject jsonObject) { + TransactionInfo info = new TransactionInfo(); + + info.setAddress(Helper.getJSONString("address", null, jsonObject)); + info.setAmount(new BigDecimal(Helper.getJSONString("amount", "0", jsonObject))); + info.setClaimId(Helper.getJSONString("claim_id", null, jsonObject)); + info.setClaimName(Helper.getJSONString("claim_name", null, jsonObject)); + info.setTip(Helper.getJSONBoolean("is_tip", false, jsonObject)); + info.setNout(Helper.getJSONInt("nout", -1, jsonObject)); + + return info; + } + } +} diff --git a/app/src/main/java/io/lbry/browser/model/UrlSuggestion.java b/app/src/main/java/io/lbry/browser/model/UrlSuggestion.java new file mode 100644 index 00000000..f0685329 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/UrlSuggestion.java @@ -0,0 +1,35 @@ +package io.lbry.browser.model; + +import io.lbry.browser.utils.LbryUri; +import lombok.Data; + +@Data +public class UrlSuggestion { + public static final int TYPE_CHANNEL = 1; + public static final int TYPE_FILE = 2; + public static final int TYPE_SEARCH = 3; + public static final int TYPE_TAG = 4; + + private int type; + private String text; + private LbryUri uri; + private Claim claim; // associated claim if resolved + + public UrlSuggestion(int type, String text) { + this.type = type; + this.text = text; + } + + public String getTitle() { + switch (type) { + case TYPE_CHANNEL: + return String.format("%s - %s", text.startsWith("@") ? text.substring(1) : text, uri.toString()); + case TYPE_FILE: + return String.format("%s - %s", text, uri.toString()); + case TYPE_TAG: + return String.format("%s - #%s", text, text); + } + + return text; + } +} diff --git a/app/src/main/java/io/lbry/browser/model/WalletBalance.java b/app/src/main/java/io/lbry/browser/model/WalletBalance.java new file mode 100644 index 00000000..993a0c31 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/WalletBalance.java @@ -0,0 +1,24 @@ +package io.lbry.browser.model; + +import java.math.BigDecimal; + +import lombok.Data; + +@Data +public class WalletBalance { + private BigDecimal available; + private BigDecimal reserved; + private BigDecimal claims; + private BigDecimal supports; + private BigDecimal tips; + private BigDecimal total; + + public WalletBalance() { + available = new BigDecimal(0); + reserved = new BigDecimal(0); + claims = new BigDecimal(0); + supports = new BigDecimal(0); + tips = new BigDecimal(0); + total = new BigDecimal(0); + } +} diff --git a/app/src/main/java/io/lbry/browser/model/WalletSync.java b/app/src/main/java/io/lbry/browser/model/WalletSync.java new file mode 100644 index 00000000..c9ab5fab --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/WalletSync.java @@ -0,0 +1,16 @@ +package io.lbry.browser.model; + +import lombok.Data; + +@Data +public class WalletSync { + private String hash; + private String data; + private boolean changed; + + public WalletSync(String hash, String data, boolean changed) { + this.hash = hash; + this.data = data; + this.changed = changed; + } +} diff --git a/app/src/main/java/io/lbry/browser/model/lbryinc/Subscription.java b/app/src/main/java/io/lbry/browser/model/lbryinc/Subscription.java new file mode 100644 index 00000000..f2bd817f --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/lbryinc/Subscription.java @@ -0,0 +1,17 @@ +package io.lbry.browser.model.lbryinc; + +import lombok.Data; + +@Data +public class Subscription { + private String channelName; + private String url; + + public Subscription() { + + } + public Subscription(String channelName, String url) { + this.channelName = channelName; + this.url = url; + } +} diff --git a/app/src/main/java/io/lbry/browser/model/lbryinc/User.java b/app/src/main/java/io/lbry/browser/model/lbryinc/User.java new file mode 100644 index 00000000..102ec87e --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/lbryinc/User.java @@ -0,0 +1,29 @@ +package io.lbry.browser.model.lbryinc; + +import java.util.List; + +import lombok.Data; + +@Data +public class User { + private String createdAt; + private String familyName; + private String givenName; + private List groups; + private boolean hasVerifiedEmail; + private long id; + private boolean inviteRewardClaimed; + private String invitedAt; + private long inivtedById; + private int invitesRemaining; + private boolean isEmailEnabled; + private boolean isIdentityVerified; + private boolean isRewardApproved; + private String language; + private long manualApprovalUserId; + private String primaryEmail; + private String rewardStatusChangeTrigger; + private String updatedAt; + private List youtubeChannels; + private List deviceTypes; +} diff --git a/app/src/main/java/io/lbry/browser/reactmodules/BackgroundMediaModule.java b/app/src/main/java/io/lbry/browser/reactmodules/BackgroundMediaModule.java deleted file mode 100644 index 38a67202..00000000 --- a/app/src/main/java/io/lbry/browser/reactmodules/BackgroundMediaModule.java +++ /dev/null @@ -1,84 +0,0 @@ -package io.lbry.browser.reactmodules; - -import android.app.Activity; -import android.app.Notification; -import android.app.NotificationChannel; -import android.app.NotificationManager; -import android.app.PendingIntent; -import android.content.Context; -import android.content.Intent; -import android.content.SharedPreferences; -import android.net.Uri; -import android.os.Build; -import androidx.core.app.NotificationCompat; -import androidx.core.app.NotificationManagerCompat; -import androidx.core.content.ContextCompat; - -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; - -import io.lbry.browser.MainActivity; -import io.lbry.browser.R; -import io.lbry.lbrysdk.LbrynetService; - -public class BackgroundMediaModule extends ReactContextBaseJavaModule { - - public static final int NOTIFICATION_ID = 30; - - public static final String ACTION_PLAY = "io.lbry.browser.ACTION_MEDIA_PLAY"; - - public static final String ACTION_PAUSE = "io.lbry.browser.ACTION_MEDIA_PAUSE"; - - public static final String ACTION_STOP = "io.lbry.browser.ACTION_MEDIA_STOP"; - - private Context context; - - public BackgroundMediaModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - } - - @Override - public String getName() { - return "BackgroundMedia"; - } - - @ReactMethod - public void showPlaybackNotification(String title, String publisher, String uri, boolean paused) { - Intent contextIntent = new Intent(context, MainActivity.class); - PendingIntent pendingIntent = PendingIntent.getActivity(context, 0, contextIntent, PendingIntent.FLAG_UPDATE_CURRENT); - - Intent playIntent = new Intent(); - playIntent.setAction(ACTION_PLAY); - PendingIntent playPendingIntent = PendingIntent.getBroadcast(context, 0, playIntent, 0); - - Intent pauseIntent = new Intent(); - pauseIntent.setAction(ACTION_PAUSE); - PendingIntent pausePendingIntent = PendingIntent.getBroadcast(context, 0, pauseIntent, 0); - - NotificationManagerCompat notificationManager = NotificationManagerCompat.from(context); - NotificationCompat.Builder builder = new NotificationCompat.Builder(context, LbrynetService.NOTIFICATION_CHANNEL_ID); - builder.setColor(ContextCompat.getColor(context, R.color.lbryGreen)) - .setContentIntent(pendingIntent) - .setContentTitle(title) - .setContentText(publisher) - .setGroup(LbrynetService.GROUP_SERVICE) - .setOngoing(!paused) - .setSmallIcon(paused ? android.R.drawable.ic_media_pause : android.R.drawable.ic_media_play) - .setStyle(new androidx.media.app.NotificationCompat.MediaStyle() - .setShowActionsInCompactView(0)) - .addAction(paused ? android.R.drawable.ic_media_play : android.R.drawable.ic_media_pause, - paused ? "Play" : "Pause", - paused ? playPendingIntent : pausePendingIntent) - .build(); - - notificationManager.notify(NOTIFICATION_ID, builder.build()); - } - - @ReactMethod - public void hidePlaybackNotification() { - NotificationManagerCompat notificationManager = NotificationManagerCompat.from(context); - notificationManager.cancel(NOTIFICATION_ID); - } -} diff --git a/app/src/main/java/io/lbry/browser/reactmodules/DaemonServiceControlModule.java b/app/src/main/java/io/lbry/browser/reactmodules/DaemonServiceControlModule.java deleted file mode 100644 index 715414d2..00000000 --- a/app/src/main/java/io/lbry/browser/reactmodules/DaemonServiceControlModule.java +++ /dev/null @@ -1,50 +0,0 @@ -package io.lbry.browser.reactmodules; - -import android.app.Activity; -import android.app.NotificationChannel; -import android.content.Context; -import android.content.pm.ActivityInfo; -import android.content.SharedPreferences; - -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; - -import io.lbry.browser.MainActivity; -import io.lbry.lbrysdk.LbrynetService; -import io.lbry.lbrysdk.ServiceHelper; - -public class DaemonServiceControlModule extends ReactContextBaseJavaModule { - - private Context context; - - public DaemonServiceControlModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - } - - @Override - public String getName() { - return "DaemonServiceControl"; - } - - @ReactMethod - public void startService() { - ServiceHelper.start(context, "", LbrynetService.class, "lbrynetservice"); - } - - @ReactMethod - public void stopService() { - ServiceHelper.stop(context, LbrynetService.class); - } - - @ReactMethod - public void setKeepDaemonRunning(boolean value) { - if (context != null) { - SharedPreferences sp = context.getSharedPreferences(MainActivity.SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - SharedPreferences.Editor editor = sp.edit(); - editor.putBoolean(MainActivity.SETTING_KEEP_DAEMON_RUNNING, value); - editor.commit(); - } - } -} diff --git a/app/src/main/java/io/lbry/browser/reactmodules/FirebaseModule.java b/app/src/main/java/io/lbry/browser/reactmodules/FirebaseModule.java deleted file mode 100644 index 74dfe9cc..00000000 --- a/app/src/main/java/io/lbry/browser/reactmodules/FirebaseModule.java +++ /dev/null @@ -1,131 +0,0 @@ -package io.lbry.browser.reactmodules; - -import android.app.Activity; -import android.content.Context; -import android.os.Bundle; -import android.widget.Toast; - -import com.facebook.react.bridge.Promise; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.bridge.ReadableMap; -import com.google.firebase.analytics.FirebaseAnalytics; -import com.google.android.gms.tasks.OnCompleteListener; -import com.google.android.gms.tasks.Task; -import com.google.firebase.iid.FirebaseInstanceId; -import com.google.firebase.iid.InstanceIdResult; - -import io.lbry.browser.BuildConfig; -import io.lbry.browser.MainActivity; -import io.lbry.lbrysdk.Utils; - -import java.util.Date; -import java.util.HashMap; -import java.util.Map; -import org.json.JSONObject; -import org.json.JSONException; - -public class FirebaseModule extends ReactContextBaseJavaModule { - - private Context context; - - private FirebaseAnalytics firebaseAnalytics; - - public FirebaseModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - this.firebaseAnalytics = FirebaseAnalytics.getInstance(context); - } - - @Override - public String getName() { - return "Firebase"; - } - - @ReactMethod - public void setCurrentScreen(String name, final Promise promise) { - final Activity activity = getCurrentActivity(); - if (activity != null && firebaseAnalytics != null) { - activity.runOnUiThread(new Runnable() { - public void run() { - firebaseAnalytics.setCurrentScreen(activity, name, Utils.capitalizeAndStrip(name)); - } - }); - } - promise.resolve(true); - } - - @ReactMethod - public void track(String name, ReadableMap payload) { - Bundle bundle = new Bundle(); - if (payload != null) { - HashMap payloadMap = payload.toHashMap(); - for (Map.Entry entry : payloadMap.entrySet()) { - Object value = entry.getValue(); - if (value != null) { - bundle.putString(entry.getKey(), entry.getValue().toString()); - } - } - } - - if (firebaseAnalytics != null) { - firebaseAnalytics.logEvent(name, bundle); - } - } - - @ReactMethod - public void logException(boolean fatal, String message, String error) { - Bundle bundle = new Bundle(); - bundle.putString("exception_message", message); - bundle.putString("exception_error", error); - if (firebaseAnalytics != null) { - firebaseAnalytics.logEvent(fatal ? "reactjs_exception" : "reactjs_warning", bundle); - } - - if (fatal) { - Toast.makeText(context, - "An application error occurred which has been automatically logged. " + - "If you keep seeing this message, please provide feedback to the LBRY " + - "team by emailing hello@lbry.com.", - Toast.LENGTH_LONG).show(); - } - } - - @ReactMethod - public void getMessagingToken(final Promise promise) { - FirebaseInstanceId.getInstance().getInstanceId() - .addOnCompleteListener(new OnCompleteListener() { - @Override - public void onComplete(Task task) { - if (!task.isSuccessful()) { - promise.reject("Firebase getInstanceId call failed"); - return; - } - - // Get new Instance ID token - String token = task.getResult().getToken(); - promise.resolve(token); - } - }); - } - - @ReactMethod - public void logLaunchTiming() { - Date end = new Date(); - MainActivity.LaunchTiming currentTiming = MainActivity.CurrentLaunchTiming; - if (currentTiming == null) { - // no start timing data, so skip this - return; - } - - long totalTimeMs = end.getTime() - currentTiming.getStart().getTime(); - String eventName = currentTiming.isColdStart() ? "app_cold_start" : "app_warm_start"; - Bundle bundle = new Bundle(); - bundle.putLong("total_ms", totalTimeMs); - bundle.putLong("total_seconds", new Double(Math.ceil(totalTimeMs / 1000.0)).longValue()); - if (firebaseAnalytics != null) { - firebaseAnalytics.logEvent(eventName, bundle); - } - } -} \ No newline at end of file diff --git a/app/src/main/java/io/lbry/browser/reactmodules/FirstRunModule.java b/app/src/main/java/io/lbry/browser/reactmodules/FirstRunModule.java deleted file mode 100644 index 73212018..00000000 --- a/app/src/main/java/io/lbry/browser/reactmodules/FirstRunModule.java +++ /dev/null @@ -1,53 +0,0 @@ -package io.lbry.browser.reactmodules; - -import android.content.Context; -import android.content.pm.PackageInfo; -import android.content.pm.PackageManager; -import android.content.SharedPreferences; -import android.os.Bundle; - -import com.facebook.react.bridge.Promise; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; - -import com.google.firebase.analytics.FirebaseAnalytics; - -import io.lbry.browser.MainActivity; - -public class FirstRunModule extends ReactContextBaseJavaModule { - private Context context; - - private SharedPreferences sp; - - public FirstRunModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - this.sp = reactContext.getSharedPreferences(MainActivity.SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - } - - @Override - public String getName() { - return "FirstRun"; - } - - @ReactMethod - public void isFirstRun(final Promise promise) { - // If firstRun flag does not exist, default to true - boolean firstRun = sp.getBoolean("firstRun", true); - promise.resolve(firstRun); - } - - @ReactMethod - public void firstRunCompleted() { - SharedPreferences.Editor editor = sp.edit(); - editor.putBoolean("firstRun", false); - editor.commit(); - - FirebaseAnalytics firebase = FirebaseAnalytics.getInstance(context); - if (firebase != null) { - Bundle bundle = new Bundle(); - firebase.logEvent("first_run_completed", bundle); - } - } -} diff --git a/app/src/main/java/io/lbry/browser/reactmodules/GalleryModule.java b/app/src/main/java/io/lbry/browser/reactmodules/GalleryModule.java deleted file mode 100644 index fd1e25f2..00000000 --- a/app/src/main/java/io/lbry/browser/reactmodules/GalleryModule.java +++ /dev/null @@ -1,313 +0,0 @@ -package io.lbry.browser.reactmodules; - -import android.content.Context; -import android.content.ContentResolver; -import android.database.Cursor; -import android.graphics.Bitmap; -import android.graphics.BitmapFactory; -import android.Manifest; -import android.media.ThumbnailUtils; -import android.os.AsyncTask; -import android.os.Bundle; -import android.provider.MediaStore; - -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.Promise; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.bridge.WritableArray; -import com.facebook.react.bridge.WritableMap; -import com.facebook.react.modules.core.DeviceEventManagerModule; - -import io.lbry.browser.MainActivity; -import io.lbry.lbrysdk.Utils; - -import java.io.File; -import java.io.FileInputStream; -import java.io.FileOutputStream; -import java.io.IOException; -import java.util.List; -import java.util.ArrayList; - -public class GalleryModule extends ReactContextBaseJavaModule { - private Context context; - - public GalleryModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - } - - @Override - public String getName() { - return "Gallery"; - } - - @ReactMethod - public void getVideos(Promise promise) { - WritableArray items = Arguments.createArray(); - List videos = loadVideos(); - for (int i = 0; i < videos.size(); i++) { - items.pushMap(videos.get(i).toMap()); - } - - promise.resolve(items); - } - - @ReactMethod - public void getThumbnailPath(Promise promise) { - if (context != null) { - File cacheDir = context.getExternalCacheDir(); - String thumbnailPath = String.format("%s/thumbnails", cacheDir.getAbsolutePath()); - promise.resolve(thumbnailPath); - return; - } - - promise.resolve(null); - } - - @ReactMethod - public void getUploadsPath(Promise promise) { - if (context != null) { - String baseFolder = Utils.getExternalStorageDir(context); - String uploadsPath = String.format("%s/LBRY/Uploads", baseFolder); - File uploadsDir = new File(uploadsPath); - if (!uploadsDir.isDirectory()) { - uploadsDir.mkdirs(); - } - promise.resolve(uploadsPath); - } - - promise.reject("The content could not be saved to the device. Please check your storage permissions."); - } - - @ReactMethod - public void createVideoThumbnail(String targetId, String filePath, Promise promise) { - (new AsyncTask() { - protected String doInBackground(Void... param) { - String thumbnailPath = null; - - if (context != null) { - Bitmap thumbnail = ThumbnailUtils.createVideoThumbnail(filePath, MediaStore.Video.Thumbnails.MINI_KIND); - File cacheDir = context.getExternalCacheDir(); - thumbnailPath = String.format("%s/thumbnails/%s.png", cacheDir.getAbsolutePath(), targetId); - - File file = new File(thumbnailPath); - try (FileOutputStream os = new FileOutputStream(thumbnailPath)) { - thumbnail.compress(Bitmap.CompressFormat.PNG, 80, os); - os.close(); - } catch (IOException ex) { - promise.reject("Could not create a thumbnail for the video"); - return null; - } - } - - return thumbnailPath; - } - - public void onPostExecute(String thumbnailPath) { - if (thumbnailPath != null && thumbnailPath.trim().length() > 0) { - promise.resolve(thumbnailPath); - } - } - }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); - } - - @ReactMethod - public void createImageThumbnail(String targetId, String filePath, Promise promise) { - (new AsyncTask() { - protected String doInBackground(Void... param) { - String thumbnailPath = null; - FileOutputStream os = null; - try { - Bitmap source = BitmapFactory.decodeFile(filePath); - // MINI_KIND dimensions - Bitmap thumbnail = Bitmap.createScaledBitmap(source, 512, 384, false); - - if (context != null) { - File cacheDir = context.getExternalCacheDir(); - thumbnailPath = String.format("%s/thumbnails/%s.png", cacheDir.getAbsolutePath(), targetId); - os = new FileOutputStream(thumbnailPath); - if (thumbnail != null) { - thumbnail.compress(Bitmap.CompressFormat.PNG, 80, os); - } - os.close(); - } - } catch (IOException ex) { - promise.reject("Could not create a thumbnail for the image"); - return null; - } finally { - if (os != null) { - try { - os.close(); - } catch (IOException ex) { - // ignoe - } - } - } - - return thumbnailPath; - } - - public void onPostExecute(String thumbnailPath) { - if (thumbnailPath != null && thumbnailPath.trim().length() > 0) { - promise.resolve(thumbnailPath); - } - } - }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); - } - - private List loadVideos() { - String[] projection = { - MediaStore.MediaColumns._ID, - MediaStore.MediaColumns.DATA, - MediaStore.MediaColumns.DISPLAY_NAME, - MediaStore.MediaColumns.MIME_TYPE, - MediaStore.Video.Media.DURATION - }; - - List ids = new ArrayList(); - List items = new ArrayList(); - Cursor cursor = context.getContentResolver().query(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, projection, null, null, - String.format("%s DESC", MediaStore.MediaColumns.DATE_MODIFIED)); - while (cursor.moveToNext()) { - int idColumn = cursor.getColumnIndex(MediaStore.MediaColumns._ID); - int nameColumn = cursor.getColumnIndex(MediaStore.MediaColumns.DISPLAY_NAME); - int typeColumn = cursor.getColumnIndex(MediaStore.MediaColumns.MIME_TYPE); - int pathColumn = cursor.getColumnIndex(MediaStore.MediaColumns.DATA); - int durationColumn = cursor.getColumnIndex(MediaStore.Video.Media.DURATION); - - String id = cursor.getString(idColumn); - GalleryItem item = new GalleryItem(); - item.setId(id); - item.setName(cursor.getString(nameColumn)); - item.setType(cursor.getString(typeColumn)); - item.setFilePath(cursor.getString(pathColumn)); - items.add(item); - ids.add(id); - } - - checkThumbnails(ids); - - return items; - } - - private void checkThumbnails(final List ids) { - (new AsyncTask() { - protected Void doInBackground(Void... param) { - if (context != null) { - ContentResolver resolver = context.getContentResolver(); - for (int i = 0; i < ids.size(); i++) { - String id = ids.get(i); - File cacheDir = context.getExternalCacheDir(); - File thumbnailsDir = new File(String.format("%s/thumbnails", cacheDir.getAbsolutePath())); - if (!thumbnailsDir.isDirectory()) { - thumbnailsDir.mkdirs(); - } - - String thumbnailPath = String.format("%s/%s.png", thumbnailsDir.getAbsolutePath(), id); - File file = new File(thumbnailPath); - if (!file.exists()) { - // save the thumbnail to the path - BitmapFactory.Options options = new BitmapFactory.Options(); - options.inSampleSize = 1; - Bitmap thumbnail = MediaStore.Video.Thumbnails.getThumbnail( - resolver, Long.parseLong(id), MediaStore.Video.Thumbnails.MINI_KIND, options); - if (thumbnail != null) { - try (FileOutputStream os = new FileOutputStream(thumbnailPath)) { - thumbnail.compress(Bitmap.CompressFormat.PNG, 80, os); - } catch (IOException ex) { - // skip - } - } - } - - if (file.exists() && file.length() > 0 && GalleryModule.this.context != null) { - WritableMap params = Arguments.createMap(); - params.putString("id", id); - ((ReactApplicationContext) GalleryModule.this.context).getJSModule( - DeviceEventManagerModule.RCTDeviceEventEmitter.class).emit("onGalleryThumbnailChecked", params); - } - } - } - - return null; - } - - public void onPostExecute(Void result) { - if (GalleryModule.this.context != null) { - ((ReactApplicationContext) GalleryModule.this.context).getJSModule( - DeviceEventManagerModule.RCTDeviceEventEmitter.class).emit("onAllGalleryThumbnailsChecked", null); - } - } - }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); - } - - private static class GalleryItem { - private String id; - - private int duration; - - private String filePath; - - private String name; - - private String type; - - public String getId() { - return id; - } - - public void setId(String id) { - this.id = id; - } - - public int getDuration() { - return duration; - } - - public void setDuration(int duration) { - this.duration = duration; - } - - public String getFilePath() { - return filePath; - } - - public void setFilePath(String filePath) { - this.filePath = filePath; - } - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } - - public WritableMap toMap() { - WritableMap map = Arguments.createMap(); - map.putString("id", id); - map.putString("name", name); - map.putString("filePath", filePath); - map.putString("type", type); - map.putInt("duration", duration); - - return map; - } - } - - @ReactMethod - public void canUseCamera(final Promise promise) { - promise.resolve(MainActivity.hasPermission(Manifest.permission.CAMERA, MainActivity.getActivity())); - } -} diff --git a/app/src/main/java/io/lbry/browser/reactmodules/RequestsModule.java b/app/src/main/java/io/lbry/browser/reactmodules/RequestsModule.java deleted file mode 100644 index a82a1707..00000000 --- a/app/src/main/java/io/lbry/browser/reactmodules/RequestsModule.java +++ /dev/null @@ -1,74 +0,0 @@ -package io.lbry.browser.reactmodules; - -import android.content.Context; -import android.os.AsyncTask; -import android.os.Bundle; - -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.Promise; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.bridge.ReadableArray; -import com.facebook.react.bridge.ReadableMap; -import com.facebook.react.bridge.ReadableMapKeySetIterator; -import com.facebook.react.bridge.ReadableType; -import com.facebook.react.bridge.WritableMap; - -import io.lbry.browser.MainActivity; -import io.lbry.lbrysdk.Utils; - -import java.util.List; -import java.util.ArrayList; - -import org.json.JSONObject; -import org.json.JSONArray; -import org.json.JSONException; - -public class RequestsModule extends ReactContextBaseJavaModule { - private Context context; - - public RequestsModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - } - - @Override - public String getName() { - return "Requests"; - } - - @ReactMethod - public void get(final String url, final Promise promise) { - (new AsyncTask() { - @Override - protected String doInBackground(Void... params) { - try { - return Utils.performRequest(url); - } catch (Exception ex) { - return null; - } - } - - protected void onPostExecute(String response) { - if (response == null) { - promise.reject(String.format("Request to %s returned null.", url)); - return; - } - - promise.resolve(response); - } - }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); - - } - - @ReactMethod - public void lbryioCall(String authToken, final Promise promise) { - // get the auth token here, or let the app pass it in? - } - - @ReactMethod - public void lbryCall(final Promise promise) { - - } -} \ No newline at end of file diff --git a/app/src/main/java/io/lbry/browser/reactmodules/ScreenOrientationModule.java b/app/src/main/java/io/lbry/browser/reactmodules/ScreenOrientationModule.java deleted file mode 100644 index 58abfa27..00000000 --- a/app/src/main/java/io/lbry/browser/reactmodules/ScreenOrientationModule.java +++ /dev/null @@ -1,47 +0,0 @@ -package io.lbry.browser.reactmodules; - -import android.app.Activity; -import android.content.Context; -import android.content.pm.ActivityInfo; - -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; - -public class ScreenOrientationModule extends ReactContextBaseJavaModule { - private Context context; - - public ScreenOrientationModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - } - - @Override - public String getName() { - return "ScreenOrientation"; - } - - @ReactMethod - public void unlockOrientation() { - Activity activity = getCurrentActivity(); - if (activity != null) { - activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_USER); - } - } - - @ReactMethod - public void lockOrientationLandscape() { - Activity activity = getCurrentActivity(); - if (activity != null) { - activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE); - } - } - - @ReactMethod - public void lockOrientationPortrait() { - Activity activity = getCurrentActivity(); - if (activity != null) { - activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT); - } - } -} diff --git a/app/src/main/java/io/lbry/browser/reactmodules/StatePersistorModule.java b/app/src/main/java/io/lbry/browser/reactmodules/StatePersistorModule.java deleted file mode 100644 index 9424abcb..00000000 --- a/app/src/main/java/io/lbry/browser/reactmodules/StatePersistorModule.java +++ /dev/null @@ -1,193 +0,0 @@ -package io.lbry.browser.reactmodules; - -import android.content.Context; -import android.os.AsyncTask; -import android.os.Bundle; - -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.Promise; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.bridge.ReadableArray; -import com.facebook.react.bridge.ReadableMap; -import com.facebook.react.bridge.ReadableMapKeySetIterator; -import com.facebook.react.bridge.ReadableType; -import com.facebook.react.bridge.WritableMap; - -import io.lbry.browser.MainActivity; - -import java.util.List; -import java.util.ArrayList; - -import org.json.JSONObject; -import org.json.JSONArray; -import org.json.JSONException; - -public class StatePersistorModule extends ReactContextBaseJavaModule { - private Context context; - - private List queue; - - private ReadableMap filter; - - private ReadableMap lastState; - - private AsyncTask persistTask; - - public StatePersistorModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - queue = new ArrayList(); - } - - @Override - public String getName() { - return "StatePersistor"; - } - - /*private WritableMap filterState(ReadableMap state) { - WritableMap filteredState = Arguments.createMap(); - - return state; - }*/ - - public boolean hasStateChanged(ReadableMap newState) { - return false; - } - - @ReactMethod - public void update(ReadableMap state, ReadableMap filter) { - if (this.filter == null) { - this.filter = filter; - } - // process state updates from the queue using a background task - synchronized (this) { - queue.add(state); - } - persistState(); - } - - private void persistState() { - persistState(false); - } - - private void persistState(final boolean flush) { - if (flush && persistTask != null) { - persistTask.cancel(true); - persistTask = null; - } - - if (persistTask == null) { - persistTask = (new AsyncTask() { - protected Boolean doInBackground(Object... param) { - // get the first item in the queue - ReadableMap queuedState = null; - if (queue.size() > 0) { - synchronized (StatePersistorModule.this) { - queuedState = queue.remove(flush ? queue.size() - 1 : 0); - if (flush) { - // we only want the final state in this scenario - queue.clear(); - } - } - } - - if (queuedState != null) { - ReadableMap state = queuedState; //(ReadableMap) filterState(queuedState); - // convert to JSON object - - try { - JSONObject json = readableMapToJSON(state); - - // save the state file - // TODO: explore this option at a later date - throw new UnsupportedOperationException(); - } catch (JSONException ex) { - // normally shouldn't happen, but if it does, reinsert into the queue - if (queuedState != null) { - synchronized (StatePersistorModule.this) { - queue.add(0, queuedState); - } - } - return false; - } - } - - return false; - } - - public void onPostExecute(Boolean result) { - if (queue.size() > 0) { - persistState(); - } - - persistTask = null; - } - }); - persistTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); - } - } - - @ReactMethod - public void flush() { - persistState(true); - } - - private static JSONObject readableMapToJSON(ReadableMap readableMap) throws JSONException { - JSONObject json = new JSONObject(); - ReadableMapKeySetIterator iterator = readableMap.keySetIterator(); - while (iterator.hasNextKey()) { - String key = iterator.nextKey(); - switch (readableMap.getType(key)) { - case Map: - json.put(key, readableMapToJSON(readableMap.getMap(key))); - break; - case Array: - json.put(key, readableArrayToJSON(readableMap.getArray(key))); - break; - case Boolean: - json.put(key, readableMap.getBoolean(key)); - break; - case Null: - json.put(key, JSONObject.NULL); - break; - case Number: - json.put(key, readableMap.getDouble(key)); - break; - case String: - json.put(key, readableMap.getString(key)); - break; - } - } - - return json; - } - - private static JSONArray readableArrayToJSON(ReadableArray readableArray) throws JSONException { - JSONArray array = new JSONArray(); - for (int i = 0; i < readableArray.size(); i++) { - switch (readableArray.getType(i)) { - case Null: - break; - case Boolean: - array.put(readableArray.getBoolean(i)); - break; - case Number: - array.put(readableArray.getDouble(i)); - break; - case String: - array.put(readableArray.getString(i)); - break; - case Map: - array.put(readableMapToJSON(readableArray.getMap(i))); - break; - case Array: - array.put(readableArrayToJSON(readableArray.getArray(i))); - break; - } - } - - return array; - } -} \ No newline at end of file diff --git a/app/src/main/java/io/lbry/browser/reactmodules/UtilityModule.java b/app/src/main/java/io/lbry/browser/reactmodules/UtilityModule.java deleted file mode 100644 index 2be60d7e..00000000 --- a/app/src/main/java/io/lbry/browser/reactmodules/UtilityModule.java +++ /dev/null @@ -1,537 +0,0 @@ -package io.lbry.browser.reactmodules; - -import android.app.Activity; -import android.app.NotificationChannel; -import android.app.NotificationManager; -import android.app.PendingIntent; -import android.content.Context; -import android.content.Intent; -import android.content.SharedPreferences; -import android.graphics.Bitmap; -import android.Manifest; -import android.net.Uri; -import android.os.AsyncTask; -import android.os.Build; -import androidx.core.content.FileProvider; -import androidx.core.app.NotificationCompat; -import androidx.core.app.NotificationManagerCompat; -import androidx.core.content.ContextCompat; -import android.telephony.TelephonyManager; -import android.view.View; -import android.view.WindowManager; - -import com.facebook.react.bridge.Callback; -import com.facebook.react.bridge.Promise; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.common.MapBuilder; - -import com.facebook.react.modules.core.DeviceEventManagerModule; -import com.squareup.picasso.Picasso; - -import java.io.File; -import java.io.Closeable; -import java.io.FileInputStream; -import java.io.FileOutputStream; -import java.io.IOException; -import java.io.PrintStream; -import java.util.HashMap; -import java.util.Map; -import java.util.Random; -import java.security.KeyStore; - -import io.lbry.browser.DownloadManager; -import io.lbry.browser.MainActivity; -import io.lbry.browser.R; -import io.lbry.lbrysdk.LbrynetService; -import io.lbry.lbrysdk.Utils; - -public class UtilityModule extends ReactContextBaseJavaModule { - private static final Map activeNotifications = new HashMap(); - - private static final String FILE_PROVIDER = "io.lbry.browser.fileprovider"; - - private static final String NOTIFICATION_CHANNEL_ID = "io.lbry.browser.SUBSCRIPTIONS_NOTIFICATION_CHANNEL"; - - public static final String ACTION_NOTIFICATION_PLAY = "io.lbry.browser.ACTION_NOTIFICATION_PLAY"; - - public static final String ACTION_NOTIFICATION_LATER = "io.lbry.browser.ACTION_NOTIFICATION_LATER"; - - // Setting keys from React Native - public static final String RECEIVE_SUBSCRIPTION_NOTIFICATIONS = "receiveSubscriptionNotifications"; - - public static final String RECEIVE_REWARD_NOTIFICATIONS = "receiveRewardNotifications"; - - public static final String RECEIVE_INTERESTS_NOTIFICATIONS = "receiveInterestsNotifications"; - - public static final String RECEIVE_CREATOR_NOTIFICATIONS = "receiveCreatorNotifications"; - - public static final String DHT_ENABLED = "dhtEnabled"; - - // the last language set to be loaded - private static final String LANGUAGE_SETTING_KEY = "language"; - - private String language; - - private Context context; - - private KeyStore keyStore; - - public UtilityModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - try { - this.keyStore = Utils.initKeyStore(context); - } catch (Exception ex) { - // continue without keystore - } - - SharedPreferences sp = context.getSharedPreferences(MainActivity.SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - language = sp.getString(LANGUAGE_SETTING_KEY, "en"); - } - - @Override - public Map getConstants() { - final Map constants = MapBuilder.newHashMap(); - constants.put("language", language); - constants.put("dhtEnabled", LbrynetService.isDHTEnabled()); - return constants; - } - - @Override - public String getName() { - return "UtilityModule"; - } - - @ReactMethod - public void keepAwakeOn() { - final Activity activity = getCurrentActivity(); - - if (activity != null) { - activity.runOnUiThread(new Runnable() { - @Override - public void run() { - activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON); - } - }); - } - } - - @ReactMethod - public void keepAwakeOff() { - final Activity activity = getCurrentActivity(); - - if (activity != null) { - activity.runOnUiThread(new Runnable() { - @Override - public void run() { - activity.getWindow().clearFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON); - } - }); - } - } - - @ReactMethod - public void hideNavigationBar() { - final Activity activity = MainActivity.getActivity(); - if (activity != null) { - activity.runOnUiThread(new Runnable() { - public void run() { - View decorView = activity.getWindow().getDecorView(); - decorView.setSystemUiVisibility(View.SYSTEM_UI_FLAG_HIDE_NAVIGATION | - View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY | - View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | - View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION | - View.SYSTEM_UI_FLAG_LAYOUT_STABLE); - } - }); - } - } - - @ReactMethod - public void showNavigationBar() { - final Activity activity = MainActivity.getActivity(); - if (activity != null) { - activity.runOnUiThread(new Runnable() { - public void run() { - View decorView = activity.getWindow().getDecorView(); - decorView.setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_STABLE | - View.SYSTEM_UI_FLAG_VISIBLE); - } - }); - } - } - - @ReactMethod - public void getDeviceId(boolean requestPermission, final Promise promise) { - if (isEmulator()) { - promise.reject("Rewards cannot be claimed from an emulator nor virtual device."); - return; - } - - TelephonyManager telephonyManager = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE); - String id = null; - try { - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { - id = telephonyManager.getImei(); // GSM - if (id == null) { - id = telephonyManager.getMeid(); // CDMA - } - } else { - id = telephonyManager.getDeviceId(); - } - } catch (SecurityException ex) { - // Maybe the permission was not granted? Try to acquire permission - /*if (requestPermission) { - requestPhoneStatePermission(); - }*/ - } catch (Exception ex) { - // id could not be obtained. Display a warning that rewards cannot be claimed. - promise.reject(ex.getMessage()); - } - - if (id == null || id.trim().length() == 0) { - promise.reject("Rewards cannot be claimed because your device could not be identified."); - return; - } - - promise.resolve(id); - } - - @ReactMethod - public void canReceiveSms(final Promise promise) { - promise.resolve(MainActivity.hasPermission(Manifest.permission.RECEIVE_SMS, MainActivity.getActivity())); - } - - @ReactMethod - public void requestReceiveSmsPermission() { - MainActivity activity = (MainActivity) MainActivity.getActivity(); - if (activity != null) { - // Request for the RECEIVE_SMS permission - MainActivity.checkReceiveSmsPermission(activity); - } - } - - @ReactMethod - public void canReadWriteStorage(final Promise promise) { - promise.resolve(MainActivity.hasPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, MainActivity.getActivity())); - } - - @ReactMethod - public void requestStoragePermission() { - MainActivity activity = (MainActivity) MainActivity.getActivity(); - if (activity != null) { - MainActivity.checkStoragePermission(activity); - } - } - - @ReactMethod - public void shareLogFile(Callback errorCallback) { - String logFileName = "lbrynet.log"; - File logFile = new File(String.format("%s/%s", Utils.getAppInternalStorageDir(context), "lbrynet"), logFileName); - if (!logFile.exists()) { - errorCallback.invoke("The lbrynet.log file could not be found."); - return; - } - - try { - Uri fileUri = FileProvider.getUriForFile(context, FILE_PROVIDER, logFile); - if (fileUri != null) { - Intent shareIntent = new Intent(); - shareIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); - shareIntent.setAction(Intent.ACTION_SEND); - shareIntent.setType("text/plain"); - shareIntent.putExtra(Intent.EXTRA_STREAM, fileUri); - - Intent sendLogIntent = Intent.createChooser(shareIntent, "Send LBRY log"); - sendLogIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); - context.startActivity(sendLogIntent); - } - } catch (IllegalArgumentException e) { - errorCallback.invoke("The lbrynet.log file cannot be shared due to permission restrictions."); - } - } - - @ReactMethod - public void shareUrl(String url) { - Intent shareIntent = new Intent(); - shareIntent.setAction(Intent.ACTION_SEND); - shareIntent.setType("text/plain"); - shareIntent.putExtra(Intent.EXTRA_TEXT, url); - - Intent shareUrlIntent = Intent.createChooser(shareIntent, "Share LBRY content"); - shareUrlIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); - context.startActivity(shareUrlIntent); - } - - @ReactMethod - public void showNotificationForContent(final String uri, String title, String publisher, final String thumbnail, boolean isPlayable) { - final NotificationManager notificationManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE); - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { - NotificationChannel channel = new NotificationChannel( - NOTIFICATION_CHANNEL_ID, "LBRY Subscriptions", NotificationManager.IMPORTANCE_DEFAULT); - channel.setDescription("LBRY subscription notifications"); - notificationManager.createNotificationChannel(channel); - } - - if (activeNotifications.containsKey(uri)) { - // the notification for the specified uri is already present, don't try to create another one - return; - } - - int id = 0; - Random random = new Random(); - do { - id = random.nextInt(); - } while (id < 100); - final int notificationId = id; - - String uriWithParam = String.format("%s?download=true", uri); - Intent playIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(uriWithParam)); - playIntent.putExtra(MainActivity.SOURCE_NOTIFICATION_ID_KEY, notificationId); - playIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP); - PendingIntent playPendingIntent = PendingIntent.getActivity(context, 0, playIntent, PendingIntent.FLAG_CANCEL_CURRENT); - - boolean hasThumbnail = false; - final NotificationCompat.Builder builder = new NotificationCompat.Builder(context, NOTIFICATION_CHANNEL_ID); - builder.setAutoCancel(true) - .setColor(ContextCompat.getColor(context, R.color.lbryGreen)) - .setContentIntent(DownloadManager.getLaunchPendingIntent(uri, context)) - .setContentTitle(publisher) - .setContentText(title) - .setSmallIcon(R.drawable.ic_lbry) - .addAction(android.R.drawable.ic_media_play, (isPlayable ? "Play" : "Open"), playPendingIntent); - - activeNotifications.put(uri, notificationId); - if (thumbnail != null) { - // attempt to load the thumbnail Bitmap before displaying the notification - final Uri thumbnailUri = Uri.parse(thumbnail); - if (thumbnailUri != null) { - hasThumbnail = true; - (new AsyncTask() { - protected Bitmap doInBackground(Void... params) { - try { - return Picasso.get().load(thumbnailUri).get(); - } catch (Exception e) { - return null; - } - } - - protected void onPostExecute(Bitmap result) { - if (result != null) { - builder.setLargeIcon(result) - .setStyle(new NotificationCompat.BigPictureStyle().bigPicture(result).bigLargeIcon(null)); - } - notificationManager.notify(notificationId, builder.build()); - } - }).execute(); - } - } - - if (!hasThumbnail) { - notificationManager.notify(notificationId, builder.build()); - } - } - - private static boolean isEmulator() { - String buildModel = Build.MODEL.toLowerCase(); - return (// Check FINGERPRINT - Build.FINGERPRINT.startsWith("generic") || - Build.FINGERPRINT.startsWith("unknown") || - Build.FINGERPRINT.contains("test-keys") || - - // Check MODEL - buildModel.contains("google_sdk") || - buildModel.contains("emulator") || - buildModel.contains("android sdk built for x86") || - - // Check MANUFACTURER - Build.MANUFACTURER.contains("Genymotion") || - "unknown".equals(Build.MANUFACTURER) || - - // Check HARDWARE - Build.HARDWARE.contains("goldfish") || - Build.HARDWARE.contains("vbox86") || - - // Check PRODUCT - "google_sdk".equals(Build.PRODUCT) || - "sdk_google_phone_x86".equals(Build.PRODUCT) || - "sdk".equals(Build.PRODUCT) || - "sdk_x86".equals(Build.PRODUCT) || - "vbox86p".equals(Build.PRODUCT) || - - // Check BRAND and DEVICE - (Build.BRAND.startsWith("generic") && Build.DEVICE.startsWith("generic")) - ); - } - - @ReactMethod - public void setSecureValue(String key, String value) { - if (keyStore != null) { - Utils.setSecureValue(key, value, context, keyStore); - } - } - - @ReactMethod - public void getSecureValue(String key, Promise promise) { - if (keyStore == null) { - promise.reject("no keyStore found"); - return; - } - - promise.resolve(Utils.getSecureValue(key, context, keyStore)); - } - - @ReactMethod - public void checkDownloads() { - Intent intent = new Intent(); - intent.setAction(LbrynetService.ACTION_CHECK_DOWNLOADS); - if (context != null) { - context.sendBroadcast(intent); - } - } - - @ReactMethod - public void queueDownload(String outpoint) { - Intent intent = new Intent(); - intent.setAction(LbrynetService.ACTION_QUEUE_DOWNLOAD); - intent.putExtra("outpoint", outpoint); - - if (context != null) { - context.sendBroadcast(intent); - } - } - - @ReactMethod - public void deleteDownload(String uri) { - Intent intent = new Intent(); - intent.setAction(LbrynetService.ACTION_DELETE_DOWNLOAD); - intent.putExtra("uri", uri); - if (context != null) { - context.sendBroadcast(intent); - } - } - - @ReactMethod - public void openDocumentPicker(String type) { - Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT); - intent.setType(type); - Activity activity = MainActivity.getActivity(); - if (activity != null) { - activity.startActivityForResult( - Intent.createChooser(intent, "Select a file"), MainActivity.DOCUMENT_PICKER_RESULT_CODE); - } - } - - @ReactMethod - public void setNativeBooleanSetting(String key, final boolean value) { - if (context != null) { - SharedPreferences sp = context.getSharedPreferences(MainActivity.SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - SharedPreferences.Editor editor = sp.edit(); - editor.putBoolean(key, value); - editor.commit(); - } - - if (DHT_ENABLED.equalsIgnoreCase(key)) { - (new AsyncTask() { - protected Void doInBackground(Void... params) { - String fileContent = value ? "on" : "off"; - String path = String.format("%s/%s", Utils.getAppInternalStorageDir(context), "dht"); - PrintStream out = null; - try { - out = new PrintStream(new FileOutputStream(path)); - out.print(fileContent); - } catch (Exception ex) { - // pass - } finally { - if (out != null) { - out.close(); - } - } - return null; - } - }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); - } - } - - @ReactMethod - public void getNativeBooleanSetting(String key, boolean defaultValue, Promise promise) { - if (context != null) { - SharedPreferences sp = context.getSharedPreferences(MainActivity.SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - promise.resolve(sp.getBoolean(key, defaultValue)); - } else { - promise.resolve(null); - } - } - - @ReactMethod - public void setNativeStringSetting(String key, String value) { - if (context != null) { - SharedPreferences sp = context.getSharedPreferences(MainActivity.SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - SharedPreferences.Editor editor = sp.edit(); - editor.putString(key, value); - editor.commit(); - } - } - - @ReactMethod - public void getNativeStringSetting(String key, String defaultValue, Promise promise) { - if (context != null) { - SharedPreferences sp = context.getSharedPreferences(MainActivity.SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); - promise.resolve(sp.getString(key, defaultValue)); - } else { - promise.resolve(null); - } - } - - @ReactMethod - public void getNotificationLaunchTarget(Promise promise) { - Activity activity = MainActivity.getActivity(); - if (activity != null) { - Intent intent = activity.getIntent(); - if (intent != null) { - String target = intent.getStringExtra("target"); - if (target != null && target.trim().length() > 0) { - promise.resolve(target); - return; - } - } - } - - promise.resolve(null); - } - - @ReactMethod - public void getDownloadDirectory(Promise promise) { - // This obtains a public default download directory after the storage permission has been granted - promise.resolve(Utils.getConfiguredDownloadDirectory(context)); - } - - @ReactMethod - public void getLbrynetDirectory(Promise promise) { - String path = String.format("%s/%s", Utils.getAppInternalStorageDir(context), "lbrynet"); - promise.resolve(path); - } - - @ReactMethod - public void getPlatform(Promise promise) { - String platform = String.format("Android %s (API %s)", Utils.getAndroidRelease(), Utils.getAndroidSdk()); - promise.resolve(platform); - } - - @ReactMethod - public void checkSdkReady() { - // check that the sdk ready when the service is already running so that we can send the ready event - ReactContext reactContext = (ReactContext) context; - if (MainActivity.lbrySdkReady && reactContext != null) { - reactContext.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class).emit("onSdkReady", null); - } - } - - @ReactMethod - public void log(String tag, String message) { - android.util.Log.d(tag, message); - } -} diff --git a/app/src/main/java/io/lbry/browser/reactmodules/VersionInfoModule.java b/app/src/main/java/io/lbry/browser/reactmodules/VersionInfoModule.java deleted file mode 100644 index 387f1d9c..00000000 --- a/app/src/main/java/io/lbry/browser/reactmodules/VersionInfoModule.java +++ /dev/null @@ -1,37 +0,0 @@ -package io.lbry.browser.reactmodules; - -import android.content.Context; -import android.content.pm.PackageInfo; -import android.content.pm.PackageManager; - -import com.facebook.react.bridge.Promise; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; - -public class VersionInfoModule extends ReactContextBaseJavaModule { - private Context context; - - public VersionInfoModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - } - - @Override - public String getName() { - return "VersionInfo"; - } - - @ReactMethod - public void getAppVersion(final Promise promise) { - PackageManager packageManager = this.context.getPackageManager(); - String packageName = this.context.getPackageName(); - try { - PackageInfo packageInfo = packageManager.getPackageInfo(packageName, 0); - promise.resolve(packageInfo.versionName); - } catch (PackageManager.NameNotFoundException e) { - // normally shouldn't happen - promise.resolve("Unknown"); - } - } -} diff --git a/app/src/main/java/io/lbry/browser/reactpackages/LbryReactPackage.java b/app/src/main/java/io/lbry/browser/reactpackages/LbryReactPackage.java deleted file mode 100644 index 7b1f53e5..00000000 --- a/app/src/main/java/io/lbry/browser/reactpackages/LbryReactPackage.java +++ /dev/null @@ -1,46 +0,0 @@ -package io.lbry.browser.reactpackages; - -import com.facebook.react.ReactPackage; -import com.facebook.react.bridge.NativeModule; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.uimanager.ViewManager; - -import io.lbry.browser.reactmodules.BackgroundMediaModule; -import io.lbry.browser.reactmodules.DaemonServiceControlModule; -import io.lbry.browser.reactmodules.FirstRunModule; -import io.lbry.browser.reactmodules.FirebaseModule; -import io.lbry.browser.reactmodules.GalleryModule; -import io.lbry.browser.reactmodules.RequestsModule; -import io.lbry.browser.reactmodules.ScreenOrientationModule; -import io.lbry.browser.reactmodules.StatePersistorModule; -import io.lbry.browser.reactmodules.VersionInfoModule; -import io.lbry.browser.reactmodules.UtilityModule; - -import java.util.ArrayList; -import java.util.Collections; -import java.util.List; - -public class LbryReactPackage implements ReactPackage { - @Override - public List createViewManagers(ReactApplicationContext reactContext) { - return Collections.emptyList(); - } - - @Override - public List createNativeModules(ReactApplicationContext reactContext) { - List modules = new ArrayList<>(); - - modules.add(new BackgroundMediaModule(reactContext)); - modules.add(new DaemonServiceControlModule(reactContext)); - modules.add(new FirstRunModule(reactContext)); - modules.add(new FirebaseModule(reactContext)); - modules.add(new GalleryModule(reactContext)); - modules.add(new RequestsModule(reactContext)); - modules.add(new ScreenOrientationModule(reactContext)); - modules.add(new StatePersistorModule(reactContext)); - modules.add(new UtilityModule(reactContext)); - modules.add(new VersionInfoModule(reactContext)); - - return modules; - } -} diff --git a/app/src/main/java/io/lbry/browser/receivers/NotificationDeletedReceiver.java b/app/src/main/java/io/lbry/browser/receivers/NotificationDeletedReceiver.java deleted file mode 100644 index b3d02f22..00000000 --- a/app/src/main/java/io/lbry/browser/receivers/NotificationDeletedReceiver.java +++ /dev/null @@ -1,17 +0,0 @@ -package io.lbry.browser.receivers; - -import android.content.BroadcastReceiver; -import android.content.Context; -import android.content.Intent; - -import io.lbry.browser.DownloadManager; - -public class NotificationDeletedReceiver extends BroadcastReceiver { - @Override - public void onReceive(Context context, Intent intent) { - int notificationId = intent.getExtras().getInt(DownloadManager.NOTIFICATION_ID_KEY); - if (DownloadManager.DOWNLOAD_NOTIFICATION_GROUP_ID == notificationId) { - DownloadManager.groupCreated = false; - } - } -} diff --git a/app/src/main/java/io/lbry/browser/tasks/ChannelSubscribeTask.java b/app/src/main/java/io/lbry/browser/tasks/ChannelSubscribeTask.java new file mode 100644 index 00000000..c027c62d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/ChannelSubscribeTask.java @@ -0,0 +1,81 @@ +package io.lbry.browser.tasks; + +import android.content.Context; +import android.database.sqlite.SQLiteDatabase; +import android.database.sqlite.SQLiteException; +import android.os.AsyncTask; + +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbryio; + +public class ChannelSubscribeTask extends AsyncTask { + private Context context; + private String channelClaimId; + private Subscription subscription; + private ChannelSubscribeHandler handler; + private Exception error; + private boolean isUnsubscribing; + + public ChannelSubscribeTask(Context context, String channelClaimId, Subscription subscription, boolean isUnsubscribing, ChannelSubscribeHandler handler) { + this.context = context; + this.channelClaimId = channelClaimId; + this.subscription = subscription; + this.handler = handler; + this.isUnsubscribing = isUnsubscribing; + } + protected Boolean doInBackground(Void... params) { + SQLiteDatabase db = null; + try { + // Save to (or delete from) local store + if (context instanceof MainActivity) { + db = ((MainActivity) context).getDbHelper().getWritableDatabase(); + } + if (db != null) { + if (!isUnsubscribing) { + DatabaseHelper.createOrUpdateSubscription(subscription, db); + } else { + DatabaseHelper.deleteSubscription(subscription, db); + } + } + + // Save with Lbryio + Map options = new HashMap<>(); + options.put("claim_id", channelClaimId); + if (!isUnsubscribing) { + options.put("channel_name", subscription.getChannelName()); + } + + String action = isUnsubscribing ? "delete" : "new"; + Lbryio.call("subscription", action, options, context); + } catch (LbryioRequestException | LbryioResponseException | SQLiteException ex) { + error = ex; + return false; + } finally { + Helper.closeDatabase(db); + } + + return true; + } + protected void onPostExecute(Boolean success) { + if (handler != null) { + if (success) { + handler.onSuccess(); + } else { + handler.onError(error); + } + } + } + + public interface ChannelSubscribeHandler { + void onSuccess(); + void onError(Exception exception); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/ClaimSearchTask.java b/app/src/main/java/io/lbry/browser/tasks/ClaimSearchTask.java new file mode 100644 index 00000000..f8bc710c --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/ClaimSearchTask.java @@ -0,0 +1,54 @@ +package io.lbry.browser.tasks; + +import android.os.AsyncTask; +import android.view.View; +import android.widget.ProgressBar; + +import java.util.List; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class ClaimSearchTask extends AsyncTask> { + private Map options; + private String connectionString; + private ClaimSearchResultHandler handler; + private View progressView; + private ApiCallException error; + + public ClaimSearchTask(Map options, String connectionString, View progressView, ClaimSearchResultHandler handler) { + this.options = options; + this.connectionString = connectionString; + this.progressView = progressView; + this.handler = handler; + } + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + protected List doInBackground(Void... params) { + try { + return Lbry.claimSearch(options, connectionString); + } catch (ApiCallException ex) { + error = ex; + return null; + } + } + protected void onPostExecute(List claims) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (claims != null) { + handler.onSuccess(claims, claims.size() < Helper.parseInt(options.get("page_size"), 0)); + } else { + handler.onError(error); + } + } + } + + public interface ClaimSearchResultHandler { + void onSuccess(List claims, boolean hasReachedEnd); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/FetchCurrentUserTask.java b/app/src/main/java/io/lbry/browser/tasks/FetchCurrentUserTask.java new file mode 100644 index 00000000..567734d0 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/FetchCurrentUserTask.java @@ -0,0 +1,38 @@ +package io.lbry.browser.tasks; + +import android.os.AsyncTask; + +import io.lbry.browser.model.lbryinc.User; +import io.lbry.browser.utils.Lbryio; + +public class FetchCurrentUserTask extends AsyncTask { + private Exception error; + private FetchUserTaskHandler handler; + + public FetchCurrentUserTask(FetchUserTaskHandler handler) { + this.handler = handler; + } + protected User doInBackground(Void... params) { + try { + return Lbryio.fetchCurrentUser(null); + } catch (Exception ex) { + error = ex; + return null; + } + } + + protected void onPostExecute(User result) { + if (handler != null) { + if (result != null) { + handler.onSuccess(result); + } else { + handler.onError(error); + } + } + } + + public interface FetchUserTaskHandler { + void onSuccess(User user); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/FetchSubscriptionsTask.java b/app/src/main/java/io/lbry/browser/tasks/FetchSubscriptionsTask.java new file mode 100644 index 00000000..fbcfc9e2 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/FetchSubscriptionsTask.java @@ -0,0 +1,88 @@ +package io.lbry.browser.tasks; + +import android.content.Context; +import android.database.sqlite.SQLiteDatabase; +import android.os.AsyncTask; +import android.view.View; +import android.widget.ProgressBar; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; + +public class FetchSubscriptionsTask extends AsyncTask> { + private Context context; + private FetchSubscriptionsHandler handler; + private ProgressBar progressBar; + private Exception error; + + public FetchSubscriptionsTask(Context context, ProgressBar progressBar, FetchSubscriptionsHandler handler) { + this.context = context; + this.progressBar = progressBar; + this.handler = handler; + } + protected void onPreExecute() { + Helper.setViewVisibility(progressBar, View.VISIBLE); + } + protected List doInBackground(Void... params) { + List subscriptions = new ArrayList<>(); + SQLiteDatabase db = null; + try { + JSONArray array = (JSONArray) Lbryio.parseResponse(Lbryio.call("subscription", "list", context)); + if (context instanceof MainActivity) { + db = ((MainActivity) context).getDbHelper().getWritableDatabase(); + } + if (array != null) { + for (int i = 0; i < array.length(); i++) { + JSONObject item = array.getJSONObject(i); + String claimId = item.getString("claim_id"); + String channelName = item.getString("channel_name"); + + LbryUri url = new LbryUri(); + url.setChannelName(channelName); + url.setClaimId(claimId); + Subscription subscription = new Subscription(channelName, url.toString()); + subscriptions.add(subscription); + // Persist the subscription locally if it doesn't exist + if (db != null) { + DatabaseHelper.createOrUpdateSubscription(subscription, db); + } + } + } + } catch (LbryioRequestException | LbryioResponseException | JSONException | ClassCastException ex) { + error = ex; + return null; + } finally { + Helper.closeDatabase(db); + } + + return subscriptions; + } + protected void onPostExecute(List subscriptions) { + Helper.setViewVisibility(progressBar, View.GONE); + if (handler != null) { + if (subscriptions != null) { + handler.onSuccess(subscriptions); + } else { + handler.onError(error); + } + } + } + + public interface FetchSubscriptionsHandler { + void onSuccess(List subscriptions); + void onError(Exception exception); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/FileListTask.java b/app/src/main/java/io/lbry/browser/tasks/FileListTask.java new file mode 100644 index 00000000..c57128b4 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/FileListTask.java @@ -0,0 +1,54 @@ +package io.lbry.browser.tasks; + +import android.os.AsyncTask; +import android.view.View; + +import java.util.List; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.model.File; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class FileListTask extends AsyncTask> { + private String claimId; + private FileListResultHandler handler; + private View progressView; + private ApiCallException error; + + public FileListTask(View progressView, FileListResultHandler handler) { + this(null, progressView, handler); + } + + public FileListTask(String claimId, View progressView, FileListResultHandler handler) { + this.claimId = claimId; + this.progressView = progressView; + this.handler = handler; + } + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + protected List doInBackground(Void... params) { + try { + return Lbry.fileList(claimId); + } catch (ApiCallException ex) { + error = ex; + return null; + } + } + protected void onPostExecute(List files) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (files != null) { + handler.onSuccess(files); + } else { + handler.onError(error); + } + } + } + + public interface FileListResultHandler { + void onSuccess(List files); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/GenericTaskHandler.java b/app/src/main/java/io/lbry/browser/tasks/GenericTaskHandler.java new file mode 100644 index 00000000..19c10d10 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/GenericTaskHandler.java @@ -0,0 +1,7 @@ +package io.lbry.browser.tasks; + +public interface GenericTaskHandler { + void beforeStart(); + void onSuccess(); + void onError(Exception error); +} diff --git a/app/src/main/java/io/lbry/browser/tasks/LighthouseAutoCompleteTask.java b/app/src/main/java/io/lbry/browser/tasks/LighthouseAutoCompleteTask.java new file mode 100644 index 00000000..a0c32b86 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/LighthouseAutoCompleteTask.java @@ -0,0 +1,51 @@ +package io.lbry.browser.tasks; + +import android.os.AsyncTask; +import android.view.View; + +import java.util.List; + +import io.lbry.browser.exceptions.LbryRequestException; +import io.lbry.browser.exceptions.LbryResponseException; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lighthouse; + +public class LighthouseAutoCompleteTask extends AsyncTask> { + private String text; + private AutoCompleteResultHandler handler; + private View progressView; + private Exception error; + + public LighthouseAutoCompleteTask(String text, View progressView, AutoCompleteResultHandler handler) { + this.text = text; + this.progressView = progressView; + this.handler = handler; + } + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + protected List doInBackground(Void... params) { + try { + return Lighthouse.autocomplete(text); + } catch (LbryRequestException | LbryResponseException ex) { + error = ex; + return null; + } + } + protected void onPostExecute(List suggestions) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (suggestions != null) { + handler.onSuccess(suggestions); + } else { + handler.onError(error); + } + } + } + + public interface AutoCompleteResultHandler { + void onSuccess(List suggestions); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/LighthouseSearchTask.java b/app/src/main/java/io/lbry/browser/tasks/LighthouseSearchTask.java new file mode 100644 index 00000000..70033991 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/LighthouseSearchTask.java @@ -0,0 +1,55 @@ +package io.lbry.browser.tasks; + +import android.os.AsyncTask; +import android.view.View; +import android.widget.ProgressBar; + +import java.util.List; + +import io.lbry.browser.exceptions.LbryRequestException; +import io.lbry.browser.exceptions.LbryResponseException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lighthouse; + +public class LighthouseSearchTask extends AsyncTask> { + private String rawQuery; + private int size; + private int from; + private boolean nsfw; + private String relatedTo; + private ClaimSearchTask.ClaimSearchResultHandler handler; + private ProgressBar progressBar; + private Exception error; + + public LighthouseSearchTask(String rawQuery, int size, int from, boolean nsfw, String relatedTo, ProgressBar progressBar, ClaimSearchTask.ClaimSearchResultHandler handler) { + this.rawQuery = rawQuery; + this.size = size; + this.from = from; + this.nsfw = nsfw; + this.relatedTo = relatedTo; + this.progressBar = progressBar; + this.handler = handler; + } + protected void onPreExecute() { + Helper.setViewVisibility(progressBar, View.VISIBLE); + } + protected List doInBackground(Void... params) { + try { + return Lighthouse.search(rawQuery, size, from, nsfw, relatedTo); + } catch (LbryRequestException | LbryResponseException ex) { + error = ex; + return null; + } + } + protected void onPostExecute(List claims) { + Helper.setViewVisibility(progressBar, View.GONE); + if (handler != null) { + if (claims != null) { + handler.onSuccess(claims, claims.size() < size); + } else { + handler.onError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/ResolveTask.java b/app/src/main/java/io/lbry/browser/tasks/ResolveTask.java new file mode 100644 index 00000000..4472b585 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/ResolveTask.java @@ -0,0 +1,57 @@ +package io.lbry.browser.tasks; + +import android.os.AsyncTask; +import android.view.View; + +import java.util.Arrays; +import java.util.List; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class ResolveTask extends AsyncTask> { + private List urls; + private String connectionString; + private ResolveResultHandler handler; + private View progressView; + private ApiCallException error; + + public ResolveTask(String url, String connectionString, View progressView, ResolveResultHandler handler) { + this(Arrays.asList(url), connectionString, progressView, handler); + } + + public ResolveTask(List urls, String connectionString, View progressView, ResolveResultHandler handler) { + this.urls = urls; + this.connectionString = connectionString; + this.progressView = progressView; + this.handler = handler; + } + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + protected List doInBackground(Void... params) { + try { + return Lbry.resolve(urls, connectionString); + } catch (ApiCallException ex) { + error = ex; + return null; + } + } + protected void onPostExecute(List claims) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (claims != null) { + handler.onSuccess(claims); + } else { + handler.onError(error); + } + } + } + + public interface ResolveResultHandler { + void onSuccess(List claims); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/verification/CheckUserEmailVerifiedTask.java b/app/src/main/java/io/lbry/browser/tasks/verification/CheckUserEmailVerifiedTask.java new file mode 100644 index 00000000..f058b7b5 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/verification/CheckUserEmailVerifiedTask.java @@ -0,0 +1,30 @@ +package io.lbry.browser.tasks.verification; + +import android.os.AsyncTask; + +import io.lbry.browser.model.lbryinc.User; +import io.lbry.browser.utils.Lbryio; + +public class CheckUserEmailVerifiedTask extends AsyncTask { + private CheckUserEmailVerifiedHandler handler; + + public CheckUserEmailVerifiedTask(CheckUserEmailVerifiedHandler handler) { + this.handler = handler; + } + + protected Boolean doInBackground(Void... params) { + User user = Lbryio.fetchCurrentUser(null); + return user != null && user.isHasVerifiedEmail(); + } + + protected void onPostExecute(Boolean result) { + if (handler != null && result) { + // we only care if the user has actually verified their email + handler.onUserEmailVerified(); + } + } + + public interface CheckUserEmailVerifiedHandler { + void onUserEmailVerified(); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/verification/EmailNewTask.java b/app/src/main/java/io/lbry/browser/tasks/verification/EmailNewTask.java new file mode 100644 index 00000000..f699835e --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/verification/EmailNewTask.java @@ -0,0 +1,81 @@ +package io.lbry.browser.tasks.verification; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONObject; + +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbryio; + +public class EmailNewTask extends AsyncTask { + private String email; + private View progressView; + private EmailNewHandler handler; + private Exception error; + + public EmailNewTask(String email, View progressView, EmailNewHandler handler) { + this.email = email; + this.progressView = progressView; + this.handler = handler; + } + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + if (handler != null) { + handler.beforeStart(); + } + } + protected Boolean doInBackground(Void... params) { + try { + Map options = new HashMap<>(); + options.put("email", email); + options.put("send_verification_email", "true"); + Lbryio.parseResponse(Lbryio.call("user_email", "new", options, Helper.METHOD_POST, null)); + } catch (LbryioResponseException ex) { + if (ex.getStatusCode() == 409) { + if (handler != null) { + handler.onEmailExists(); + } + + // email already exists + Map options = new HashMap<>(); + options.put("email", email); + options.put("only_if_expired", "true"); + try { + Lbryio.parseResponse(Lbryio.call("user_email", "resend_token", options, Helper.METHOD_POST, null)); + } catch (LbryioRequestException | LbryioResponseException e) { + error = e; + return false; + } + } else { + error = ex; + return false; + } + } catch (LbryioRequestException ex) { + error = ex; + return false; + } + + return true; + } + protected void onPostExecute(Boolean result) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (result) { + handler.onSuccess(); + } else { + handler.onError(error); + } + } + } + + public interface EmailNewHandler extends GenericTaskHandler { + void onEmailExists(); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/verification/EmailResendTask.java b/app/src/main/java/io/lbry/browser/tasks/verification/EmailResendTask.java new file mode 100644 index 00000000..715acb30 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/verification/EmailResendTask.java @@ -0,0 +1,54 @@ +package io.lbry.browser.tasks.verification; + +import android.os.AsyncTask; +import android.view.View; + +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbryio; + +public class EmailResendTask extends AsyncTask { + private String email; + private View progressView; + private GenericTaskHandler handler; + private Exception error; + + public EmailResendTask(String email, View progressView, GenericTaskHandler handler) { + this.email = email; + this.progressView = progressView; + this.handler = handler; + } + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + if (handler != null) { + handler.beforeStart(); + } + } + protected Boolean doInBackground(Void... params) { + try { + Map options = new HashMap<>(); + options.put("email", email); + Lbryio.parseResponse(Lbryio.call("user_email", "resend_token", options, Helper.METHOD_POST, null)); + } catch (LbryioRequestException | LbryioResponseException ex) { + error = ex; + return false; + } + + return true; + } + protected void onPostExecute(Boolean result) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (result) { + handler.onSuccess(); + } else { + handler.onError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/DefaultSyncTaskHandler.java b/app/src/main/java/io/lbry/browser/tasks/wallet/DefaultSyncTaskHandler.java new file mode 100644 index 00000000..8e7a5d53 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/DefaultSyncTaskHandler.java @@ -0,0 +1,27 @@ +package io.lbry.browser.tasks.wallet; + +import io.lbry.browser.model.WalletSync; + +public abstract class DefaultSyncTaskHandler implements SyncTaskHandler { + public void onSyncGetSuccess(WalletSync walletSync) { + throw new UnsupportedOperationException(); + } + public void onSyncGetWalletNotFound() { + throw new UnsupportedOperationException(); + } + public void onSyncGetError(Exception error) { + throw new UnsupportedOperationException(); + } + public void onSyncSetSuccess(String hash) { + throw new UnsupportedOperationException(); + } + public void onSyncSetError(Exception error) { + throw new UnsupportedOperationException(); + } + public void onSyncApplySuccess(String hash, String data) { + throw new UnsupportedOperationException(); + } + public void onSyncApplyError(Exception error) { + throw new UnsupportedOperationException(); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/SyncApplyTask.java b/app/src/main/java/io/lbry/browser/tasks/wallet/SyncApplyTask.java new file mode 100644 index 00000000..0d7f853f --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/SyncApplyTask.java @@ -0,0 +1,73 @@ +package io.lbry.browser.tasks.wallet; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONObject; + +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class SyncApplyTask extends AsyncTask { + // flag to indicate if this sync_apply is to fetch wallet data or apply data + private boolean fetch; + private Exception error; + private String password; + private String data; + private View progressView; + private SyncTaskHandler handler; + + private String syncHash; + private String syncData; + + public SyncApplyTask(boolean fetch, SyncTaskHandler handler) { + this.fetch = fetch; + this.handler = handler; + } + + public SyncApplyTask(String password, String data, View progressView, SyncTaskHandler handler) { + this.password = password; + this.data = data; + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + + public Boolean doInBackground(Void... params) { + Map options = new HashMap<>(); + options.put("password", Helper.isNullOrEmpty(password) ? "" : password); + if (!fetch) { + options.put("data", data); + options.put("blocking", true); + } + + try { + JSONObject response = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_SYNC_APPLY, options); + syncHash = Helper.getJSONString("hash", null, response); + syncData = Helper.getJSONString("data", null, response); + } catch (ApiCallException ex) { + error = ex; + return false; + } + + return true; + } + + protected void onPostExecute(Boolean result) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (result) { + handler.onSyncApplySuccess(syncHash, syncData); + } else { + handler.onSyncApplyError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/SyncGetTask.java b/app/src/main/java/io/lbry/browser/tasks/wallet/SyncGetTask.java new file mode 100644 index 00000000..4fd57c65 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/SyncGetTask.java @@ -0,0 +1,116 @@ +package io.lbry.browser.tasks.wallet; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONObject; + +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.exceptions.WalletException; +import io.lbry.browser.model.WalletSync; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.Lbryio; + +public class SyncGetTask extends AsyncTask { + + private boolean applySyncChanges; + private boolean applySyncSuccessful; + private Exception error; + private Exception syncApplyError; + private String password; + private SyncTaskHandler handler; + private View progressView; + + private String syncHash; + private String syncData; + + public SyncGetTask(String password, boolean applySyncChanges, View progressView, SyncTaskHandler handler) { + this.password = password; + this.progressView = progressView; + this.applySyncChanges = applySyncChanges; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + protected WalletSync doInBackground(Void... params) { + try { + password = Helper.isNullOrEmpty(password) ? "" : password; + JSONObject result = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_WALLET_STATUS); + boolean isLocked = Helper.getJSONBoolean("is_locked", false, result); + boolean unlockSuccessful = + !isLocked || (boolean) Lbry.genericApiCall(Lbry.METHOD_WALLET_UNLOCK, Lbry.buildSingleParam("password", password)); + if (!unlockSuccessful) { + throw new WalletException("The wallet could be unlocked with the provided password."); + } + + String hash = (String) Lbry.genericApiCall(Lbry.METHOD_SYNC_HASH); + try { + JSONObject response = (JSONObject) Lbryio.parseResponse( + Lbryio.call("sync", "get", Lbryio.buildSingleParam("hash", hash), Helper.METHOD_POST, null)); + WalletSync walletSync = new WalletSync( + Helper.getJSONString("hash", null, response), + Helper.getJSONString("data", null, response), + Helper.getJSONBoolean("changed", false, response) + ); + if (applySyncChanges && (!hash.equalsIgnoreCase(walletSync.getHash()) || walletSync.isChanged())) { + //Lbry.sync_apply({ password, data: response.data, blocking: true }); + try { + Map options = new HashMap<>(); + options.put("hash", walletSync.getHash()); + options.put("data", walletSync.getData()); + options.put("blocking", true); + + JSONObject syncApplyResponse = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_SYNC_APPLY, options); + syncHash = Helper.getJSONString("hash", null, syncApplyResponse); + syncData = Helper.getJSONString("data", null, syncApplyResponse); + applySyncSuccessful = true; + } catch (ApiCallException | ClassCastException ex) { + // sync_apply failed + syncApplyError = ex; + } + } + + if (Lbryio.isSignedIn() && !Lbryio.userHasSyncedWallet) { + // indicate that the user owns a synced wallet (only if the user is signed in) + Lbryio.userHasSyncedWallet = true; + } + + return walletSync; + } catch (LbryioResponseException ex) { + // wallet sync data doesn't exist + return null; + } + } catch (ApiCallException | WalletException | ClassCastException | LbryioRequestException ex) { + error = ex; + return null; + } + } + protected void onPostExecute(WalletSync result) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (result != null) { + handler.onSyncGetSuccess(result); + } else if (error != null) { + handler.onSyncGetError(error); + } else { + handler.onSyncGetWalletNotFound(); + } + + if (applySyncChanges) { + if (applySyncSuccessful) { + handler.onSyncApplySuccess(syncHash, syncData); + } else { + handler.onSyncApplyError(syncApplyError); + } + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/SyncSetTask.java b/app/src/main/java/io/lbry/browser/tasks/wallet/SyncSetTask.java new file mode 100644 index 00000000..311879d4 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/SyncSetTask.java @@ -0,0 +1,53 @@ +package io.lbry.browser.tasks.wallet; + +import android.os.AsyncTask; + +import org.json.JSONObject; + +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbryio; + +public class SyncSetTask extends AsyncTask { + private Exception error; + private String oldHash; + private String newHash; + private String data; + private SyncTaskHandler handler; + + public SyncSetTask(String oldHash, String newHash, String data, SyncTaskHandler handler) { + this.oldHash = oldHash; + this.newHash = newHash; + this.data = data; + this.handler = handler; + } + + protected String doInBackground(Void... params) { + try { + Map options = new HashMap<>(); + options.put("old_hash", oldHash); + options.put("new_hash", newHash); + options.put("data", data); + JSONObject response = (JSONObject) Lbryio.parseResponse( + Lbryio.call("sync", "set", options, Helper.METHOD_POST, null)); + String hash = Helper.getJSONString("hash", null, response); + return hash; + } catch (LbryioRequestException | LbryioResponseException | ClassCastException ex) { + error = ex; + return null; + } + } + protected void onPostExecute(String hash) { + if (handler != null) { + if (!Helper.isNullOrEmpty(hash)) { + handler.onSyncSetSuccess(hash); + } else if (error != null) { + handler.onSyncSetError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/SyncTaskHandler.java b/app/src/main/java/io/lbry/browser/tasks/wallet/SyncTaskHandler.java new file mode 100644 index 00000000..fd1008e5 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/SyncTaskHandler.java @@ -0,0 +1,13 @@ +package io.lbry.browser.tasks.wallet; + +import io.lbry.browser.model.WalletSync; + +public interface SyncTaskHandler { + void onSyncGetSuccess(WalletSync walletSync); + void onSyncGetWalletNotFound(); + void onSyncGetError(Exception error); + void onSyncSetSuccess(String hash); + void onSyncSetError(Exception error); + void onSyncApplySuccess(String hash, String data); + void onSyncApplyError(Exception error); +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/TransactionListTask.java b/app/src/main/java/io/lbry/browser/tasks/wallet/TransactionListTask.java new file mode 100644 index 00000000..b5e35539 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/TransactionListTask.java @@ -0,0 +1,56 @@ +package io.lbry.browser.tasks.wallet; + +import android.os.AsyncTask; +import android.view.View; + +import java.util.List; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.model.Transaction; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class TransactionListTask extends AsyncTask> { + private int page; + private int pageSize; + private View progressView; + private TransactionListHandler handler; + private Exception error; + + public TransactionListTask(int page, int pageSize, View progressView, TransactionListHandler handler) { + this.page = page; + this.pageSize = pageSize; + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + protected List doInBackground(Void... params) { + List transactions = null; + try { + transactions = Lbry.transactionList(page, pageSize); + } catch (ApiCallException ex) { + error = ex; + } + + return transactions; + } + + protected void onPostExecute(List transactions) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (transactions != null) { + handler.onSuccess(transactions, transactions.size() < pageSize); + } else { + handler.onError(error); + } + } + } + + public interface TransactionListHandler { + void onSuccess(List transactions, boolean hasReachedEnd); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/WalletAddressUnusedTask.java b/app/src/main/java/io/lbry/browser/tasks/wallet/WalletAddressUnusedTask.java new file mode 100644 index 00000000..437eb2ef --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/WalletAddressUnusedTask.java @@ -0,0 +1,55 @@ +package io.lbry.browser.tasks.wallet; + +import android.os.AsyncTask; + +import org.json.JSONException; +import org.json.JSONObject; + +import java.math.BigDecimal; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class WalletAddressUnusedTask extends AsyncTask { + private WalletAddressUnusedHandler handler; + private Exception error; + + public WalletAddressUnusedTask(WalletAddressUnusedHandler handler) { + this.handler = handler; + } + + protected void onPreExecute() { + if (handler != null) { + handler.beforeStart(); + } + } + + protected String doInBackground(Void... params) { + String address = null; + try { + address = (String) Lbry.genericApiCall(Lbry.METHOD_ADDRESS_UNUSED); + } catch (ApiCallException | ClassCastException ex) { + error = ex; + } + + return address; + } + + protected void onPostExecute(String unusedAddress) { + if (handler != null) { + if (!Helper.isNullOrEmpty(unusedAddress)) { + handler.onSuccess(unusedAddress); + } else { + handler.onError(error); + } + } + } + + public interface WalletAddressUnusedHandler { + void beforeStart(); + void onSuccess(String newAddress); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/WalletBalanceTask.java b/app/src/main/java/io/lbry/browser/tasks/wallet/WalletBalanceTask.java new file mode 100644 index 00000000..09d26c42 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/WalletBalanceTask.java @@ -0,0 +1,58 @@ +package io.lbry.browser.tasks.wallet; + +import android.os.AsyncTask; + +import org.json.JSONObject; + +import java.math.BigDecimal; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class WalletBalanceTask extends AsyncTask { + private WalletBalanceHandler handler; + private Exception error; + + public WalletBalanceTask(WalletBalanceHandler handler) { + this.handler = handler; + } + + protected WalletBalance doInBackground(Void... params) { + WalletBalance balance = new WalletBalance(); + try { + JSONObject json = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_WALLET_BALANCE); + JSONObject reservedSubtotals = Helper.getJSONObject("reserved_subtotals", json); + + balance.setAvailable(new BigDecimal(Helper.getJSONString("available", "0", json))); + balance.setReserved(new BigDecimal(Helper.getJSONString("reserved", "0", json))); + balance.setTotal(new BigDecimal(Helper.getJSONString("total", "0", json))); + if (reservedSubtotals != null) { + balance.setClaims(new BigDecimal(Helper.getJSONString("claims", "0", reservedSubtotals))); + balance.setSupports(new BigDecimal(Helper.getJSONString("supports", "0", reservedSubtotals))); + balance.setTips(new BigDecimal(Helper.getJSONString("tips", "0", reservedSubtotals))); + } + } catch (ApiCallException | ClassCastException ex) { + error = ex; + return null; + } + + return balance; + } + + protected void onPostExecute(WalletBalance walletBalance) { + if (handler != null) { + if (walletBalance != null) { + handler.onSuccess(walletBalance); + } else { + handler.onError(error); + } + } + } + + public interface WalletBalanceHandler { + void onSuccess(WalletBalance walletBalance); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/WalletSendTask.java b/app/src/main/java/io/lbry/browser/tasks/wallet/WalletSendTask.java new file mode 100644 index 00000000..8487b0a2 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/WalletSendTask.java @@ -0,0 +1,60 @@ +package io.lbry.browser.tasks.wallet; + +import android.os.AsyncTask; +import android.view.View; + +import java.util.Arrays; +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class WalletSendTask extends AsyncTask { + private String recipientAddress; + private String amount; + private View progressView; + private WalletSendHandler handler; + private Exception error; + + public WalletSendTask(String recipientAddress, String amount, View progressView, WalletSendHandler handler) { + this.recipientAddress = recipientAddress; + this.amount = amount; + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + protected Boolean doInBackground(Void... params) { + try { + Map options = new HashMap<>(); + options.put("addresses", Arrays.asList(recipientAddress)); + options.put("amount", amount); + Lbry.genericApiCall(Lbry.METHOD_WALLET_SEND, options); + } catch (ApiCallException ex) { + error = ex; + return false; + } + + return true; + } + + protected void onPostExecute(Boolean result) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (result) { + handler.onSuccess(); + } else { + handler.onError(error); + } + } + } + + public interface WalletSendHandler { + void onSuccess(); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/BaseFragment.java b/app/src/main/java/io/lbry/browser/ui/BaseFragment.java new file mode 100644 index 00000000..79893ada --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/BaseFragment.java @@ -0,0 +1,25 @@ +package io.lbry.browser.ui; + +import android.content.Context; + +import androidx.fragment.app.Fragment; + +import java.util.Map; + +import io.lbry.browser.MainActivity; +import lombok.Getter; +import lombok.Setter; + +public class BaseFragment extends Fragment { + @Getter + @Setter + private Map params; + + public void onResume() { + super.onResume(); + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).setSelectedMenuItemForFragment(this); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/allcontent/AllContentFragment.java b/app/src/main/java/io/lbry/browser/ui/allcontent/AllContentFragment.java new file mode 100644 index 00000000..df7a79de --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/allcontent/AllContentFragment.java @@ -0,0 +1,315 @@ +package io.lbry.browser.ui.allcontent; + +import android.content.Context; +import android.content.Intent; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.FileViewActivity; +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.ClaimListAdapter; +import io.lbry.browser.dialog.ContentFromDialogFragment; +import io.lbry.browser.dialog.ContentSortDialogFragment; +import io.lbry.browser.model.Claim; +import io.lbry.browser.tasks.ClaimSearchTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +// TODO: Similar code to FollowingFragment and Channel page fragment. Probably make common operations (sorting/filtering) into a control +public class AllContentFragment extends BaseFragment { + + private boolean singleTagView; + private List tags; + private View layoutFilterContainer; + private View sortLink; + private View contentFromLink; + private View scopeLink; + private TextView titleView; + private TextView sortLinkText; + private TextView contentFromLinkText; + private TextView scopeLinkText; + private RecyclerView contentList; + private int currentSortBy; + private int currentContentFrom; + private int currentScope; + private String contentReleaseTime; + private List contentSortOrder; + private View fromPrefix; + private View forPrefix; + private View contentLoading; + private View bigContentLoading; + private ClaimListAdapter contentListAdapter; + private boolean contentClaimSearchLoading; + private boolean contentHasReachedEnd; + private int currentClaimSearchPage; + private ClaimSearchTask contentClaimSearchTask; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_all_content, container, false); + + // All content page is sorted by trending by default, past week if sort is top + currentSortBy = ContentSortDialogFragment.ITEM_SORT_BY_TRENDING; + currentContentFrom = ContentFromDialogFragment.ITEM_FROM_PAST_WEEK; + + layoutFilterContainer = root.findViewById(R.id.all_content_filter_container); + titleView = root.findViewById(R.id.all_content_page_title); + sortLink = root.findViewById(R.id.all_content_sort_link); + contentFromLink = root.findViewById(R.id.all_content_time_link); + scopeLink = root.findViewById(R.id.all_content_scope_link); + fromPrefix = root.findViewById(R.id.all_content_from_prefix); + forPrefix = root.findViewById(R.id.all_content_for_prefix); + + sortLinkText = root.findViewById(R.id.all_content_sort_link_text); + contentFromLinkText = root.findViewById(R.id.all_content_time_link_text); + scopeLinkText = root.findViewById(R.id.all_content_scope_link_text); + + bigContentLoading = root.findViewById(R.id.all_content_main_progress); + contentLoading = root.findViewById(R.id.all_content_load_progress); + + contentList = root.findViewById(R.id.all_content_list); + LinearLayoutManager llm = new LinearLayoutManager(getContext()); + contentList.setLayoutManager(llm); + contentList.addOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { + if (contentClaimSearchLoading) { + return; + } + + LinearLayoutManager lm = (LinearLayoutManager) recyclerView.getLayoutManager(); + if (lm != null) { + int visibleItemCount = lm.getChildCount(); + int totalItemCount = lm.getItemCount(); + int pastVisibleItems = lm.findFirstVisibleItemPosition(); + if (pastVisibleItems + visibleItemCount >= totalItemCount) { + if (!contentHasReachedEnd) { + // load more + currentClaimSearchPage++; + fetchClaimSearchContent(); + } + } + } + } + }); + + sortLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + ContentSortDialogFragment dialog = ContentSortDialogFragment.newInstance(); + dialog.setCurrentSortByItem(currentSortBy); + dialog.setSortByListener(new ContentSortDialogFragment.SortByListener() { + @Override + public void onSortByItemSelected(int sortBy) { + onSortByChanged(sortBy); + } + }); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + dialog.show(activity.getSupportFragmentManager(), ContentSortDialogFragment.TAG); + } + } + }); + contentFromLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + ContentFromDialogFragment dialog = ContentFromDialogFragment.newInstance(); + dialog.setCurrentFromItem(currentContentFrom); + dialog.setContentFromListener(new ContentFromDialogFragment.ContentFromListener() { + @Override + public void onContentFromItemSelected(int contentFromItem) { + onContentFromChanged(contentFromItem); + } + }); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + dialog.show(activity.getSupportFragmentManager(), ContentFromDialogFragment.TAG); + } + } + }); + + checkParams(false); + + return root; + } + + public void setParams(Map params) { + super.setParams(params); + if (getView() != null) { + checkParams(true); + } + } + + private void checkParams(boolean reload) { + Map params = getParams(); + if (params != null && params.containsKey("singleTag")) { + String tagName = params.get("singleTag").toString(); + singleTagView = true; + tags = Arrays.asList(tagName); + titleView.setText(Helper.capitalize(tagName)); + } else { + singleTagView = false; + tags = null; + titleView.setText(getString(R.string.all_content)); + } + + forPrefix.setVisibility(singleTagView ? View.GONE : View.VISIBLE); + scopeLink.setVisibility(singleTagView ? View.GONE : View.VISIBLE); + + if (reload) { + fetchClaimSearchContent(true); + } + } + + private void onContentFromChanged(int contentFrom) { + currentContentFrom = contentFrom; + + // rebuild options and search + updateContentFromLinkText(); + contentReleaseTime = Helper.buildReleaseTime(currentContentFrom); + fetchClaimSearchContent(true); + } + + private void onSortByChanged(int sortBy) { + currentSortBy = sortBy; + + // rebuild options and search + Helper.setViewVisibility(fromPrefix, currentSortBy == ContentSortDialogFragment.ITEM_SORT_BY_TOP ? View.VISIBLE : View.GONE); + Helper.setViewVisibility(contentFromLink, currentSortBy == ContentSortDialogFragment.ITEM_SORT_BY_TOP ? View.VISIBLE : View.GONE); + currentContentFrom = currentSortBy == ContentSortDialogFragment.ITEM_SORT_BY_TOP ? + (currentContentFrom == 0 ? ContentFromDialogFragment.ITEM_FROM_PAST_WEEK : currentContentFrom) : 0; + + updateSortByLinkText(); + contentSortOrder = Helper.buildContentSortOrder(currentSortBy); + contentReleaseTime = Helper.buildReleaseTime(currentContentFrom); + fetchClaimSearchContent(true); + } + + private void updateSortByLinkText() { + int stringResourceId = -1; + switch (currentSortBy) { + case ContentSortDialogFragment.ITEM_SORT_BY_NEW: default: stringResourceId = R.string.new_text; break; + case ContentSortDialogFragment.ITEM_SORT_BY_TOP: stringResourceId = R.string.top; break; + case ContentSortDialogFragment.ITEM_SORT_BY_TRENDING: stringResourceId = R.string.trending; break; + } + + Helper.setViewText(sortLinkText, stringResourceId); + } + + private void updateContentFromLinkText() { + int stringResourceId = -1; + switch (currentContentFrom) { + case ContentFromDialogFragment.ITEM_FROM_PAST_24_HOURS: stringResourceId = R.string.past_24_hours; break; + case ContentFromDialogFragment.ITEM_FROM_PAST_WEEK: default: stringResourceId = R.string.past_week; break; + case ContentFromDialogFragment.ITEM_FROM_PAST_MONTH: stringResourceId = R.string.past_month; break; + case ContentFromDialogFragment.ITEM_FROM_PAST_YEAR: stringResourceId = R.string.past_year; break; + case ContentFromDialogFragment.ITEM_FROM_ALL_TIME: stringResourceId = R.string.all_time; break; + } + + Helper.setViewText(contentFromLinkText, stringResourceId); + } + + public void onResume() { + super.onResume(); + fetchClaimSearchContent(); + } + + private Map buildContentOptions() { + return Lbry.buildClaimSearchOptions( + Claim.TYPE_STREAM, + tags != null ? tags : null, + null, // TODO: Check mature + null, + null, + getContentSortOrder(), + contentReleaseTime, + currentClaimSearchPage == 0 ? 1 : currentClaimSearchPage, + Helper.CONTENT_PAGE_SIZE); + } + + private List getContentSortOrder() { + if (contentSortOrder == null) { + return Arrays.asList(Claim.ORDER_BY_TRENDING_GROUP, Claim.ORDER_BY_TRENDING_MIXED); + } + return contentSortOrder; + } + + private View getLoadingView() { + return (contentListAdapter == null || contentListAdapter.getItemCount() == 0) ? bigContentLoading : contentLoading; + } + + private void fetchClaimSearchContent() { + fetchClaimSearchContent(false); + } + + private void fetchClaimSearchContent(boolean reset) { + if (reset && contentListAdapter != null) { + contentListAdapter.clearItems(); + currentClaimSearchPage = 1; + } + + contentClaimSearchLoading = true; + Map claimSearchOptions = buildContentOptions(); + contentClaimSearchTask = new ClaimSearchTask(claimSearchOptions, Lbry.LBRY_TV_CONNECTION_STRING, getLoadingView(), new ClaimSearchTask.ClaimSearchResultHandler() { + @Override + public void onSuccess(List claims, boolean hasReachedEnd) { + if (contentListAdapter == null) { + contentListAdapter = new ClaimListAdapter(claims, getContext()); + contentListAdapter.setListener(new ClaimListAdapter.ClaimListItemListener() { + @Override + public void onClaimClicked(Claim claim) { + String claimId = claim.getClaimId(); + String url = claim.getPermanentUrl(); + if (claim.getName().startsWith("@")) { + // channel claim + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openChannelClaim(claim); + } + } else { + Intent intent = new Intent(getContext(), FileViewActivity.class); + intent.putExtra("claimId", claimId); + intent.putExtra("url", url); + MainActivity.startingFileViewActivity = true; + startActivity(intent); + } + } + }); + } else { + contentListAdapter.addItems(claims); + } + + if (contentList != null && contentList.getAdapter() == null) { + contentList.setAdapter(contentListAdapter); + } + + contentHasReachedEnd = hasReachedEnd; + contentClaimSearchLoading = false; + } + + @Override + public void onError(Exception error) { + contentClaimSearchLoading = false; + } + }); + contentClaimSearchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/channel/ChannelAboutFragment.java b/app/src/main/java/io/lbry/browser/ui/channel/ChannelAboutFragment.java new file mode 100644 index 00000000..89d0bbe8 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/channel/ChannelAboutFragment.java @@ -0,0 +1,71 @@ +package io.lbry.browser.ui.channel; + +import android.os.Bundle; +import android.text.method.LinkMovementMethod; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.core.text.HtmlCompat; +import androidx.fragment.app.Fragment; + +import io.lbry.browser.R; +import io.lbry.browser.utils.Helper; +import lombok.Setter; + +public class ChannelAboutFragment extends Fragment { + private View layoutWebsite; + private View layoutEmail; + private View layoutInfoArea; + private View layoutNoAboutInfo; + private TextView textWebsite; + private TextView textEmail; + private TextView textDescription; + + @Setter + private String website; + @Setter + private String email; + @Setter + private String description; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_channel_about, container, false); + + layoutInfoArea = root.findViewById(R.id.channel_about_info_area); + layoutNoAboutInfo = root.findViewById(R.id.channel_about_no_info_container); + layoutWebsite = root.findViewById(R.id.channel_about_website_container); + layoutEmail = root.findViewById(R.id.channel_about_email_container); + textWebsite = root.findViewById(R.id.channel_about_website); + textEmail = root.findViewById(R.id.channel_about_email); + textDescription = root.findViewById(R.id.channel_about_description); + + boolean noInfo = (Helper.isNullOrEmpty(website) && Helper.isNullOrEmpty(email) && Helper.isNullOrEmpty(description)); + layoutNoAboutInfo.setVisibility(noInfo ? View.VISIBLE : View.GONE); + layoutInfoArea.setVisibility(noInfo ? View.GONE : View.VISIBLE); + layoutWebsite.setVisibility(!Helper.isNullOrEmpty(website) ? View.VISIBLE : View.GONE); + layoutEmail.setVisibility(!Helper.isNullOrEmpty(email) ? View.VISIBLE : View.GONE); + textDescription.setVisibility(!Helper.isNullOrEmpty(description) ? View.VISIBLE : View.GONE); + + textWebsite.setLinksClickable(true); + textWebsite.setMovementMethod(LinkMovementMethod.getInstance()); + textWebsite.setText(!Helper.isNullOrEmpty(website) ? + HtmlCompat.fromHtml(String.format("%s", website, website), HtmlCompat.FROM_HTML_MODE_LEGACY) : null); + + textEmail.setText(email); + textDescription.setText(description); + + return root; + } + + public void refresh() { + textWebsite.setText(!Helper.isNullOrEmpty(website) ? + HtmlCompat.fromHtml(String.format("%s", website, website), HtmlCompat.FROM_HTML_MODE_LEGACY) : null); + + textEmail.setText(email); + textDescription.setText(description); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/channel/ChannelContentFragment.java b/app/src/main/java/io/lbry/browser/ui/channel/ChannelContentFragment.java new file mode 100644 index 00000000..02b03280 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/channel/ChannelContentFragment.java @@ -0,0 +1,273 @@ +package io.lbry.browser.ui.channel; + +import android.content.Context; +import android.content.Intent; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.fragment.app.Fragment; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.FileViewActivity; +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.ClaimListAdapter; +import io.lbry.browser.dialog.ContentFromDialogFragment; +import io.lbry.browser.dialog.ContentSortDialogFragment; +import io.lbry.browser.model.Claim; +import io.lbry.browser.tasks.ClaimSearchTask; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import lombok.Setter; + +public class ChannelContentFragment extends Fragment { + + @Setter + private String channelId; + private View sortLink; + private View contentFromLink; + private TextView sortLinkText; + private TextView contentFromLinkText; + private RecyclerView contentList; + private int currentSortBy; + private int currentContentFrom; + private String contentReleaseTime; + private List contentSortOrder; + private View contentLoading; + private View bigContentLoading; + private ClaimListAdapter contentListAdapter; + private boolean contentClaimSearchLoading; + private boolean contentHasReachedEnd; + private int currentClaimSearchPage; + private ClaimSearchTask contentClaimSearchTask; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_channel_content, container, false); + + currentSortBy = ContentSortDialogFragment.ITEM_SORT_BY_TRENDING; + currentContentFrom = ContentFromDialogFragment.ITEM_FROM_PAST_WEEK; + + sortLink = root.findViewById(R.id.channel_content_sort_link); + contentFromLink = root.findViewById(R.id.channel_content_time_link); + + sortLinkText = root.findViewById(R.id.channel_content_sort_link_text); + contentFromLinkText = root.findViewById(R.id.channel_content_time_link_text); + + bigContentLoading = root.findViewById(R.id.channel_content_main_progress); + contentLoading = root.findViewById(R.id.channel_content_load_progress); + + contentList = root.findViewById(R.id.channel_content_list); + LinearLayoutManager llm = new LinearLayoutManager(getContext()); + contentList.setLayoutManager(llm); + contentList.addOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { + if (contentClaimSearchLoading) { + return; + } + + LinearLayoutManager lm = (LinearLayoutManager) recyclerView.getLayoutManager(); + if (lm != null) { + int visibleItemCount = lm.getChildCount(); + int totalItemCount = lm.getItemCount(); + int pastVisibleItems = lm.findFirstVisibleItemPosition(); + if (pastVisibleItems + visibleItemCount >= totalItemCount) { + if (!contentHasReachedEnd) { + // load more + currentClaimSearchPage++; + fetchClaimSearchContent(); + } + } + } + } + }); + + sortLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + ContentSortDialogFragment dialog = ContentSortDialogFragment.newInstance(); + dialog.setCurrentSortByItem(currentSortBy); + dialog.setSortByListener(new ContentSortDialogFragment.SortByListener() { + @Override + public void onSortByItemSelected(int sortBy) { + onSortByChanged(sortBy); + } + }); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + dialog.show(activity.getSupportFragmentManager(), ContentSortDialogFragment.TAG); + } + } + }); + contentFromLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + ContentFromDialogFragment dialog = ContentFromDialogFragment.newInstance(); + dialog.setCurrentFromItem(currentContentFrom); + dialog.setContentFromListener(new ContentFromDialogFragment.ContentFromListener() { + @Override + public void onContentFromItemSelected(int contentFromItem) { + onContentFromChanged(contentFromItem); + } + }); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + dialog.show(activity.getSupportFragmentManager(), ContentFromDialogFragment.TAG); + } + } + }); + + return root; + } + + private void onContentFromChanged(int contentFrom) { + currentContentFrom = contentFrom; + + // rebuild options and search + updateContentFromLinkText(); + contentReleaseTime = Helper.buildReleaseTime(currentContentFrom); + fetchClaimSearchContent(true); + } + + private void onSortByChanged(int sortBy) { + currentSortBy = sortBy; + + // rebuild options and search + Helper.setViewVisibility(contentFromLink, currentSortBy == ContentSortDialogFragment.ITEM_SORT_BY_TOP ? View.VISIBLE : View.GONE); + currentContentFrom = currentSortBy == ContentSortDialogFragment.ITEM_SORT_BY_TOP ? + (currentContentFrom == 0 ? ContentFromDialogFragment.ITEM_FROM_PAST_WEEK : currentContentFrom) : 0; + + updateSortByLinkText(); + contentSortOrder = Helper.buildContentSortOrder(currentSortBy); + contentReleaseTime = Helper.buildReleaseTime(currentContentFrom); + fetchClaimSearchContent(true); + } + + private void updateSortByLinkText() { + int stringResourceId = -1; + switch (currentSortBy) { + case ContentSortDialogFragment.ITEM_SORT_BY_NEW: default: stringResourceId = R.string.new_text; break; + case ContentSortDialogFragment.ITEM_SORT_BY_TOP: stringResourceId = R.string.top; break; + case ContentSortDialogFragment.ITEM_SORT_BY_TRENDING: stringResourceId = R.string.trending; break; + } + + Helper.setViewText(sortLinkText, stringResourceId); + } + + private void updateContentFromLinkText() { + int stringResourceId = -1; + switch (currentContentFrom) { + case ContentFromDialogFragment.ITEM_FROM_PAST_24_HOURS: stringResourceId = R.string.past_24_hours; break; + case ContentFromDialogFragment.ITEM_FROM_PAST_WEEK: default: stringResourceId = R.string.past_week; break; + case ContentFromDialogFragment.ITEM_FROM_PAST_MONTH: stringResourceId = R.string.past_month; break; + case ContentFromDialogFragment.ITEM_FROM_PAST_YEAR: stringResourceId = R.string.past_year; break; + case ContentFromDialogFragment.ITEM_FROM_ALL_TIME: stringResourceId = R.string.all_time; break; + } + + Helper.setViewText(contentFromLinkText, stringResourceId); + } + + public void onResume() { + super.onResume(); + fetchClaimSearchContent(); + } + + public void refresh() { + fetchClaimSearchContent(true); + } + + private Map buildContentOptions() { + return Lbry.buildClaimSearchOptions( + Claim.TYPE_STREAM, + null, + null, // TODO: Check mature + Arrays.asList(channelId), + null, + getContentSortOrder(), + contentReleaseTime, + currentClaimSearchPage == 0 ? 1 : currentClaimSearchPage, + Helper.CONTENT_PAGE_SIZE); + } + + private List getContentSortOrder() { + if (contentSortOrder == null) { + return Arrays.asList(Claim.ORDER_BY_RELEASE_TIME); + } + return contentSortOrder; + } + + private View getLoadingView() { + return (contentListAdapter == null || contentListAdapter.getItemCount() == 0) ? bigContentLoading : contentLoading; + } + + private void fetchClaimSearchContent() { + fetchClaimSearchContent(false); + } + + private void fetchClaimSearchContent(boolean reset) { + if (reset && contentListAdapter != null) { + contentListAdapter.clearItems(); + currentClaimSearchPage = 1; + } + + contentClaimSearchLoading = true; + Map claimSearchOptions = buildContentOptions(); + contentClaimSearchTask = new ClaimSearchTask(claimSearchOptions, Lbry.LBRY_TV_CONNECTION_STRING, getLoadingView(), new ClaimSearchTask.ClaimSearchResultHandler() { + @Override + public void onSuccess(List claims, boolean hasReachedEnd) { + if (contentListAdapter == null) { + contentListAdapter = new ClaimListAdapter(claims, getContext()); + contentListAdapter.setListener(new ClaimListAdapter.ClaimListItemListener() { + @Override + public void onClaimClicked(Claim claim) { + String claimId = claim.getClaimId(); + String url = claim.getPermanentUrl(); + if (claim.getName().startsWith("@")) { + // channel claim + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openChannelClaim(claim); + } + } else { + Intent intent = new Intent(getContext(), FileViewActivity.class); + intent.putExtra("claimId", claimId); + intent.putExtra("url", url); + MainActivity.startingFileViewActivity = true; + startActivity(intent); + } + } + }); + } else { + contentListAdapter.addItems(claims); + } + + if (contentList != null && contentList.getAdapter() == null) { + contentList.setAdapter(contentListAdapter); + } + + contentHasReachedEnd = hasReachedEnd; + contentClaimSearchLoading = false; + } + + @Override + public void onError(Exception error) { + contentClaimSearchLoading = false; + } + }); + contentClaimSearchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/channel/ChannelFragment.java b/app/src/main/java/io/lbry/browser/ui/channel/ChannelFragment.java new file mode 100644 index 00000000..9d9d7636 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/channel/ChannelFragment.java @@ -0,0 +1,242 @@ +package io.lbry.browser.ui.channel; + +import android.content.Context; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ImageView; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.fragment.app.Fragment; +import androidx.fragment.app.FragmentActivity; +import androidx.fragment.app.FragmentManager; +import androidx.fragment.app.FragmentTransaction; +import androidx.viewpager2.adapter.FragmentStateAdapter; +import androidx.viewpager2.widget.ViewPager2; + +import com.bumptech.glide.Glide; +import com.bumptech.glide.request.RequestOptions; +import com.google.android.material.tabs.TabLayout; +import com.google.android.material.tabs.TabLayoutMediator; + +import java.util.List; +import java.util.Map; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.model.Claim; +import io.lbry.browser.tasks.ResolveTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import lombok.SneakyThrows; + +public class ChannelFragment extends BaseFragment { + private Claim claim; + private boolean resolving; + private String url; + + private View layoutResolving; + private View layoutDisplayArea; + private ImageView imageCover; + private ImageView imageThumbnail; + private View noThumbnailView; + private TextView textAlpha; + private TextView textTitle; + private TextView textFollowerCount; + private TabLayout tabLayout; + private ViewPager2 tabPager; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_channel, container, false); + + layoutDisplayArea = root.findViewById(R.id.channel_view_claim_display_area); + layoutResolving = root.findViewById(R.id.channel_view_loading_container); + + imageCover = root.findViewById(R.id.channel_view_cover_image); + imageThumbnail = root.findViewById(R.id.channel_view_thumbnail); + noThumbnailView = root.findViewById(R.id.channel_view_no_thumbnail); + textAlpha = root.findViewById(R.id.channel_view_icon_alpha); + textTitle = root.findViewById(R.id.channel_view_title); + textFollowerCount = root.findViewById(R.id.channel_view_follower_count); + + tabPager = root.findViewById(R.id.channel_view_pager); + tabLayout = root.findViewById(R.id.channel_view_tabs); + tabPager.setSaveEnabled(false); + + return root; + } + + public void onResume() { + super.onResume(); + checkParams(); + } + + private void checkParams() { + boolean updateRequired = false; + Map params = getParams(); + + if (params.containsKey("claim")) { + Claim claim = (Claim) params.get("claim"); + if (claim != null && !claim.equals(this.claim)) { + this.claim = claim; + updateRequired = true; + } + } + if (!updateRequired && params.containsKey("url")) { + String newUrl = params.get("url").toString(); + if (!newUrl.equalsIgnoreCase(url) || claim == null) { + this.claim = null; + this.url = newUrl; + updateRequired = true; + } + } + if (updateRequired) { + resetFragments(); + if (!Helper.isNullOrEmpty(url)) { + resolveUrl(); + } else if (claim == null) { + // nothing at this location + renderNothingAtLocation(); + } + } + + if (claim != null) { + renderClaim(); + } + } + + private void resolveUrl() { + layoutDisplayArea.setVisibility(View.INVISIBLE); + ResolveTask task = new ResolveTask(url, Lbry.LBRY_TV_CONNECTION_STRING, layoutResolving, new ResolveTask.ResolveResultHandler() { + @Override + public void onSuccess(List claims) { + if (claims.size() > 0) { + claim = claims.get(0); + renderClaim(); + // TODO: Load follower count + } else { + renderNothingAtLocation(); + } + } + + @Override + public void onError(Exception error) { + renderNothingAtLocation(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void renderNothingAtLocation() { + + } + + public void setParams(Map params) { + super.setParams(params); + if (getView() != null) { + checkParams(); + } + } + + private void renderClaim() { + if (claim == null) { + renderNothingAtLocation(); + return; + } + + layoutDisplayArea.setVisibility(View.VISIBLE); + + String thumbnailUrl = claim.getThumbnailUrl(); + String coverUrl = claim.getCoverUrl(); + textTitle.setText(Helper.isNullOrEmpty(claim.getTitle()) ? claim.getName() : claim.getTitle()); + + if (!Helper.isNullOrEmpty(coverUrl)) { + Glide.with(getContext().getApplicationContext()).load(coverUrl).centerCrop().into(imageCover); + } + if (!Helper.isNullOrEmpty(thumbnailUrl)) { + Glide.with(getContext().getApplicationContext()).load(thumbnailUrl).apply(RequestOptions.circleCropTransform()).into(imageThumbnail); + noThumbnailView.setVisibility(View.GONE); + } else { + imageThumbnail.setVisibility(View.GONE); + + int bgColor = Helper.generateRandomColorForValue(claim.getClaimId()); + Helper.setIconViewBackgroundColor(noThumbnailView, bgColor, false, getContext()); + noThumbnailView.setVisibility(View.VISIBLE); + textAlpha.setText(claim.getName().substring(1, 2)); + } + + tabPager.setAdapter(new ChannelPagerAdapter(claim, (MainActivity) getContext())); + new TabLayoutMediator(tabLayout, tabPager, new TabLayoutMediator.TabConfigurationStrategy() { + @Override + public void onConfigureTab(@NonNull TabLayout.Tab tab, int position) { + tab.setText(position == 0 ? R.string.content : R.string.about); + } + }).attach(); + } + + private void resetFragments() { + try { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) getContext(); + FragmentManager manager = activity.getSupportFragmentManager(); + FragmentTransaction tx = manager.beginTransaction(); + for (Fragment fragment : manager.getFragments()) { + if (fragment.getClass().equals(ChannelAboutFragment.class) || fragment.getClass().equals(ChannelContentFragment.class)) { + tx.remove(fragment); + } + } + tx.commitAllowingStateLoss(); + } + } catch (Exception ex) { + // pass + } + } + + private static class ChannelPagerAdapter extends FragmentStateAdapter { + private Claim channelClaim; + public ChannelPagerAdapter(Claim channelClaim, FragmentActivity activity) { + super(activity); + this.channelClaim = channelClaim; + } + + @SneakyThrows + @Override + public Fragment createFragment(int position) { + switch (position) { + case 0: + ChannelContentFragment contentFragment = ChannelContentFragment.class.newInstance(); + contentFragment.setChannelId(channelClaim.getClaimId()); + return contentFragment; + + case 1: + ChannelAboutFragment aboutFragment = ChannelAboutFragment.class.newInstance(); + try { + Claim.ChannelMetadata metadata = (Claim.ChannelMetadata) channelClaim.getValue(); + aboutFragment.setDescription(metadata.getDescription()); + aboutFragment.setEmail(metadata.getEmail()); + aboutFragment.setWebsite(metadata.getWebsiteUrl()); + } catch (ClassCastException ex) { + // pass + } + return aboutFragment; + } + + return null; + } + + public long getItemId(int position) { + return String.format("%s-%d", channelClaim.getClaimId(), position).hashCode(); + } + + @Override + public int getItemCount() { + return 2; + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/controls/SolidIconView.java b/app/src/main/java/io/lbry/browser/ui/controls/SolidIconView.java new file mode 100644 index 00000000..e9de291b --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/controls/SolidIconView.java @@ -0,0 +1,29 @@ +package io.lbry.browser.ui.controls; + +import android.content.Context; +import android.graphics.Typeface; +import android.util.AttributeSet; +import android.view.Gravity; + +import androidx.appcompat.widget.AppCompatTextView; + +public class SolidIconView extends AppCompatTextView { + private Context context; + + public SolidIconView(Context context) { + super(context); + this.context = context; + init(); + } + + public SolidIconView(Context context, AttributeSet attrs) { + super(context, attrs); + this.context = context; + init(); + } + + private void init() { + setGravity(Gravity.CENTER); + setTypeface(Typeface.createFromAsset(context.getAssets(), "font_awesome_5_free_solid.otf")); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/following/FollowingFragment.java b/app/src/main/java/io/lbry/browser/ui/following/FollowingFragment.java new file mode 100644 index 00000000..39354664 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/following/FollowingFragment.java @@ -0,0 +1,581 @@ +package io.lbry.browser.ui.following; + +import android.content.Context; +import android.content.Intent; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.recyclerview.widget.GridLayoutManager; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import com.google.android.material.button.MaterialButton; +import com.google.android.material.snackbar.Snackbar; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.FileViewActivity; +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.ChannelFilterListAdapter; +import io.lbry.browser.adapter.ClaimListAdapter; +import io.lbry.browser.adapter.SuggestedChannelGridAdapter; +import io.lbry.browser.dialog.ContentFromDialogFragment; +import io.lbry.browser.dialog.ContentSortDialogFragment; +import io.lbry.browser.exceptions.LbryUriException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.tasks.ChannelSubscribeTask; +import io.lbry.browser.tasks.ClaimSearchTask; +import io.lbry.browser.tasks.FetchSubscriptionsTask; +import io.lbry.browser.tasks.ResolveTask; +import io.lbry.browser.listener.ChannelItemSelectionListener; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; + +public class FollowingFragment extends BaseFragment implements FetchSubscriptionsTask.FetchSubscriptionsHandler, ChannelItemSelectionListener { + + private static final int SUGGESTED_PAGE_SIZE = 45; + private static final int MIN_SUGGESTED_SUBSCRIBE_COUNT = 5; + + private MaterialButton suggestedDoneButton; + private TextView titleView; + private TextView infoView; + private RecyclerView horizontalChannelList; + private RecyclerView suggestedChannelGrid; + private RecyclerView contentList; + private ProgressBar bigContentLoading; + private ProgressBar contentLoading; + private ProgressBar channelListLoading; + private View layoutSortContainer; + private View sortLink; + private TextView sortLinkText; + private View contentFromLink; + private TextView contentFromLinkText; + private int currentSortBy; + private int currentContentFrom; + private String contentReleaseTime; + private List contentSortOrder; + private boolean contentClaimSearchLoading = false; + + private List queuedContentPages = new ArrayList<>(); + private List queuedSuggestedPages = new ArrayList<>(); + + private int currentSuggestedPage = 0; + private int currentClaimSearchPage; + private boolean suggestedHasReachedEnd; + private boolean contentHasReachedEnd; + private boolean contentPendingFetch = false; + private int numSuggestedSelected; + + // adapters + private SuggestedChannelGridAdapter suggestedChannelAdapter; + private ChannelFilterListAdapter channelFilterListAdapter; + private ClaimListAdapter contentListAdapter; + + private List channelIds; + private List channelUrls; + private List subscriptionsList; + private List suggestedChannels; + private ClaimSearchTask suggestedChannelClaimSearchTask; + private ClaimSearchTask contentClaimSearchTask; + private boolean loadingSuggested; + private boolean loadingContent; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_following, container, false); + + // Following page is sorted by new by default, past week if sort is top + currentSortBy = ContentSortDialogFragment.ITEM_SORT_BY_NEW; + currentContentFrom = ContentFromDialogFragment.ITEM_FROM_PAST_WEEK; + + titleView = root.findViewById(R.id.following_page_title); + infoView = root.findViewById(R.id.following_page_info); + horizontalChannelList = root.findViewById(R.id.following_channel_list); + layoutSortContainer = root.findViewById(R.id.following_filter_container); + sortLink = root.findViewById(R.id.following_sort_link); + sortLinkText = root.findViewById(R.id.following_sort_link_text); + contentFromLink = root.findViewById(R.id.following_time_link); + contentFromLinkText = root.findViewById(R.id.following_time_link_text); + suggestedChannelGrid = root.findViewById(R.id.following_suggested_grid); + suggestedDoneButton = root.findViewById(R.id.following_suggested_done_button); + contentList = root.findViewById(R.id.following_content_list); + bigContentLoading = root.findViewById(R.id.following_main_progress); + contentLoading = root.findViewById(R.id.following_content_progress); + channelListLoading = root.findViewById(R.id.following_channel_load_progress); + + Context context = getContext(); + GridLayoutManager glm = new GridLayoutManager(context, 3); + suggestedChannelGrid.setLayoutManager(glm); + + LinearLayoutManager cllm = new LinearLayoutManager(context, RecyclerView.HORIZONTAL, false); + horizontalChannelList.setLayoutManager(cllm); + + LinearLayoutManager llm = new LinearLayoutManager(context); + contentList.setLayoutManager(llm); + contentList.addOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { + if (contentClaimSearchLoading) { + return; + } + + LinearLayoutManager lm = (LinearLayoutManager) recyclerView.getLayoutManager(); + if (lm != null) { + int visibleItemCount = lm.getChildCount(); + int totalItemCount = lm.getItemCount(); + int pastVisibleItems = lm.findFirstVisibleItemPosition(); + if (pastVisibleItems + visibleItemCount >= totalItemCount) { + if (!contentHasReachedEnd) { + // load more + currentClaimSearchPage++; + fetchClaimSearchContent(); + } + } + } + } + }); + + suggestedDoneButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + int selected = suggestedChannelAdapter == null ? 0 : suggestedChannelAdapter.getSelectedCount(); + int remaining = MIN_SUGGESTED_SUBSCRIBE_COUNT - selected; + if (remaining == MIN_SUGGESTED_SUBSCRIBE_COUNT) { + Snackbar.make(getView(), R.string.select_five_subscriptions, Snackbar.LENGTH_LONG).show(); + } else { + fetchSubscriptions(); + showSubscribedContent(); + fetchAndResolveChannelList(); + } + } + }); + + sortLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + ContentSortDialogFragment dialog = ContentSortDialogFragment.newInstance(); + dialog.setCurrentSortByItem(currentSortBy); + dialog.setSortByListener(new ContentSortDialogFragment.SortByListener() { + @Override + public void onSortByItemSelected(int sortBy) { + onSortByChanged(sortBy); + } + }); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + dialog.show(activity.getSupportFragmentManager(), ContentSortDialogFragment.TAG); + } + } + }); + contentFromLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + ContentFromDialogFragment dialog = ContentFromDialogFragment.newInstance(); + dialog.setCurrentFromItem(currentContentFrom); + dialog.setContentFromListener(new ContentFromDialogFragment.ContentFromListener() { + @Override + public void onContentFromItemSelected(int contentFromItem) { + onContentFromChanged(contentFromItem); + } + }); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + dialog.show(activity.getSupportFragmentManager(), ContentFromDialogFragment.TAG); + } + } + }); + + return root; + } + + private void onContentFromChanged(int contentFrom) { + currentContentFrom = contentFrom; + + // rebuild options and search + updateContentFromLinkText(); + contentReleaseTime = Helper.buildReleaseTime(currentContentFrom); + fetchClaimSearchContent(true); + } + + private void onSortByChanged(int sortBy) { + currentSortBy = sortBy; + + // rebuild options and search + Helper.setViewVisibility(contentFromLink, currentSortBy == ContentSortDialogFragment.ITEM_SORT_BY_TOP ? View.VISIBLE : View.GONE); + currentContentFrom = currentSortBy == ContentSortDialogFragment.ITEM_SORT_BY_TOP ? + (currentContentFrom == 0 ? ContentFromDialogFragment.ITEM_FROM_PAST_WEEK : currentContentFrom) : 0; + + updateSortByLinkText(); + contentSortOrder = Helper.buildContentSortOrder(currentSortBy); + contentReleaseTime = Helper.buildReleaseTime(currentContentFrom); + fetchClaimSearchContent(true); + } + + private void updateSortByLinkText() { + int stringResourceId = -1; + switch (currentSortBy) { + case ContentSortDialogFragment.ITEM_SORT_BY_NEW: default: stringResourceId = R.string.new_text; break; + case ContentSortDialogFragment.ITEM_SORT_BY_TOP: stringResourceId = R.string.top; break; + case ContentSortDialogFragment.ITEM_SORT_BY_TRENDING: stringResourceId = R.string.trending; break; + } + + Helper.setViewText(sortLinkText, stringResourceId); + } + + private void updateContentFromLinkText() { + int stringResourceId = -1; + switch (currentContentFrom) { + case ContentFromDialogFragment.ITEM_FROM_PAST_24_HOURS: stringResourceId = R.string.past_24_hours; break; + case ContentFromDialogFragment.ITEM_FROM_PAST_WEEK: default: stringResourceId = R.string.past_week; break; + case ContentFromDialogFragment.ITEM_FROM_PAST_MONTH: stringResourceId = R.string.past_month; break; + case ContentFromDialogFragment.ITEM_FROM_PAST_YEAR: stringResourceId = R.string.past_year; break; + case ContentFromDialogFragment.ITEM_FROM_ALL_TIME: stringResourceId = R.string.all_time; break; + } + + Helper.setViewText(contentFromLinkText, stringResourceId); + } + + public void onResume() { + super.onResume(); + + // check if subscriptions exist + if (suggestedChannelAdapter != null) { + showSuggestedChannels(); + } + + if (Lbryio.cacheSubscriptions != null && Lbryio.cacheSubscriptions.size() > 0) { + subscriptionsList = new ArrayList<>(Lbryio.cacheSubscriptions); + buildChannelIdsAndUrls(); + if (Lbryio.cacheResolvedSubscriptions.size() > 0) { + updateChannelFilterListAdapter(Lbryio.cacheResolvedSubscriptions); + } else { + fetchAndResolveChannelList(); + } + fetchClaimSearchContent(); + showSubscribedContent(); + } else { + fetchSubscriptions(); + } + } + + public void loadFollowing() { + // wrapper to just re-fetch subscriptions (upon user sign in, for example) + fetchSubscriptions(); + } + + private void fetchSubscriptions() { + FetchSubscriptionsTask task = new FetchSubscriptionsTask(getContext(), channelListLoading, this); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private Map buildSuggestedOptions() { + return Lbry.buildClaimSearchOptions( + Claim.TYPE_CHANNEL, + null, + null, + null, + null, + Arrays.asList(Claim.ORDER_BY_EFFECTIVE_AMOUNT), + null, + currentSuggestedPage == 0 ? 1 : currentSuggestedPage, + SUGGESTED_PAGE_SIZE); + } + + private Map buildContentOptions() { + return Lbry.buildClaimSearchOptions( + Claim.TYPE_STREAM, + null, + null, + getChannelIds(), + null, + getContentSortOrder(), + contentReleaseTime, + currentClaimSearchPage == 0 ? 1 : currentClaimSearchPage, + Helper.CONTENT_PAGE_SIZE); + } + + private List getChannelIds() { + if (channelFilterListAdapter != null) { + Claim selected = channelFilterListAdapter.getSelectedItem(); + if (selected != null) { + return Arrays.asList(selected.getClaimId()); + } + } + + return channelIds; + } + + private List getContentSortOrder() { + if (contentSortOrder == null) { + return Arrays.asList(Claim.ORDER_BY_RELEASE_TIME); + } + return contentSortOrder; + } + + private void showSuggestedChannels() { + Helper.setViewText(titleView, R.string.find_channels_to_follow); + + Helper.setViewVisibility(horizontalChannelList, View.GONE); + Helper.setViewVisibility(contentList, View.GONE); + Helper.setViewVisibility(infoView, View.VISIBLE); + Helper.setViewVisibility(layoutSortContainer, View.GONE); + Helper.setViewVisibility(suggestedChannelGrid, View.VISIBLE); + Helper.setViewVisibility(suggestedDoneButton, View.VISIBLE); + + updateSuggestedDoneButtonText(); + } + + private void showSubscribedContent() { + Helper.setViewText(titleView, R.string.channels_you_follow); + + Helper.setViewVisibility(horizontalChannelList, View.VISIBLE); + Helper.setViewVisibility(contentList, View.VISIBLE); + Helper.setViewVisibility(infoView, View.GONE); + Helper.setViewVisibility(layoutSortContainer, View.VISIBLE); + Helper.setViewVisibility(suggestedChannelGrid, View.GONE); + Helper.setViewVisibility(suggestedDoneButton, View.GONE); + } + + private void buildChannelIdsAndUrls() { + channelIds = new ArrayList<>(); + channelUrls = new ArrayList<>(); + if (subscriptionsList != null) { + for (Subscription subscription : subscriptionsList) { + try { + String url = subscription.getUrl(); + LbryUri uri = LbryUri.parse(url); + String claimId = uri.getClaimId(); + channelIds.add(claimId); + channelUrls.add(url); + } catch (LbryUriException ex) { + // pass + } + } + } + } + + private void fetchAndResolveChannelList() { + buildChannelIdsAndUrls(); + if (channelIds.size() > 0) { + ResolveTask resolveSubscribedTask = new ResolveTask(channelUrls, Lbry.LBRY_TV_CONNECTION_STRING, channelListLoading, new ResolveTask.ResolveResultHandler() { + @Override + public void onSuccess(List claims) { + updateChannelFilterListAdapter(claims); + Lbryio.cacheResolvedSubscriptions = claims; + } + + @Override + public void onError(Exception error) { + fetchAndResolveChannelList(); + } + }); + resolveSubscribedTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + + fetchClaimSearchContent(); + } + } + + private View getLoadingView() { + return (contentListAdapter == null || contentListAdapter.getItemCount() == 0) ? bigContentLoading : contentLoading; + } + + private void updateChannelFilterListAdapter(List resolvedSubs) { + if (channelFilterListAdapter == null) { + channelFilterListAdapter = new ChannelFilterListAdapter(getContext()); + channelFilterListAdapter.setListener(new ChannelItemSelectionListener() { + @Override + public void onChannelItemSelected(Claim claim) { + if (contentClaimSearchTask != null && contentClaimSearchTask.getStatus() != AsyncTask.Status.FINISHED) { + contentClaimSearchTask.cancel(true); + } + if (contentListAdapter != null) { + contentListAdapter.clearItems(); + } + currentClaimSearchPage = 1; + contentClaimSearchLoading = false; + fetchClaimSearchContent(); + } + + @Override + public void onChannelItemDeselected(Claim claim) { + + } + + @Override + public void onChannelSelectionCleared() { + if (contentClaimSearchTask != null && contentClaimSearchTask.getStatus() != AsyncTask.Status.FINISHED) { + contentClaimSearchTask.cancel(true); + } + if (contentListAdapter != null) { + contentListAdapter.clearItems(); + } + currentClaimSearchPage = 1; + contentClaimSearchLoading = false; + fetchClaimSearchContent(); + } + }); + } + + if (horizontalChannelList != null && horizontalChannelList.getAdapter() == null) { + horizontalChannelList.setAdapter(channelFilterListAdapter); + } + channelFilterListAdapter.addClaims(resolvedSubs); + } + + private void fetchClaimSearchContent() { + fetchClaimSearchContent(false); + } + + private void fetchClaimSearchContent(boolean reset) { + if (reset && contentListAdapter != null) { + contentListAdapter.clearItems(); + currentClaimSearchPage = 1; + } + + contentClaimSearchLoading = true; + Map claimSearchOptions = buildContentOptions(); + contentClaimSearchTask = new ClaimSearchTask(claimSearchOptions, Lbry.LBRY_TV_CONNECTION_STRING, getLoadingView(), new ClaimSearchTask.ClaimSearchResultHandler() { + @Override + public void onSuccess(List claims, boolean hasReachedEnd) { + if (contentListAdapter == null) { + contentListAdapter = new ClaimListAdapter(claims, getContext()); + contentListAdapter.setListener(new ClaimListAdapter.ClaimListItemListener() { + @Override + public void onClaimClicked(Claim claim) { + String claimId = claim.getClaimId(); + String url = claim.getPermanentUrl(); + + if (claim.getName().startsWith("@")) { + // channel claim + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openChannelClaim(claim); + } + } else { + Intent intent = new Intent(getContext(), FileViewActivity.class); + intent.putExtra("claimId", claimId); + intent.putExtra("url", url); + MainActivity.startingFileViewActivity = true; + startActivity(intent); + } + } + }); + } else { + contentListAdapter.addItems(claims); + } + + if (contentList != null && contentList.getAdapter() == null) { + contentList.setAdapter(contentListAdapter); + } + + contentHasReachedEnd = hasReachedEnd; + contentClaimSearchLoading = false; + } + + @Override + public void onError(Exception error) { + contentClaimSearchLoading = false; + } + }); + contentClaimSearchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void updateSuggestedDoneButtonText() { + int selected = suggestedChannelAdapter == null ? 0 : suggestedChannelAdapter.getSelectedCount(); + int remaining = MIN_SUGGESTED_SUBSCRIBE_COUNT - selected; + String buttonText = remaining <= 0 ? getString(R.string.done) : getString(R.string.n_remaining, remaining); + if (suggestedDoneButton != null) { + suggestedDoneButton.setText(buttonText); + } + } + + // handler methods + public void onSuccess(List subscriptions) { + if (subscriptions.size() == 0) { + // fresh start + // TODO: Only do this if there are no local subscriptions stored + currentSuggestedPage = 1; + buildSuggestedOptions(); + loadingSuggested = true; + loadingContent = false; + + suggestedChannelClaimSearchTask = new ClaimSearchTask( + buildSuggestedOptions(), + Lbry.LBRY_TV_CONNECTION_STRING, + suggestedChannelAdapter == null ? bigContentLoading : contentLoading, + new ClaimSearchTask.ClaimSearchResultHandler() { + @Override + public void onSuccess(List claims, boolean hasReachedEnd) { + if (suggestedChannelAdapter == null) { + suggestedChannelAdapter = new SuggestedChannelGridAdapter(claims, getContext()); + suggestedChannelAdapter.setListener(FollowingFragment.this); + if (suggestedChannelGrid != null) { + suggestedChannelGrid.setAdapter(suggestedChannelAdapter); + } + } else { + suggestedChannelAdapter.addClaims(claims); + } + } + + @Override + public void onError(Exception error) { + + } + }); + suggestedChannelClaimSearchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + showSuggestedChannels(); + } else { + Lbryio.cacheSubscriptions = subscriptions; + subscriptionsList = new ArrayList<>(subscriptions); + showSubscribedContent(); + fetchAndResolveChannelList(); + } + } + + public void onError(Exception exception) { + + } + + public void onChannelItemSelected(Claim claim) { + // subscribe + Subscription subscription = new Subscription(); + subscription.setChannelName(claim.getName()); + subscription.setUrl(claim.getPermanentUrl()); + String channelClaimId = claim.getClaimId(); + + ChannelSubscribeTask task = new ChannelSubscribeTask(getContext(), channelClaimId, subscription, false, null); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + updateSuggestedDoneButtonText(); + } + public void onChannelItemDeselected(Claim claim) { + // unsubscribe + Subscription subscription = new Subscription(); + subscription.setChannelName(claim.getName()); + subscription.setUrl(claim.getPermanentUrl()); + String channelClaimId = claim.getClaimId(); + + ChannelSubscribeTask task = new ChannelSubscribeTask(getContext(), channelClaimId, subscription, true, null); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + updateSuggestedDoneButtonText(); + } + public void onChannelSelectionCleared() { + + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/search/SearchFragment.java b/app/src/main/java/io/lbry/browser/ui/search/SearchFragment.java new file mode 100644 index 00000000..a344cf81 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/search/SearchFragment.java @@ -0,0 +1,230 @@ +package io.lbry.browser.ui.search; + +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.fragment.app.Fragment; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import java.util.List; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.ClaimListAdapter; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.ClaimCacheKey; +import io.lbry.browser.tasks.ClaimSearchTask; +import io.lbry.browser.tasks.LighthouseSearchTask; +import io.lbry.browser.tasks.ResolveTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryUri; +import lombok.Setter; + +public class SearchFragment extends BaseFragment implements ClaimListAdapter.ClaimListItemListener { + private ClaimListAdapter resultListAdapter; + private static final int PAGE_SIZE = 25; + + private ProgressBar loadingView; + private RecyclerView resultList; + private TextView noQueryView; + private TextView noResultsView; + + @Setter + private String currentQuery; + private boolean searchLoading; + private boolean contentHasReachedEnd; + private int currentFrom; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_search, container, false); + + loadingView = root.findViewById(R.id.search_loading); + noQueryView = root.findViewById(R.id.search_no_query); + noResultsView = root.findViewById(R.id.search_no_results); + + resultList = root.findViewById(R.id.search_result_list); + LinearLayoutManager llm = new LinearLayoutManager(getContext()); + resultList.setLayoutManager(llm); + resultList.setAdapter(resultListAdapter); + resultList.addOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { + if (searchLoading) { + return; + } + + LinearLayoutManager lm = (LinearLayoutManager) recyclerView.getLayoutManager(); + if (lm != null) { + int visibleItemCount = lm.getChildCount(); + int totalItemCount = lm.getItemCount(); + int pastVisibleItems = lm.findFirstVisibleItemPosition(); + if (pastVisibleItems + visibleItemCount >= totalItemCount) { + if (!contentHasReachedEnd) { + // load more + int newFrom = currentFrom + PAGE_SIZE; + search(currentQuery, newFrom); + } + } + } + } + }); + + return root; + } + + public void onResume() { + super.onResume(); + if (resultListAdapter == null || resultListAdapter.getItemCount() == 0) { + // new search + if (!Helper.isNullOrEmpty(currentQuery)) { + search(currentQuery, currentFrom); + } + } + if (Helper.isNullOrEmpty(currentQuery)) { + noQueryView.setVisibility(View.VISIBLE); + noResultsView.setVisibility(View.GONE); + } + } + + private boolean checkQuery(String query) { + if (!Helper.isNullOrEmpty(query) && !query.equalsIgnoreCase(currentQuery)) { + // new query, reset values + currentFrom = 0; + currentQuery = query; + if (resultListAdapter != null) { + resultListAdapter.clearItems(); + } + return true; + } + + return false; + } + + private Claim buildFeaturedItem(String query) { + Claim claim = new Claim(); + claim.setName(query); + claim.setFeatured(true); + claim.setUnresolved(true); + return claim; + } + + private String buildVanityUrl(String query) { + LbryUri url = new LbryUri(); + url.setClaimName(query); + return url.toString(); + } + + private void resolveFeaturedItem(String vanityUrl) { + final ClaimCacheKey key = new ClaimCacheKey(); + key.setVanityUrl(vanityUrl); + if (Lbry.claimCache.containsKey(key)) { + Claim cachedClaim = Lbry.claimCache.get(key); + updateFeaturedItemFromResolvedClaim(cachedClaim); + return; + } + + ResolveTask task = new ResolveTask(vanityUrl, Lbry.LBRY_TV_CONNECTION_STRING, null, new ResolveTask.ResolveResultHandler() { + @Override + public void onSuccess(List claims) { + if (claims.size() > 0) { + Claim resolved = claims.get(0); + Lbry.claimCache.put(key, resolved); + updateFeaturedItemFromResolvedClaim(resolved); + } + } + + @Override + public void onError(Exception error) { + + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void updateFeaturedItemFromResolvedClaim(Claim resolved) { + if (resultListAdapter != null) { + Claim unresolved = resultListAdapter.getFeaturedItem(); + + // only set the values we need + unresolved.setClaimId(resolved.getClaimId()); + unresolved.setName(resolved.getName()); + unresolved.setTimestamp(resolved.getTimestamp()); + unresolved.setValueType(resolved.getValueType()); + unresolved.setPermanentUrl(resolved.getPermanentUrl()); + unresolved.setValue(resolved.getValue()); + unresolved.setSigningChannel(resolved.getSigningChannel()); + unresolved.setUnresolved(false); + + resultListAdapter.notifyDataSetChanged(); + } + } + + public void search(String query, int from) { + boolean queryChanged = checkQuery(query); + if (!queryChanged && from > 0) { + currentFrom = from; + } + + searchLoading = true; + LighthouseSearchTask task = new LighthouseSearchTask(currentQuery, PAGE_SIZE, currentFrom, false, null, loadingView, new ClaimSearchTask.ClaimSearchResultHandler() { + @Override + public void onSuccess(List claims, boolean hasReachedEnd) { + contentHasReachedEnd = hasReachedEnd; + searchLoading = false; + + if (resultListAdapter == null) { + resultListAdapter = new ClaimListAdapter(claims, getContext()); + resultListAdapter.addFeaturedItem(buildFeaturedItem(query)); + resolveFeaturedItem(buildVanityUrl(query)); + resultListAdapter.setListener(SearchFragment.this); + if (resultList != null) { + resultList.setAdapter(resultListAdapter); + } + } else { + resultListAdapter.addItems(claims); + } + + int itemCount = resultListAdapter.getItemCount(); + noQueryView.setVisibility(View.GONE); + noResultsView.setVisibility(itemCount == 0 ? View.VISIBLE : View.GONE); + noResultsView.setText(getString(R.string.search_no_results, currentQuery)); + } + + @Override + public void onError(Exception error) { + int itemCount = resultListAdapter == null ? 0 : resultListAdapter.getItemCount(); + noQueryView.setVisibility(View.GONE); + noResultsView.setVisibility(itemCount == 0 ? View.VISIBLE : View.GONE); + noResultsView.setText(getString(R.string.search_no_results, currentQuery)); + searchLoading = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void onClaimClicked(Claim claim) { + if (Helper.isNullOrEmpty(claim.getName())) { + // never should happen, but if it does, do nothing + return; + } + + if (claim.isUnresolved()) { + // open the publish page + } else if (claim.getName().startsWith("@")) { + ((MainActivity) getContext()).openChannelUrl(claim.getPermanentUrl()); + } else { + // not a channel + MainActivity.openFileClaim(claim, getContext()); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/settings/SettingsFragment.java b/app/src/main/java/io/lbry/browser/ui/settings/SettingsFragment.java new file mode 100644 index 00000000..039eee06 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/settings/SettingsFragment.java @@ -0,0 +1,73 @@ +package io.lbry.browser.ui.settings; + +import android.content.Context; +import android.content.SharedPreferences; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; + +import androidx.appcompat.app.ActionBar; +import androidx.appcompat.app.AppCompatDelegate; +import androidx.preference.PreferenceFragmentCompat; +import androidx.preference.PreferenceManager; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; + +public class SettingsFragment extends PreferenceFragmentCompat implements SharedPreferences.OnSharedPreferenceChangeListener { + @Override + public void onCreatePreferences(Bundle savedInstanceState, String rootKey) { + setPreferencesFromResource(R.xml.settings, rootKey); + } + + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View view = super.onCreateView(inflater, container, savedInstanceState); + view.setBackgroundColor(getResources().getColor(R.color.pageBackground)); + + return view; + } + + @Override + public void onStart() { + super.onStart(); + MainActivity activity = (MainActivity) getContext(); + if (activity != null) { + activity.hideSearchBar(); + activity.showNavigationBackIcon(); + activity.lockDrawer(); + + ActionBar actionBar = activity.getSupportActionBar(); + if (actionBar != null) { + actionBar.setTitle(R.string.settings); + } + } + } + + @Override + public void onResume() { + super.onResume(); + PreferenceManager.getDefaultSharedPreferences(getContext()).registerOnSharedPreferenceChangeListener(this); + } + @Override + public void onPause() { + PreferenceManager.getDefaultSharedPreferences(getContext()).unregisterOnSharedPreferenceChangeListener(this); + super.onPause(); + } + @Override + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).restoreToggle(); + } + super.onStop(); + } + + public void onSharedPreferenceChanged(SharedPreferences sp, String key) { + if (key.equalsIgnoreCase(MainActivity.PREFERENCE_KEY_DARK_MODE)) { + boolean darkMode = sp.getBoolean(MainActivity.PREFERENCE_KEY_DARK_MODE, false); + AppCompatDelegate.setDefaultNightMode(darkMode ? AppCompatDelegate.MODE_NIGHT_YES : AppCompatDelegate.MODE_NIGHT_NO); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/verification/EmailVerificationFragment.java b/app/src/main/java/io/lbry/browser/ui/verification/EmailVerificationFragment.java new file mode 100644 index 00000000..55f9a4f3 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/verification/EmailVerificationFragment.java @@ -0,0 +1,197 @@ +package io.lbry.browser.ui.verification; + +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.fragment.app.Fragment; + +import com.google.android.material.button.MaterialButton; +import com.google.android.material.snackbar.Snackbar; +import com.google.android.material.textfield.TextInputEditText; +import com.google.android.material.textfield.TextInputLayout; + +import java.util.concurrent.Executors; +import java.util.concurrent.ScheduledExecutorService; +import java.util.concurrent.TimeUnit; + +import io.lbry.browser.R; +import io.lbry.browser.listener.SignInListener; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.tasks.verification.CheckUserEmailVerifiedTask; +import io.lbry.browser.tasks.verification.EmailNewTask; +import io.lbry.browser.tasks.verification.EmailResendTask; +import io.lbry.browser.utils.Helper; +import lombok.Setter; + +public class EmailVerificationFragment extends Fragment { + + @Setter + private SignInListener listener; + private View layoutCollect; + private View layoutVerify; + private ProgressBar emailAddProgress; + private TextView textAddedEmail; + private TextInputEditText inputEmail; + private TextInputLayout inputLayoutEmail; + private MaterialButton buttonContinue; + private MaterialButton buttonResend; + private View buttonEdit; + + private String currentEmail; + + private ScheduledExecutorService emailVerifyCheckScheduler; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_verification_email, container, false); + + layoutCollect = root.findViewById(R.id.verification_email_collect_container); + layoutVerify = root.findViewById(R.id.verification_email_verify_container); + inputEmail = root.findViewById(R.id.verification_email_input); + inputLayoutEmail = root.findViewById(R.id.verification_email_input_layout); + emailAddProgress = root.findViewById(R.id.verification_email_add_progress); + textAddedEmail = root.findViewById(R.id.verification_email_added_address); + buttonContinue = root.findViewById(R.id.verification_email_continue_button); + buttonResend = root.findViewById(R.id.verification_email_resend_button); + buttonEdit = root.findViewById(R.id.verification_email_edit_button); + + layoutCollect.setVisibility(View.VISIBLE); + + inputEmail.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + String layoutHint = !hasFocus ? "" : getString(R.string.email); + inputLayoutEmail.setHint(layoutHint); + } + }); + buttonContinue.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + addEmail(); + } + }); + buttonEdit.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + editEmail(); + } + }); + buttonResend.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + resendEmail(); + } + }); + + return root; + } + + private void addEmail() { + currentEmail = Helper.getValue(inputEmail.getText()); + if (Helper.isNullOrEmpty(currentEmail) || currentEmail.indexOf("@") == -1) { + Snackbar.make(getView(), R.string.provide_valid_email, Snackbar.LENGTH_LONG).setBackgroundTint(getResources().getColor(R.color.red)).show(); + return; + } + + EmailNewTask task = new EmailNewTask(currentEmail, emailAddProgress, new EmailNewTask.EmailNewHandler() { + @Override + public void beforeStart() { + Helper.setViewVisibility(buttonContinue, View.INVISIBLE); + } + + @Override + public void onSuccess() { + layoutCollect.setVisibility(View.GONE); + layoutVerify.setVisibility(View.VISIBLE); + Helper.setViewText(textAddedEmail, currentEmail); + if (listener != null) { + listener.onEmailAdded(currentEmail); + } + scheduleEmailVerify(); + + Helper.setViewVisibility(buttonContinue, View.VISIBLE); + } + + @Override + public void onEmailExists() { + // TODO: Update wording based on email already existing + } + + @Override + public void onError(Exception error) { + Snackbar.make(getView(), error.getMessage(), Snackbar.LENGTH_LONG).setBackgroundTint(getResources().getColor(R.color.red)).show(); + Helper.setViewVisibility(buttonContinue, View.VISIBLE); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void scheduleEmailVerify() { + emailVerifyCheckScheduler = Executors.newSingleThreadScheduledExecutor(); + emailVerifyCheckScheduler.scheduleAtFixedRate(new Runnable() { + @Override + public void run() { + checkEmailVerified(); + } + }, 5, 5, TimeUnit.SECONDS); + } + + private void checkEmailVerified() { + CheckUserEmailVerifiedTask task = new CheckUserEmailVerifiedTask(new CheckUserEmailVerifiedTask.CheckUserEmailVerifiedHandler() { + @Override + public void onUserEmailVerified() { + if (listener != null) { + listener.onEmailVerified(); + } + layoutCollect.setVisibility(View.GONE); + layoutVerify.setVisibility(View.GONE); + if (emailVerifyCheckScheduler != null) { + emailVerifyCheckScheduler.shutdownNow(); + emailVerifyCheckScheduler = null; + } + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void editEmail() { + if (emailVerifyCheckScheduler != null) { + emailVerifyCheckScheduler.shutdownNow(); + emailVerifyCheckScheduler = null; + } + + if (listener != null) { + listener.onEmailEdit(); + } + layoutVerify.setVisibility(View.GONE); + layoutCollect.setVisibility(View.VISIBLE); + } + + private void resendEmail() { + EmailResendTask task = new EmailResendTask(currentEmail, null, new GenericTaskHandler() { + @Override + public void beforeStart() { + Helper.setViewEnabled(buttonResend, false); + } + + @Override + public void onSuccess() { + Snackbar.make(getView(), R.string.please_follow_instructions, Snackbar.LENGTH_LONG).show(); + Helper.setViewEnabled(buttonResend, true); + } + + @Override + public void onError(Exception error) { + Snackbar.make(getView(), error.getMessage(), Snackbar.LENGTH_LONG).setBackgroundTint(getResources().getColor(R.color.red)).show(); + Helper.setViewEnabled(buttonResend, true); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/verification/WalletVerificationFragment.java b/app/src/main/java/io/lbry/browser/ui/verification/WalletVerificationFragment.java new file mode 100644 index 00000000..06d13562 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/verification/WalletVerificationFragment.java @@ -0,0 +1,250 @@ +package io.lbry.browser.ui.verification; + +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.fragment.app.Fragment; + +import com.google.android.material.button.MaterialButton; +import com.google.android.material.snackbar.Snackbar; +import com.google.android.material.textfield.TextInputEditText; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.listener.WalletSyncListener; +import io.lbry.browser.model.WalletSync; +import io.lbry.browser.tasks.wallet.DefaultSyncTaskHandler; +import io.lbry.browser.tasks.wallet.SyncApplyTask; +import io.lbry.browser.tasks.wallet.SyncGetTask; +import io.lbry.browser.tasks.wallet.SyncSetTask; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.Lbryio; +import io.lbry.lbrysdk.Utils; +import lombok.Setter; + +public class WalletVerificationFragment extends Fragment { + + @Setter + private WalletSyncListener listener = null; + private ProgressBar loading; + private TextView textLoading; + private View inputArea; + private MaterialButton doneButton; + private TextInputEditText inputPassword; + private WalletSync currentWalletSync; + private boolean verificationStarted; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_verification_wallet, container, false); + + loading = root.findViewById(R.id.verification_wallet_loading_progress); + textLoading = root.findViewById(R.id.verification_wallet_loading_text); + inputArea = root.findViewById(R.id.verification_wallet_input_area); + doneButton = root.findViewById(R.id.verification_wallet_done_button); + inputPassword = root.findViewById(R.id.verification_wallet_password_input); + + doneButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + String password = Helper.getValue(inputPassword.getText()); + if (Helper.isNullOrEmpty(password)) { + showError(getString(R.string.please_enter_your_password)); + return; + } + if (listener != null) { + listener.onWalletSyncProcessing(); + } + processExistingWalletWithPassword(password); + } + }); + + loading.setVisibility(View.VISIBLE); + inputArea.setVisibility(View.GONE); + + return root; + } + + @Override + public void onResume() { + super.onResume(); + start(); + } + + public void start() { + if (verificationStarted) { + return; + } + if (listener != null) { + listener.onWalletSyncProcessing(); + } + + verificationStarted = true; + Helper.setViewVisibility(loading, View.VISIBLE); + Helper.setViewVisibility(textLoading, View.VISIBLE); + String password = Utils.getSecureValue(MainActivity.SECURE_VALUE_KEY_SAVED_PASSWORD, getContext(), Lbry.KEYSTORE); + // start verification process + SyncGetTask task = new SyncGetTask(password, false, null, new DefaultSyncTaskHandler() { + @Override + public void onSyncGetSuccess(WalletSync walletSync) { + currentWalletSync = walletSync; + processExistingWallet(walletSync); + } + + @Override + public void onSyncGetWalletNotFound() { + // no wallet found, get sync apply data and run the process + processNewWallet(); + } + @Override + public void onSyncGetError(Exception error) { + // try again + Helper.setViewVisibility(loading, View.GONE); + Helper.setViewText(textLoading, error.getMessage()); + showError(error.getMessage()); + if (listener != null) { + listener.onWalletSyncFailed(error); + } + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void processExistingWallet(WalletSync walletSync) { + // Try first sync apply + SyncApplyTask applyTask = new SyncApplyTask("", walletSync.getData(), null, new DefaultSyncTaskHandler() { + @Override + public void onSyncApplySuccess(String hash, String data) { + // check if local and remote hash are different, and then run sync set + Utils.setSecureValue(MainActivity.SECURE_VALUE_KEY_SAVED_PASSWORD, "", getContext(), Lbry.KEYSTORE); + if (!hash.equalsIgnoreCase(Lbryio.lastRemoteHash)) { + setSyncAfterApply(Lbryio.lastRemoteHash); + } + if (listener != null) { + listener.onWalletSyncEnabled(); + } + } + + @Override + public void onSyncApplyError(Exception error) { + // failed, request the user to enter a password + Helper.setViewVisibility(loading, View.GONE); + Helper.setViewVisibility(textLoading, View.GONE); + Helper.setViewVisibility(inputArea, View.VISIBLE); + if (listener != null) { + listener.onWalletSyncWaitingForInput(); + } + } + }); + applyTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void setSyncAfterApply(String oldHash) { + SyncApplyTask fetchTask = new SyncApplyTask(true, new DefaultSyncTaskHandler() { + @Override + public void onSyncApplySuccess(String hash, String data) { + SyncSetTask setTask = new SyncSetTask(oldHash, hash, data, null); + setTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + @Override + public void onSyncApplyError(Exception error) { + // pass + } + }); + fetchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void processExistingWalletWithPassword(String password) { + Helper.setViewVisibility(loading, View.VISIBLE); + Helper.setViewVisibility(textLoading, View.VISIBLE); + Helper.setViewVisibility(inputArea, View.GONE); + + if (currentWalletSync == null) { + showError(getString(R.string.wallet_sync_op_failed)); + Helper.setViewText(textLoading, R.string.wallet_sync_op_failed); + return; + } + + Helper.setViewText(textLoading, R.string.apply_wallet_data); + SyncApplyTask applyTask = new SyncApplyTask(password, currentWalletSync.getData(), null, new DefaultSyncTaskHandler() { + @Override + public void onSyncApplySuccess(String hash, String data) { + Utils.setSecureValue(MainActivity.SECURE_VALUE_KEY_SAVED_PASSWORD, password, getContext(), Lbry.KEYSTORE); + // check if local and remote hash are different, and then run sync set + if (!hash.equalsIgnoreCase(Lbryio.lastRemoteHash)) { + SyncSetTask setTask = new SyncSetTask(Lbryio.lastRemoteHash, hash, data, null); + setTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + if (listener != null) { + listener.onWalletSyncEnabled(); + } + } + + @Override + public void onSyncApplyError(Exception error) { + // failed, request the user to enter a password + showError(error.getMessage()); + Helper.setViewVisibility(loading, View.GONE); + Helper.setViewVisibility(textLoading, View.GONE); + Helper.setViewVisibility(inputArea, View.VISIBLE); + if (listener != null) { + listener.onWalletSyncWaitingForInput(); + } + } + }); + applyTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void processNewWallet() { + SyncApplyTask fetchTask = new SyncApplyTask(true, new DefaultSyncTaskHandler() { + @Override + public void onSyncApplySuccess(String hash, String data) { createNewRemoteSync(hash, data); } + @Override + public void onSyncApplyError(Exception error) { + showError(error.getMessage()); + Helper.setViewVisibility(loading, View.GONE); + Helper.setViewText(textLoading, R.string.wallet_sync_op_failed); + if (listener != null) { + listener.onWalletSyncFailed(error); + } + } + }); + fetchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void createNewRemoteSync(String hash, String data) { + SyncSetTask setTask = new SyncSetTask("", hash, data, new DefaultSyncTaskHandler() { + @Override + public void onSyncSetSuccess(String hash) { + Lbryio.lastRemoteHash = hash; + if (listener != null) { + listener.onWalletSyncEnabled(); + } + } + + @Override + public void onSyncSetError(Exception error) { + showError(error.getMessage()); + Helper.setViewVisibility(loading, View.GONE); + Helper.setViewText(textLoading, R.string.wallet_sync_op_failed); + if (listener != null) { + listener.onWalletSyncFailed(error); + } + } + }); + setTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void showError(String message) { + Snackbar.make(getView(), message, Snackbar.LENGTH_LONG).setBackgroundTint( + getResources().getColor(R.color.red) + ).show(); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/wallet/WalletFragment.java b/app/src/main/java/io/lbry/browser/ui/wallet/WalletFragment.java new file mode 100644 index 00000000..5b9aeb95 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/wallet/WalletFragment.java @@ -0,0 +1,471 @@ +package io.lbry.browser.ui.wallet; + +import android.content.ClipData; +import android.content.ClipboardManager; +import android.content.Context; +import android.content.SharedPreferences; +import android.gesture.Gesture; +import android.os.AsyncTask; +import android.os.Bundle; +import android.text.method.LinkMovementMethod; +import android.view.GestureDetector; +import android.view.LayoutInflater; +import android.view.MotionEvent; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ImageButton; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.core.text.HtmlCompat; +import androidx.fragment.app.Fragment; +import androidx.preference.PreferenceManager; +import androidx.recyclerview.widget.DividerItemDecoration; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import com.google.android.material.button.MaterialButton; +import com.google.android.material.snackbar.Snackbar; +import com.google.android.material.switchmaterial.SwitchMaterial; +import com.google.android.material.textfield.TextInputEditText; + +import java.text.DecimalFormat; +import java.util.List; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.TransactionListAdapter; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.listener.WalletBalanceListener; +import io.lbry.browser.model.Transaction; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.tasks.wallet.TransactionListTask; +import io.lbry.browser.tasks.wallet.WalletAddressUnusedTask; +import io.lbry.browser.tasks.wallet.WalletSendTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; + +public class WalletFragment extends BaseFragment implements SdkStatusListener, WalletBalanceListener { + + private boolean sdkReady; + + private View layoutAccountRecommended; + private View layoutSdkInitializing; + private View linkSkipAccount; + private TextView textWalletBalance; + private TextView textWalletBalanceUSD; + private TextView textTipsBalance; + private TextView textTipsBalanceUSD; + private TextView textClaimsBalance; + private TextView textSupportsBalance; + private ProgressBar walletSendProgress; + + private View loadingRecentContainer; + private View inlineBalanceContainer; + private TextView textWalletInlineBalance; + private MaterialButton buttonSignUp; + private RecyclerView recentTransactionsList; + private View linkViewAll; + private TextView textConvertCredits; + private TextView textConvertCreditsBittrex; + private TextView textEarnMoreTips; + private TextView textWhatSyncMeans; + private TextView textWalletReceiveAddress; + private TextView textWalletHintSyncStatus; + private ImageButton buttonCopyReceiveAddress; + private MaterialButton buttonGetNewAddress; + private TextInputEditText inputSendAddress; + private TextInputEditText inputSendAmount; + private MaterialButton buttonSend; + private TextView textConnectedEmail; + private SwitchMaterial switchSyncStatus; + private TextView linkManualBackup; + private TextView linkSyncFAQ; + private TextView textNoRecentTransactions; + + private boolean hasFetchedRecentTransactions = false; + private TransactionListAdapter recentTransactionsAdapter; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_wallet, container, false); + + loadingRecentContainer = root.findViewById(R.id.wallet_loading_recent_container); + layoutAccountRecommended = root.findViewById(R.id.wallet_account_recommended_container); + layoutSdkInitializing = root.findViewById(R.id.wallet_sdk_initializing_container); + linkSkipAccount = root.findViewById(R.id.wallet_skip_account_link); + buttonSignUp = root.findViewById(R.id.wallet_sign_up_button); + + inlineBalanceContainer = root.findViewById(R.id.wallet_inline_balance_container); + textWalletInlineBalance = root.findViewById(R.id.wallet_inline_balance_value); + walletSendProgress = root.findViewById(R.id.wallet_send_progress); + + textWalletBalance = root.findViewById(R.id.wallet_balance_value); + textWalletBalanceUSD = root.findViewById(R.id.wallet_balance_usd_value); + textTipsBalance = root.findViewById(R.id.wallet_balance_tips); + textTipsBalanceUSD = root.findViewById(R.id.wallet_balance_tips_usd_value); + textClaimsBalance = root.findViewById(R.id.wallet_balance_staked_publishes); + textSupportsBalance = root.findViewById(R.id.wallet_balance_staked_supports); + textWalletHintSyncStatus = root.findViewById(R.id.wallet_hint_sync_status); + + recentTransactionsList = root.findViewById(R.id.wallet_recent_transactions_list); + linkViewAll = root.findViewById(R.id.wallet_link_view_all); + textNoRecentTransactions = root.findViewById(R.id.wallet_no_recent_transactions); + textConvertCredits = root.findViewById(R.id.wallet_hint_convert_credits); + textConvertCreditsBittrex = root.findViewById(R.id.wallet_hint_convert_credits_bittrex); + textEarnMoreTips = root.findViewById(R.id.wallet_hint_earn_more_tips); + textWhatSyncMeans = root.findViewById(R.id.wallet_hint_what_sync_means); + textWalletReceiveAddress = root.findViewById(R.id.wallet_receive_address); + buttonCopyReceiveAddress = root.findViewById(R.id.wallet_copy_receive_address); + buttonGetNewAddress = root.findViewById(R.id.wallet_get_new_address); + inputSendAddress = root.findViewById(R.id.wallet_input_send_address); + inputSendAmount = root.findViewById(R.id.wallet_input_amount); + buttonSend = root.findViewById(R.id.wallet_send); + textConnectedEmail = root.findViewById(R.id.wallet_connected_email); + switchSyncStatus = root.findViewById(R.id.wallet_switch_sync_status); + linkManualBackup = root.findViewById(R.id.wallet_link_manual_backup); + linkSyncFAQ = root.findViewById(R.id.wallet_link_sync_faq); + + initUi(); + + return root; + } + + private void copyReceiveAddress() { + Context context = getContext(); + if (context != null && textWalletReceiveAddress != null) { + ClipboardManager clipboard = (ClipboardManager) context.getSystemService(Context.CLIPBOARD_SERVICE); + ClipData data = ClipData.newPlainText("address", textWalletReceiveAddress.getText()); + clipboard.setPrimaryClip(data); + } + Snackbar.make(getView(), R.string.address_copied, Snackbar.LENGTH_SHORT).show(); + } + + private void fetchRecentTransactions() { + if (hasFetchedRecentTransactions) { + return; + } + + Helper.setViewVisibility(textNoRecentTransactions, View.GONE); + TransactionListTask task = new TransactionListTask(1, 5, loadingRecentContainer, new TransactionListTask.TransactionListHandler() { + @Override + public void onSuccess(List transactions, boolean hasReachedEnd) { + hasFetchedRecentTransactions = true; + recentTransactionsAdapter = new TransactionListAdapter(transactions, getContext()); + recentTransactionsList.setAdapter(recentTransactionsAdapter); + displayNoRecentTransactions(); + } + + @Override + public void onError(Exception error) { + hasFetchedRecentTransactions = true; + displayNoRecentTransactions(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void displayNoRecentTransactions() { + boolean showNoTransactionsView = hasFetchedRecentTransactions && + (recentTransactionsAdapter == null || recentTransactionsAdapter.getItemCount() == 0); + Helper.setViewVisibility(textNoRecentTransactions, showNoTransactionsView ? View.VISIBLE : View.GONE); + } + + private boolean validateSend() { + String recipientAddress = Helper.getValue(inputSendAddress.getText()); + String amountString = Helper.getValue(inputSendAmount.getText()); + if (!recipientAddress.matches(LbryUri.REGEX_ADDRESS)) { + Snackbar.make(getView(), R.string.invalid_recipient_address, Snackbar.LENGTH_LONG). + setBackgroundTint(getResources().getColor(R.color.red)).show(); + return false; + } + + if (!Helper.isNullOrEmpty(amountString)) { + try { + double amountValue = Double.valueOf(amountString); + double availableAmount = Lbry.walletBalance.getAvailable().doubleValue(); + if (availableAmount < amountValue) { + Snackbar.make(getView(), R.string.insufficient_balance, Snackbar.LENGTH_LONG). + setBackgroundTint(getResources().getColor(R.color.red)).show(); + return false; + } + } catch (NumberFormatException ex) { + // pass + Snackbar.make(getView(), R.string.invalid_amount, Snackbar.LENGTH_LONG). + setBackgroundTint(getResources().getColor(R.color.red)).show(); + return false; + } + } + + return true; + } + + @SuppressWarnings("ClickableViewAccessibility") + private void initUi() { + onWalletBalanceUpdated(Lbry.walletBalance); + + applyHtmlForTextView(textConvertCredits); + applyHtmlForTextView(textConvertCreditsBittrex); + applyHtmlForTextView(textWhatSyncMeans); + applyHtmlForTextView(linkManualBackup); + applyHtmlForTextView(linkSyncFAQ); + + Context context = getContext(); + LinearLayoutManager llm = new LinearLayoutManager(context); + recentTransactionsList.setLayoutManager(llm); + recentTransactionsList.addItemDecoration(new DividerItemDecoration(context, DividerItemDecoration.VERTICAL)); + + + buttonGetNewAddress.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + generateNewAddress(); + } + }); + textWalletReceiveAddress.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + copyReceiveAddress(); + } + }); + buttonCopyReceiveAddress.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + copyReceiveAddress(); + } + }); + buttonSend.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (validateSend()) { + sendCredits(); + } + } + }); + + inputSendAddress.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + inputSendAddress.setHint(hasFocus ? getString(R.string.recipient_address_placeholder) : ""); + } + }); + inputSendAmount.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + inputSendAmount.setHint(hasFocus ? getString(R.string.zero) : ""); + inlineBalanceContainer.setVisibility(hasFocus ? View.VISIBLE : View.INVISIBLE); + } + }); + + layoutSdkInitializing.setVisibility(Lbry.SDK_READY ? View.GONE : View.VISIBLE); + layoutAccountRecommended.setVisibility(hasSkippedAccount() || Lbryio.isSignedIn() ? View.GONE : View.VISIBLE); + linkSkipAccount.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + sp.edit().putBoolean(MainActivity.PREFERENCE_KEY_INTERNAL_SKIP_WALLET_ACCOUNT, true).apply(); + layoutAccountRecommended.setVisibility(View.GONE); + } + }); + + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + final boolean walletSyncEnabled = sp.getBoolean(MainActivity.PREFERENCE_KEY_INTERNAL_WALLET_SYNC_ENABLED, false); + switchSyncStatus.setChecked(walletSyncEnabled); + switchSyncStatus.setText(walletSyncEnabled ? R.string.on : R.string.off); + textWalletHintSyncStatus.setText(walletSyncEnabled ? R.string.backup_synced : R.string.backup_notsynced); + textConnectedEmail.setText(walletSyncEnabled ? Lbryio.getSignedInEmail() : null); + GestureDetector.SimpleOnGestureListener gestureListener = new GestureDetector.SimpleOnGestureListener() { + @Override + public boolean onSingleTapConfirmed(MotionEvent e) { + if (switchSyncStatus.isChecked()) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + sp.edit().putBoolean(MainActivity.PREFERENCE_KEY_INTERNAL_WALLET_SYNC_ENABLED, false).apply(); + switchSyncStatus.setText(R.string.off); + switchSyncStatus.setChecked(false); + } else { + // launch verification activity for wallet sync flow + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).walletSyncSignIn(); + } + } + return true; + } + }; + GestureDetector detector = new GestureDetector(getContext(), gestureListener); + + switchSyncStatus.setOnTouchListener(new View.OnTouchListener() { + @Override + public boolean onTouch(View view, MotionEvent motionEvent) { + detector.onTouchEvent(motionEvent); + return true; + } + }); + } + + public void onWalletSyncEnabled() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + sp.edit().putBoolean(MainActivity.PREFERENCE_KEY_INTERNAL_WALLET_SYNC_ENABLED, true).apply(); + switchSyncStatus.setText(R.string.on); + switchSyncStatus.setChecked(true); + textWalletHintSyncStatus.setText(R.string.backup_synced); + textConnectedEmail.setText(Lbryio.getSignedInEmail()); + fetchRecentTransactions(); + } + + private void disableSendControls() { + inputSendAddress.clearFocus(); + inputSendAmount.clearFocus(); + Helper.setViewEnabled(buttonSend, false); + Helper.setViewEnabled(inputSendAddress, false); + Helper.setViewEnabled(inputSendAmount, false); + } + + private void enableSendControls() { + Helper.setViewEnabled(buttonSend, true); + Helper.setViewEnabled(inputSendAddress, true); + Helper.setViewEnabled(inputSendAmount, true); + } + + private void sendCredits() { + // wallet_send task + String recipientAddress = Helper.getValue(inputSendAddress.getText()); + String amountString = Helper.getValue(inputSendAmount.getText()); + String amount = String.valueOf(Double.valueOf(amountString)); + + disableSendControls(); + WalletSendTask task = new WalletSendTask(recipientAddress, amount, walletSendProgress, new WalletSendTask.WalletSendHandler() { + @Override + public void onSuccess() { + String message = getResources().getQuantityString(R.plurals.you_sent_credits, Double.valueOf(amount) == 1.0 ? 1 : 2, amountString); + Snackbar.make(getView(), message, Snackbar.LENGTH_LONG).show(); + inputSendAddress.setText(null); + inputSendAmount.setText(null); + enableSendControls(); + } + + @Override + public void onError(Exception error) { + Snackbar.make(getView(), R.string.send_credit_error, Snackbar.LENGTH_LONG). + setBackgroundTint(getResources().getColor(R.color.red)).show(); + enableSendControls(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void checkReceiveAddress() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + String receiveAddress = sp.getString(MainActivity.PREFERENCE_KEY_INTERNAL_WALLET_RECEIVE_ADDRESS, null); + if (Helper.isNullOrEmpty(receiveAddress)) { + if (Lbry.SDK_READY) { + generateNewAddress(); + } + } else if (textWalletReceiveAddress != null) { + textWalletReceiveAddress.setText(receiveAddress); + } + } + + private static void applyHtmlForTextView(TextView textView) { + textView.setMovementMethod(LinkMovementMethod.getInstance()); + textView.setText(HtmlCompat.fromHtml(textView.getText().toString(), HtmlCompat.FROM_HTML_MODE_LEGACY)); + } + + private boolean hasSkippedAccount() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + return sp.getBoolean(MainActivity.PREFERENCE_KEY_INTERNAL_SKIP_WALLET_ACCOUNT, false); + } + + public void onResume() { + super.onResume(); + if (!Lbry.SDK_READY) { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.addSdkStatusListener(this); + activity.addWalletBalanceListener(this); + } + + checkReceiveAddress(); + } else { + onSdkReady(); + } + } + + public void onPause() { + hasFetchedRecentTransactions = false; + super.onPause(); + } + + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.removeWalletBalanceListener(this); + } + super.onStop(); + } + + public void onSdkReady() { + sdkReady = true; + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).removeSdkStatusListener(this); + } + + // update view + if (layoutSdkInitializing != null) { + layoutSdkInitializing.setVisibility(View.GONE); + } + + checkReceiveAddress(); + fetchRecentTransactions(); + } + + public void generateNewAddress() { + WalletAddressUnusedTask task = new WalletAddressUnusedTask(new WalletAddressUnusedTask.WalletAddressUnusedHandler() { + @Override + public void beforeStart() { + Helper.setViewEnabled(buttonGetNewAddress, false); + } + @Override + public void onSuccess(String newAddress) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + sp.edit().putString(MainActivity.PREFERENCE_KEY_INTERNAL_WALLET_RECEIVE_ADDRESS, newAddress).apply(); + Helper.setViewText(textWalletReceiveAddress, newAddress); + Helper.setViewEnabled(buttonGetNewAddress, true); + } + + @Override + public void onError(Exception error) { + Helper.setViewEnabled(buttonGetNewAddress, true); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private static final DecimalFormat LBC_CURRENCY_FORMAT = new DecimalFormat("#,###.##"); + private static final DecimalFormat USD_CURRENCY_FORMAT = new DecimalFormat("#,##0.00"); + + public void onWalletBalanceUpdated(WalletBalance walletBalance) { + double balance = walletBalance.getAvailable().doubleValue(); + double usdBalance = balance * Lbryio.LBCUSDRate; + double tipsBalance = walletBalance.getTips().doubleValue(); + double tipsUsdBalance = tipsBalance * Lbryio.LBCUSDRate; + + Helper.setViewText(textWalletBalance, LBC_CURRENCY_FORMAT.format(balance)); + Helper.setViewText(textTipsBalance, Helper.shortCurrencyFormat(tipsBalance)); + Helper.setViewText(textClaimsBalance, Helper.shortCurrencyFormat(walletBalance.getClaims().doubleValue())); + Helper.setViewText(textSupportsBalance, Helper.shortCurrencyFormat(walletBalance.getSupports().doubleValue())); + Helper.setViewText(textWalletInlineBalance, Helper.shortCurrencyFormat(balance)); + if (Lbryio.LBCUSDRate > 0) { + // only update display usd values if the rate is loaded + Helper.setViewText(textWalletBalanceUSD, String.format("≈$%s", USD_CURRENCY_FORMAT.format(usdBalance))); + Helper.setViewText(textTipsBalanceUSD, String.format("≈$%s", USD_CURRENCY_FORMAT.format(tipsUsdBalance))); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/utils/Events.java b/app/src/main/java/io/lbry/browser/utils/Events.java new file mode 100644 index 00000000..3b77d1e2 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/Events.java @@ -0,0 +1,5 @@ +package io.lbry.browser.utils; + +public class Events { + public static final String FIRST_RUN_COMPLETED = "first_run_completed"; +} diff --git a/app/src/main/java/io/lbry/browser/utils/Helper.java b/app/src/main/java/io/lbry/browser/utils/Helper.java new file mode 100644 index 00000000..6dd67d0b --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/Helper.java @@ -0,0 +1,286 @@ +package io.lbry.browser.utils; + +import android.content.BroadcastReceiver; +import android.content.Context; +import android.database.Cursor; +import android.database.sqlite.SQLiteDatabase; +import android.graphics.Color; +import android.graphics.drawable.ColorDrawable; +import android.graphics.drawable.Drawable; +import android.graphics.drawable.GradientDrawable; +import android.graphics.drawable.ShapeDrawable; +import android.view.View; +import android.widget.TextView; + +import androidx.core.content.ContextCompat; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.io.Closeable; +import java.io.IOException; +import java.text.DecimalFormat; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Calendar; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Random; + +import io.lbry.browser.dialog.ContentFromDialogFragment; +import io.lbry.browser.dialog.ContentSortDialogFragment; +import io.lbry.browser.model.Claim; +import okhttp3.MediaType; + +public final class Helper { + public static final String METHOD_GET = "GET"; + public static final String METHOD_POST = "POST"; + public static final MediaType FORM_MEDIA_TYPE = MediaType.parse("application/x-www-form-urlencoded"); + public static final MediaType JSON_MEDIA_TYPE = MediaType.get("application/json; charset=utf-8"); + public static final List MATURE_TAG_NAMES = Arrays.asList("mature", "nsfw", "porn", "xxx"); + public static final int CONTENT_PAGE_SIZE = 25; + + public static boolean isNull(String value) { + return value == null; + } + public static boolean isNullOrEmpty(String value) { + return value == null || value.trim().length() == 0; + } + + public static String capitalize(String value) { + StringBuilder sb = new StringBuilder(); + boolean capitalizeNext = true; + for (char c : value.toCharArray()) { + if (c == ' ') { + capitalizeNext = true; + sb.append(c); + } else { + if (capitalizeNext) { + sb.append(String.valueOf(c).toUpperCase()); + } else { + sb.append(c); + } + capitalizeNext = false; + } + } + return sb.toString(); + } + + public static String join(List list, String delimiter) { + StringBuilder sb = new StringBuilder(); + String delim = ""; + for (String s : list) { + sb.append(delim).append(s); + delim = delimiter; + } + return sb.toString(); + } + + public static JSONArray jsonArrayFromList(List values) { + JSONArray array = new JSONArray(); + for (T value : values) { + array.put(value); + } + return array; + } + + public static void unregisterReceiver(BroadcastReceiver receiver, Context context) { + if (receiver != null && context != null) { + context.unregisterReceiver(receiver); + } + } + + public static void setViewVisibility(View view, int visibility) { + if (view != null) { + view.setVisibility(visibility); + } + } + + public static void setViewText(TextView view, int stringResourceId) { + if (view != null) { + view.setText(stringResourceId); + } + } + + public static void setViewText(TextView view, String text) { + if (view != null) { + view.setText(text); + } + } + + public static void closeCursor(Cursor cursor) { + if (cursor != null && !cursor.isClosed()) { + cursor.close(); + } + } + + public static void closeDatabase(SQLiteDatabase db) { + if (db != null) { + db.close(); + } + } + + public static void closeCloseable(Closeable closeable) { + try { + if (closeable != null) { + closeable.close(); + } + } catch (IOException ex) { + // pass + } + } + + public static int parseInt(Object value, int defaultValue) { + try { + return Integer.parseInt(String.valueOf(value), 10); + } catch (NumberFormatException ex) { + return defaultValue; + } + } + + public static String formatDuration(long duration) { + long seconds = duration; + long hours = Double.valueOf(Math.floor(seconds / 3600.0)).longValue(); + seconds = duration - hours * 3600; + long minutes = Double.valueOf(Math.floor(seconds / 60.0)).longValue(); + seconds = seconds % 60; + + if (hours > 0) { + return String.format("%d:%02d:%02d", hours, minutes, seconds); + } + return String.format("%d:%02d", minutes, seconds); + } + + public static JSONObject getJSONObject(String name, JSONObject object) { + try { + return object.has(name) && !object.isNull(name) ? object.getJSONObject(name) : null; + } catch (JSONException ex) { + return null; + } + } + public static boolean getJSONBoolean(String name, boolean defaultValue, JSONObject object) { + try { + return object.has(name) && !object.isNull(name) ? object.getBoolean(name) : defaultValue; + } catch (JSONException ex) { + return defaultValue; + } + } + + public static String getJSONString(String name, String defaultValue, JSONObject object) { + try { + return object.has(name) && !object.isNull(name) ? object.getString(name) : defaultValue; + } catch (JSONException ex) { + return defaultValue; + } + } + + public static double getJSONDouble(String name, double defaultValue, JSONObject object) { + try { + return object.has(name) && !object.isNull(name) ? object.getDouble(name) : defaultValue; + } catch (JSONException ex) { + return defaultValue; + } + } + + + public static long getJSONLong(String name, long defaultValue, JSONObject object) { + try { + return object.has(name) && !object.isNull(name) ? object.getLong(name) : defaultValue; + } catch (JSONException ex) { + return defaultValue; + } + } + public static int getJSONInt(String name, int defaultValue, JSONObject object) { + try { + return object.has(name) && !object.isNull(name) ? object.getInt(name) : defaultValue; + } catch (JSONException ex) { + return defaultValue; + } + } + public static void setViewEnabled(View view, boolean enabled) { + if (view != null) { + view.setEnabled(enabled); + } + } + + public static String shortCurrencyFormat(double value) { + if (value > 1000000000.00) { + return String.format("%.1fB", value / 1000000000.0); + } + if (value > 1000000.0) { + return String.format("%.1fM", value / 1000000.0); + } + if (value > 1000.0) { + return String.format("%.1fK", value / 1000.0); + } + + if (value == 0) { + return "0"; + } + + return new DecimalFormat("###.##").format(value); + } + + public static String getValue(CharSequence cs) { + return cs != null ? cs.toString() : ""; + } + + public static List buildContentSortOrder(int sortBy) { + List sortOrder = new ArrayList<>(); + switch (sortBy) { + case ContentSortDialogFragment.ITEM_SORT_BY_NEW: + sortOrder = Arrays.asList(Claim.ORDER_BY_RELEASE_TIME); break; + case ContentSortDialogFragment.ITEM_SORT_BY_TOP: + sortOrder = Arrays.asList(Claim.ORDER_BY_EFFECTIVE_AMOUNT); break; + case ContentSortDialogFragment.ITEM_SORT_BY_TRENDING: + sortOrder = Arrays.asList(Claim.ORDER_BY_TRENDING_GROUP, Claim.ORDER_BY_TRENDING_MIXED); break; + } + + return sortOrder; + } + + public static String buildReleaseTime(int contentFrom) { + if (contentFrom == 0 || contentFrom == ContentFromDialogFragment.ITEM_FROM_ALL_TIME) { + return null; + } + + Calendar cal = Calendar.getInstance(); + switch (contentFrom) { + case ContentFromDialogFragment.ITEM_FROM_PAST_24_HOURS: cal.add(Calendar.HOUR_OF_DAY, -24) ; break; + case ContentFromDialogFragment.ITEM_FROM_PAST_WEEK: default: cal.add(Calendar.DAY_OF_YEAR, -7); break; + case ContentFromDialogFragment.ITEM_FROM_PAST_MONTH: cal.add(Calendar.MONTH, -1); break; + case ContentFromDialogFragment.ITEM_FROM_PAST_YEAR: cal.add(Calendar.YEAR, -1); break; + } + + return String.format(">%d", Double.valueOf(cal.getTimeInMillis() / 1000.0).longValue()); + } + + public static final Map randomColorMap = new HashMap<>(); + public static int generateRandomColorForValue(String value) { + if (Helper.isNullOrEmpty(value)) { + return 0; + } + + if (randomColorMap.containsKey(value)) { + return randomColorMap.get(value); + } + + Random random = new Random(value.hashCode()); + int color = Color.argb(255, random.nextInt(256), random.nextInt(256), random.nextInt(256)); + randomColorMap.put(value, color); + return color; + } + + public static void setIconViewBackgroundColor(View view, int color, boolean isPlaceholder, Context context) { + Drawable bg = view.getBackground(); + if (bg instanceof ShapeDrawable) { + ((ShapeDrawable) bg).getPaint().setColor(isPlaceholder ? ContextCompat.getColor(context, android.R.color.transparent) : color); + } else if (bg instanceof GradientDrawable) { + ((GradientDrawable) bg).setColor(isPlaceholder ? ContextCompat.getColor(context, android.R.color.transparent) : color); + } else if (bg instanceof ColorDrawable) { + ((ColorDrawable) bg).setColor(isPlaceholder ? ContextCompat.getColor(context, android.R.color.transparent) : color); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/utils/Lbry.java b/app/src/main/java/io/lbry/browser/utils/Lbry.java new file mode 100644 index 00000000..31cbe984 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/Lbry.java @@ -0,0 +1,412 @@ +package io.lbry.browser.utils; + +import com.google.gson.FieldNamingPolicy; +import com.google.gson.Gson; +import com.google.gson.GsonBuilder; +import com.google.gson.JsonArray; +import com.google.gson.JsonObject; +import com.google.gson.reflect.TypeToken; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.io.IOException; +import java.lang.reflect.Type; +import java.security.KeyStore; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Iterator; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; +import java.util.concurrent.TimeUnit; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.exceptions.LbryRequestException; +import io.lbry.browser.exceptions.LbryResponseException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.ClaimCacheKey; +import io.lbry.browser.model.File; +import io.lbry.browser.model.Transaction; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.model.lbryinc.User; +import io.lbry.lbrysdk.Utils; +import okhttp3.OkHttpClient; +import okhttp3.Request; +import okhttp3.RequestBody; +import okhttp3.Response; + +public final class Lbry { + public static final String TAG = "Lbry"; + public static LinkedHashMap claimCache = new LinkedHashMap<>(); + public static LinkedHashMap, List> claimSearchCache = new LinkedHashMap<>(); + public static WalletBalance walletBalance = new WalletBalance(); + + public static final String SDK_CONNECTION_STRING = "http://127.0.0.1:5279"; + public static final String LBRY_TV_CONNECTION_STRING = "https://api.lbry.tv/api/v1/proxy"; + + // Values to obtain from LBRY SDK status + public static boolean IS_STATUS_PARSED = false; // Check if the status has been parsed at least once + public static final String PLATFORM = String.format("Android %s (API %d)", Utils.getAndroidRelease(), Utils.getAndroidSdk()); + public static final String OS = "android"; + public static String INSTALLATION_ID = null; + public static String NODE_ID = null; + public static String DAEMON_VERSION = null; + + // JSON RPC API Call methods + public static final String METHOD_RESOLVE = "resolve"; + public static final String METHOD_CLAIM_SEARCH = "claim_search"; + public static final String METHOD_FILE_LIST = "file_list"; + public static final String METHOD_GET = "get"; + + public static final String METHOD_WALLET_BALANCE = "wallet_balance"; + public static final String METHOD_WALLET_ENCRYPT = "wallet_encrypt"; + public static final String METHOD_WALLET_DECRYPT = "wallet_decrypt"; + + public static final String METHOD_WALLET_LIST = "wallet_list"; + public static final String METHOD_WALLET_SEND = "wallet_send"; + public static final String METHOD_WALLET_STATUS = "wallet_status"; + public static final String METHOD_WALLET_UNLOCK = "wallet_unlock"; + public static final String METHOD_ADDRESS_IS_MINE = "address_is_mine"; + public static final String METHOD_ADDRESS_UNUSED = "address_unused"; + public static final String METHOD_ADDRESS_LIST = "address_list"; + public static final String METHOD_TRANSACTION_LIST = "transaction_list"; + public static final String METHOD_UTXO_RELEASE = "utxo_release"; + public static final String METHOD_SUPPORT_ABANDON = "support_abandon"; + public static final String METHOD_SYNC_HASH = "sync_hash"; + public static final String METHOD_SYNC_APPLY = "sync_apply"; + public static final String METHOD_PREFERENCE_GET = "preference_get"; + public static final String METHOD_PREFERENCE_SET = "preference_set"; + + + public static KeyStore KEYSTORE; + public static boolean SDK_READY = false; + + public static void parseStatus(String response) { + try { + JSONObject json = parseSdkResponse(response); + INSTALLATION_ID = json.getString("installation_id"); + if (json.has("lbry_id")) { + // if DHT is not enabled, lbry_id won't be set + NODE_ID = json.getString("lbry_id"); + } + IS_STATUS_PARSED = true; + } catch (JSONException | LbryResponseException ex) { + // pass + android.util.Log.e(TAG, "Could not parse status response.", ex); + } + } + + public static Response apiCall(String method) throws LbryRequestException { + return apiCall(method, null); + } + + public static Response apiCall(String method, Map params) throws LbryRequestException { + return apiCall(method, params, SDK_CONNECTION_STRING); + } + + public static Response apiCall(String method, Map params, String connectionString) throws LbryRequestException { + long counter = new Double(System.currentTimeMillis() / 1000.0).longValue(); + JSONObject requestParams = buildJsonParams(params); + JSONObject requestBody = new JSONObject(); + try { + requestBody.put("jsonrpc", "2.0"); + requestBody.put("method", method); + requestBody.put("params", requestParams); + requestBody.put("counter", counter); + } catch (JSONException ex) { + throw new LbryRequestException("Could not build the JSON request body.", ex); + } + + RequestBody body = RequestBody.create(requestBody.toString(), Helper.JSON_MEDIA_TYPE); + Request request = new Request.Builder().url(connectionString).post(body).build(); + OkHttpClient client = new OkHttpClient.Builder(). + writeTimeout(300, TimeUnit.SECONDS). + readTimeout(300, TimeUnit.SECONDS). + build(); + + try { + return client.newCall(request).execute(); + } catch (IOException ex) { + throw new LbryRequestException(String.format("\"%s\" method to %s failed", method, connectionString), ex); + } + } + + public static JSONObject buildJsonParams(Map params) { + JSONObject jsonParams = new JSONObject(); + if (params != null) { + try { + for (Map.Entry param : params.entrySet()) { + Object value = param.getValue(); + if (value instanceof List) { + value = Helper.jsonArrayFromList((List) value); + } + jsonParams.put(param.getKey(), value); + } + } catch (JSONException ex) { + // pass + } + } + + return jsonParams; + } + + public static Object parseResponse(Response response) throws LbryResponseException { + String responseString = null; + try { + responseString = response.body().string(); + JSONObject json = new JSONObject(responseString); + if (response.code() >= 200 && response.code() < 300) { + if (json.has("result")) { + Object result = json.get("result"); + return result; + } else { + processErrorJson(json); + } + } + + processErrorJson(json); + } catch (JSONException | IOException ex) { + throw new LbryResponseException(String.format("Could not parse response: %s", responseString), ex); + } + + return null; + } + + private static void processErrorJson(JSONObject json) throws JSONException, LbryResponseException { + if (json.has("error")) { + String errorMessage = null; + Object jsonError = json.get("error"); + if (jsonError instanceof String) { + errorMessage = jsonError.toString(); + } else { + errorMessage = ((JSONObject) jsonError).getString("message"); + } + throw new LbryResponseException(json.getString("error")); + } else { + throw new LbryResponseException("Protocol error with unknown response signature."); + } + } + + public static JSONObject parseSdkResponse(String responseString) throws LbryResponseException { + try { + JSONObject json = new JSONObject(responseString); + if (json.has("error")) { + String errorMessage = null; + Object jsonError = json.get("error"); + if (jsonError instanceof String) { + errorMessage = jsonError.toString(); + } else { + errorMessage = ((JSONObject) jsonError).getString("message"); + } + throw new LbryResponseException(json.getString("error")); + } + + return json; + } catch (JSONException ex) { + throw new LbryResponseException(String.format("Could not parse response: %s", responseString), ex); + } + } + + /** + * API Calls + */ + public static Claim resolve(String url, String connectionString) throws ApiCallException { + List results = resolve(Arrays.asList(url), connectionString); + return results.size() > 0 ? results.get(0) : null; + } + public static List resolve(List urls, String connectionString) throws ApiCallException { + List claims = new ArrayList<>(); + Map params = new HashMap<>(); + params.put("urls", urls); + try { + JSONObject result = (JSONObject) parseResponse(apiCall(METHOD_RESOLVE, params, connectionString)); + Iterator keys = result.keys(); + if (keys != null) { + while (keys.hasNext()) { + Claim claim = Claim.fromJSONObject(result.getJSONObject(keys.next())); + claims.add(claim); + + // TODO: Add/remove/prune entries claim cache management + ClaimCacheKey key = ClaimCacheKey.fromClaim(claim); + claimCache.put(key, claim); + } + } + } catch (LbryRequestException | LbryResponseException | JSONException ex) { + throw new ApiCallException("Could not execute resolve call", ex); + } + + return claims; + } + public static List transactionList(int page, int pageSize) throws ApiCallException { + List transactions = new ArrayList<>(); + Map params = new HashMap<>(); + if (page > 0) { + params.put("page", page); + } + if (pageSize > 0) { + params.put("page_size", pageSize); + } + try { + JSONObject result = (JSONObject) parseResponse(apiCall(METHOD_TRANSACTION_LIST, params, SDK_CONNECTION_STRING)); + JSONArray items = result.getJSONArray("items"); + for (int i = 0; i < items.length(); i++) { + Transaction tx = Transaction.fromJSONObject(items.getJSONObject(i)); + transactions.add(tx); + } + } catch (LbryRequestException | LbryResponseException | JSONException ex) { + throw new ApiCallException("Could not execute transaction_list call", ex); + } + + return transactions; + } + + public static File get(boolean saveFile) throws ApiCallException { + File file = null; + Map params = new HashMap<>(); + params.put("save_file", saveFile); + try { + JSONObject result = (JSONObject) parseResponse(apiCall(METHOD_GET, params)); + file = File.fromJSONObject(result); + + if (file != null) { + String fileClaimId = file.getClaimId(); + if (!Helper.isNullOrEmpty(fileClaimId)) { + ClaimCacheKey key = new ClaimCacheKey(); + key.setClaimId(fileClaimId); + if (claimCache.containsKey(key)) { + claimCache.get(key).setFile(file); + } + } + } + } catch (LbryRequestException | LbryResponseException ex) { + throw new ApiCallException("Could not execute resolve call", ex); + } + + return file; + } + + public static List fileList(String claimId) throws ApiCallException { + List files = new ArrayList<>(); + Map params = new HashMap<>(); + if (!Helper.isNullOrEmpty(claimId)) { + params.put("claim_id", claimId); + } + try { + JSONObject result = (JSONObject) parseResponse(apiCall(METHOD_FILE_LIST, params)); + JSONArray items = result.getJSONArray("items"); + for (int i = 0; i < items.length(); i++) { + JSONObject fileObject = items.getJSONObject(i); + File file = File.fromJSONObject(fileObject); + files.add(file); + + String fileClaimId = file.getClaimId(); + if (!Helper.isNullOrEmpty(fileClaimId)) { + ClaimCacheKey key = new ClaimCacheKey(); + key.setClaimId(fileClaimId); + if (claimCache.containsKey(key)) { + claimCache.get(key).setFile(file); + } + } + } + } catch (LbryRequestException | LbryResponseException | JSONException ex) { + throw new ApiCallException("Could not execute resolve call", ex); + } + + return files; + } + + private static final String[] listParamTypes = new String[] { + "any_tags", "channel_ids", "order_by", "not_tags", "not_channel_ids", "urls" + }; + + public static Map buildClaimSearchOptions( + String claimType, + List anyTags, + List notTags, + List channelIds, + List notChannelIds, + List orderBy, + String releaseTime, + int page, + int pageSize) { + Map options = new HashMap<>(); + if (!Helper.isNullOrEmpty(claimType)) { + options.put("claim_type", claimType); + } + options.put("no_totals", true); + options.put("page", page); + options.put("page_size", pageSize); + if (!Helper.isNullOrEmpty(releaseTime)) { + options.put("release_time", releaseTime); + } + + addClaimSearchListOption("any_tags", anyTags, options); + addClaimSearchListOption("not_tags", notTags, options); + addClaimSearchListOption("channel_ids", channelIds, options); + addClaimSearchListOption("not_channel_ids", notChannelIds, options); + addClaimSearchListOption("order_by", orderBy, options); + + return options; + } + + private static void addClaimSearchListOption(String key, List list, Map options) { + if (list != null && list.size() > 0) { + options.put(key, list); + } + } + + public static List claimSearch(Map options, String connectionString) throws ApiCallException { + if (claimSearchCache.containsKey(options)) { + return claimSearchCache.get(options); + } + + List claims = new ArrayList<>(); + try { + JSONObject result = (JSONObject) parseResponse(apiCall(METHOD_CLAIM_SEARCH, options, connectionString)); + JSONArray items = result.getJSONArray("items"); + if (items != null) { + for (int i = 0; i < items.length(); i++) { + Claim claim = Claim.fromJSONObject(items.getJSONObject(i)); + claims.add(claim); + + ClaimCacheKey key = ClaimCacheKey.fromClaim(claim); + claimCache.put(key, claim); + } + } + + claimSearchCache.put(options, claims); + } catch (LbryRequestException | LbryResponseException | JSONException ex) { + throw new ApiCallException("Could not execute resolve call", ex); + } + + return claims; + } + + public static Map buildSingleParam(String key, Object value) { + Map params = new HashMap<>(); + params.put(key, value); + return params; + } + + /** + * Call to return a generic JSONObject which can be further parsed as required + * @param method + * @param params + * @return + */ + public static Object genericApiCall(String method, Map params) throws ApiCallException { + Object response = null; + try { + response = parseResponse(apiCall(method, params)); + } catch (LbryRequestException | LbryResponseException ex) { + throw new ApiCallException(String.format("Could not execute %s call", method), ex); + } + return response; + } + public static Object genericApiCall(String method) throws ApiCallException { + return genericApiCall(method, null); + } +} diff --git a/app/src/main/java/io/lbry/browser/utils/LbryUri.java b/app/src/main/java/io/lbry/browser/utils/LbryUri.java new file mode 100644 index 00000000..5fb901b8 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/LbryUri.java @@ -0,0 +1,263 @@ +package io.lbry.browser.utils; + +import java.util.ArrayList; +import java.util.List; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +import io.lbry.browser.exceptions.LbryUriException; +import lombok.Data; + +@Data +public class LbryUri { + public static final String PROTO_DEFAULT = "lbry://"; + public static final String REGEX_INVALID_URI = "[ =&#:$@%?;/\\\\\"<>%\\{\\}|^~\\[\\]`\u0000-\u0008\u000b-\u000c\u000e-\u001F\uD800-\uDFFF\uFFFE-\uFFFF]"; + public static final String REGEX_ADDRESS = "^(b|r)(?=[^0OIl]{32,33})[0-9A-Za-z]{32,33}$"; + public static final int CHANNEL_NAME_MIN_LENGTH = 1; + public static final int CLAIM_ID_MAX_LENGTH = 40; + + private static final String REGEX_PART_PROTOCOL = "^((?:lbry://)?)"; + private static final String REGEX_PART_STREAM_OR_CHANNEL_NAME = "([^:$#/]*)"; + private static final String REGEX_PART_MODIFIER_SEPARATOR = "([:$#]?)([^/]*)"; + private static final String QUERY_STRING_BREAKER = "^([\\S]+)([?][\\S]*)"; + private static final Pattern PATTERN_SEPARATE_QUERY_STRING = Pattern.compile(QUERY_STRING_BREAKER); + + private String path; + private boolean isChannel; + private String streamName; + private String streamClaimId; + private String channelName; + private String channelClaimId; + private int primaryClaimSequence; + private int secondaryClaimSequence; + private int primaryBidPosition; + private int secondaryBidPosition; + + private String claimName; + private String claimId; + private String contentName; + private String queryString; + + public static LbryUri parse(String url) throws LbryUriException { + return parse(url, false); + } + public static LbryUri parse(String url, boolean requireProto) throws LbryUriException { + Pattern componentsPattern = Pattern.compile(String.format("%s%s%s(/?)%s%s", + REGEX_PART_PROTOCOL, + REGEX_PART_STREAM_OR_CHANNEL_NAME, + REGEX_PART_MODIFIER_SEPARATOR, + REGEX_PART_STREAM_OR_CHANNEL_NAME, + REGEX_PART_MODIFIER_SEPARATOR)); + + String cleanUrl = url, queryString = null; + Matcher qsMatcher = PATTERN_SEPARATE_QUERY_STRING.matcher(url); + if (qsMatcher.matches()) { + queryString = qsMatcher.group(2); + cleanUrl = !Helper.isNullOrEmpty(queryString) ? url.substring(0, url.indexOf(queryString)) : url; + if (queryString != null && queryString.length() > 0) { + queryString = queryString.substring(1); + } + } + + List components = new ArrayList<>(); + Matcher matcher = componentsPattern.matcher(cleanUrl); + if (matcher.matches()) { + // Note: For Java regex, group index 0 is always the full match + for (int i = 1; i <= matcher.groupCount(); i++) { + components.add(matcher.group(i)); + } + } + + // components[0] = proto + // components[1] = streamNameOrChannelName + // components[2] = primaryModSeparator + // components[3] = primaryModValue + // components[4] = pathSep + // components[5] = possibleStreamName + // components[6] = secondaryModSeparator + // components[7] = secondaryModValue + if (requireProto && Helper.isNullOrEmpty(components.get(0))) { + throw new LbryUriException("LBRY URLs must include a protocol prefix (lbry://)."); + } + + if (Helper.isNullOrEmpty(components.get(1))) { + throw new LbryUriException("URL does not include name."); + } + + for (String component : components.subList(1, components.size() - 1)) { + if (component.indexOf(' ') > -1) { + throw new LbryUriException("URL cannot include a space."); + } + } + + String streamOrChannelName = components.get(1); + String primaryModSeparator = components.get(2); + String primaryModValue = components.get(3); + String possibleStreamName = components.get(5); + String secondaryModSeparator = components.get(6); + String secondaryModValue = components.get(7); + + boolean includesChannel = streamOrChannelName.startsWith("@"); + boolean isChannel = includesChannel && Helper.isNullOrEmpty(possibleStreamName); + String channelName = includesChannel && streamOrChannelName.length() > 1 ? streamOrChannelName.substring(1) : null; + if (includesChannel) { + if (Helper.isNullOrEmpty(channelName)) { + throw new LbryUriException("No channel name after @."); + } + if (channelName.length() < CHANNEL_NAME_MIN_LENGTH) { + throw new LbryUriException(String.format("Channel names must be at least %d character long.", CHANNEL_NAME_MIN_LENGTH)); + } + } + + UriModifier primaryMod = null, secondaryMod = null; + if (!Helper.isNullOrEmpty(primaryModSeparator) && !Helper.isNullOrEmpty(primaryModValue)) { + primaryMod = UriModifier.parse(primaryModSeparator, primaryModValue); + } + if (!Helper.isNullOrEmpty(secondaryModSeparator) && !Helper.isNullOrEmpty(secondaryModValue)) { + secondaryMod = UriModifier.parse(secondaryModSeparator, secondaryModValue); + } + String streamName = includesChannel ? possibleStreamName : streamOrChannelName; + String streamClaimId = (includesChannel && secondaryMod != null) ? + secondaryMod.getClaimId() : primaryMod != null ? primaryMod.getClaimId() : null; + String channelClaimId = (includesChannel && primaryMod != null) ? primaryMod.getClaimId() : null; + + LbryUri uri = new LbryUri(); + uri.setChannel(isChannel); + uri.setPath(Helper.join(components.subList(1, components.size() - 1), "")); + uri.setStreamName(streamName); + uri.setStreamClaimId(streamClaimId); + uri.setChannelName(channelName); + uri.setChannelClaimId(channelClaimId); + uri.setPrimaryClaimSequence(primaryMod != null ? primaryMod.getClaimSequence() : -1); + uri.setSecondaryClaimSequence(secondaryMod != null ? secondaryMod.getClaimSequence() : -1); + uri.setPrimaryBidPosition(primaryMod != null ? primaryMod.getBidPosition() : -1); + uri.setSecondaryBidPosition(secondaryMod != null ? secondaryMod.getBidPosition() : -1); + + // Values that will not work properly with canonical urls + uri.setClaimName(streamOrChannelName); + uri.setClaimId(primaryMod != null ? primaryMod.getClaimId() : null); + uri.setContentName(streamName); + uri.setQueryString(queryString); + return uri; + } + + public String build(boolean includeProto, String protoDefault) { + String formattedChannelName = null; + if (channelName != null) { + formattedChannelName = channelName.startsWith("@") ? channelName : String.format("@%s", channelName); + } + String primaryClaimName = claimName; + if (Helper.isNullOrEmpty(primaryClaimName)) { + primaryClaimName = contentName; + } + if (Helper.isNullOrEmpty(primaryClaimName)) { + primaryClaimName = formattedChannelName; + } + if (Helper.isNullOrEmpty(primaryClaimName)) { + primaryClaimName = streamName; + } + + String primaryClaimId = claimId; + if (Helper.isNullOrEmpty(primaryClaimId)) { + primaryClaimId = !Helper.isNullOrEmpty(formattedChannelName) ? channelClaimId : streamClaimId; + } + + String secondaryClaimName = null; + if (Helper.isNullOrEmpty(claimName) && !Helper.isNullOrEmpty(contentName)) { + secondaryClaimName = contentName; + } + if (Helper.isNullOrEmpty(secondaryClaimName)) { + secondaryClaimName = !Helper.isNullOrEmpty(formattedChannelName) ? streamName : null; + } + String secondaryClaimId = !Helper.isNullOrEmpty(secondaryClaimName) ? streamClaimId : null; + + StringBuilder sb = new StringBuilder(); + if (includeProto) { + sb.append(protoDefault); + } + sb.append(primaryClaimName); + if (!Helper.isNullOrEmpty(primaryClaimId)) { + sb.append('#').append(primaryClaimId); + } + if (primaryClaimSequence > 0) { + sb.append(':').append(primaryClaimSequence); + } + if (primaryBidPosition > 0) { + sb.append('$').append(primaryBidPosition); + } + if (!Helper.isNullOrEmpty(secondaryClaimName)) { + sb.append('/').append(secondaryClaimName); + } + if (!Helper.isNullOrEmpty(secondaryClaimId)) { + sb.append('#').append(secondaryClaimId); + } + if (secondaryClaimSequence > 0) { + sb.append(':').append(secondaryClaimSequence); + } + if (secondaryBidPosition > 0) { + sb.append('$').append(secondaryBidPosition); + } + + return sb.toString(); + } + + public static String normalize(String url) throws LbryUriException { + return parse(url).toString(); + } + + public String toString() { + return build(true, PROTO_DEFAULT); + } + public int hashCode() { + return toString().hashCode(); + } + public boolean equals(Object o) { + if (o == null || !(o instanceof LbryUri)) { + return false; + } + return toString().equalsIgnoreCase(o.toString()); + } + + @Data + public static class UriModifier { + private String claimId; + private int claimSequence; + private int bidPosition; + + public UriModifier(String claimId, int claimSequence, int bidPosition) { + this.claimId = claimId; + this.claimSequence = claimSequence; + this.bidPosition = bidPosition; + } + + public static UriModifier parse(String modSeparator, String modValue) throws LbryUriException { + String claimId = null; + int claimSequence = 0, bidPosition = 0; + if (!Helper.isNullOrEmpty(modSeparator)) { + if (Helper.isNullOrEmpty(modValue)) { + throw new LbryUriException(String.format("No modifier provided after separator %s", modSeparator)); + } + + if ("#".equals(modSeparator)) { + claimId = modValue; + } else if (":".equals(modSeparator)) { + claimSequence = Helper.parseInt(modValue, -1); + } else if ("$".equals(modSeparator)) { + bidPosition = Helper.parseInt(modValue, -1); + } + } + + if (!Helper.isNullOrEmpty(claimId) && (claimId.length() > CLAIM_ID_MAX_LENGTH || !claimId.matches("^[0-9a-f]+$"))) { + throw new LbryUriException(String.format("Invalid claim ID %s", claimId)); + } + if (claimSequence == -1) { + throw new LbryUriException("Claim sequence must be a number"); + } + if (bidPosition == -1) { + throw new LbryUriException("Bid position must be a number"); + } + + return new UriModifier(claimId, claimSequence, bidPosition); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/utils/Lbryio.java b/app/src/main/java/io/lbry/browser/utils/Lbryio.java new file mode 100644 index 00000000..10d959f2 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/Lbryio.java @@ -0,0 +1,283 @@ +package io.lbry.browser.utils; + +import android.content.Context; +import android.content.Intent; +import android.content.pm.PackageInfo; +import android.content.pm.PackageManager; +import android.net.Uri; +import android.util.Log; + +import com.google.gson.FieldNamingPolicy; +import com.google.gson.Gson; +import com.google.gson.GsonBuilder; +import com.google.gson.reflect.TypeToken; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.io.IOException; +import java.io.UnsupportedEncodingException; +import java.lang.reflect.Type; +import java.net.URLEncoder; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.WalletSync; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.model.lbryinc.User; +import io.lbry.lbrysdk.Utils; +import lombok.Data; +import okhttp3.OkHttpClient; +import okhttp3.Request; +import okhttp3.RequestBody; +import okhttp3.Response; + +@Data +public final class Lbryio { + + public static User currentUser; + public static boolean userHasSyncedWallet = false; + public static String lastRemoteHash; + public static WalletSync lastWalletSync; + public static final Object lock = new Object(); + + public static final String TAG = "Lbryio"; + public static final String CONNECTION_STRING = "https://api.lbry.com"; + public static final String AUTH_TOKEN_PARAM = "auth_token"; + public static List cacheSubscriptions = new ArrayList<>(); + public static List cacheResolvedSubscriptions = new ArrayList<>(); + public static double LBCUSDRate = 0; + public static String AUTH_TOKEN; + private static boolean generatingAuthToken = false; + + public static Response call(String resource, String action, Context context) throws LbryioRequestException, LbryioResponseException { + return call(resource, action, null, Helper.METHOD_GET, context); + } + + public static Response call(String resource, String action, Map options, Context context) throws LbryioRequestException, LbryioResponseException { + return call(resource, action, options, Helper.METHOD_GET, context); + } + + public static Response call(String resource, String action, Map options, String method, Context context) + throws LbryioRequestException, LbryioResponseException { + String authToken = AUTH_TOKEN; + if (Helper.isNullOrEmpty(authToken) && !generatingAuthToken) { + // Only call getAuthToken if not calling /user/new + authToken = getAuthToken(context); + } + + String url = String.format("%s/%s/%s", CONNECTION_STRING, resource, action); + if (Helper.METHOD_GET.equalsIgnoreCase(method)) { + Uri.Builder uriBuilder = Uri.parse(url).buildUpon(); + if (!Helper.isNullOrEmpty(authToken)) { + uriBuilder.appendQueryParameter(AUTH_TOKEN_PARAM, authToken); + } + if (options != null) { + for (Map.Entry option : options.entrySet()) { + uriBuilder.appendQueryParameter(option.getKey(), option.getValue()); + } + } + url = uriBuilder.build().toString(); + } + + Request.Builder builder = new Request.Builder().url(url); + if (Helper.METHOD_POST.equalsIgnoreCase(method)) { + RequestBody body = RequestBody.create(buildQueryString(authToken, options), Helper.FORM_MEDIA_TYPE); + builder.post(body); + } + + Request request = builder.build(); + OkHttpClient client = new OkHttpClient(); + try { + return client.newCall(request).execute(); + } catch (IOException ex) { + throw new LbryioRequestException(String.format("%s request to %s/%s failed", method, resource, action), ex); + } + } + + private static String buildQueryString(String authToken, Map options) { + StringBuilder qs = new StringBuilder(); + try { + String delim = ""; + if (!Helper.isNullOrEmpty(authToken)) { + qs.append(AUTH_TOKEN_PARAM).append("=").append(URLEncoder.encode(authToken, "UTF8")); + delim = "&"; + } + + if (options != null) { + for (Map.Entry option : options.entrySet()) { + qs.append(delim).append(option.getKey()).append("=").append(URLEncoder.encode(option.getValue(), "UTF8")); + delim = "&"; + } + } + } catch (UnsupportedEncodingException ex) { + // pass + } + + return qs.toString(); + } + + public static String getAuthToken(Context context) throws LbryioRequestException, LbryioResponseException { + // fetch a new auth token + if (Helper.isNullOrEmpty(Lbry.INSTALLATION_ID)) { + throw new LbryioRequestException("The LBRY installation ID is not set."); + } + + generatingAuthToken = true; + + Map options = new HashMap<>(); + options.put("auth_token", ""); + options.put("language", "en"); + options.put("app_id", Lbry.INSTALLATION_ID); + Response response = Lbryio.call("user", "new", options, "post", context); + try { + JSONObject json = (JSONObject) parseResponse(response); + if (!json.has(AUTH_TOKEN_PARAM)) { + throw new LbryioResponseException("auth_token was not set in the response"); + } + + AUTH_TOKEN = json.getString(AUTH_TOKEN_PARAM); + broadcastAuthTokenGenerated(context); + } catch (JSONException | ClassCastException ex) { + throw new LbryioResponseException("auth_token was not set in the response", ex); + } finally { + generatingAuthToken = false; + } + + return AUTH_TOKEN; + } + + public static Object parseResponse(Response response) throws LbryioResponseException { + String responseString = null; + try { + responseString = response.body().string(); + JSONObject json = new JSONObject(responseString); + if (response.code() >= 200 && response.code() < 300) { + if (json.isNull("data")) { + return null; + } + return json.get("data"); + } + + if (json.has("error")) { + throw new LbryioResponseException(json.getString("error"), response.code()); + } else { + throw new LbryioResponseException("Unknown API error signature.", response.code()); + } + } catch (JSONException | IOException ex) { + throw new LbryioResponseException(String.format("Could not parse response: %s", responseString), ex); + } + } + + public static User fetchCurrentUser(Context context) { + try { + Response response = Lbryio.call("user", "me", context); + JSONObject object = (JSONObject) parseResponse(response); + Type type = new TypeToken(){}.getType(); + Gson gson = new GsonBuilder().setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES).create(); + User user = gson.fromJson(object.toString(), type); + return user; + } catch (LbryioRequestException | LbryioResponseException | ClassCastException ex) { + android.util.Log.e(TAG, "Cannot retrieve the current user", ex); + return null; + } + } + + public static void newInstall(Context context) { + String appVersion = ""; + if (context != null) { + try { + PackageManager manager = context.getPackageManager(); + PackageInfo info = manager.getPackageInfo(context.getPackageName(), PackageManager.GET_ACTIVITIES); + appVersion = info.versionName; + } catch (PackageManager.NameNotFoundException ex) { + + } + } + + Map options = new HashMap<>(); + //options.put("firebase_token", null); + options.put("app_version", appVersion); + options.put("app_id", Lbry.INSTALLATION_ID); + options.put("node_id", ""); + options.put("daemon_version", "0.67.1"); + options.put("operating_system", "android"); + options.put("platform", String.format("Android %s (API %d)", Utils.getAndroidRelease(), Utils.getAndroidSdk())); + try { + JSONObject response = (JSONObject) parseResponse(call("install", "new", options, Helper.METHOD_POST, context)); + } catch (LbryioRequestException | LbryioResponseException | ClassCastException ex) { + // pass + Log.e(TAG, String.format("install/new failed: %s", ex.getMessage()), ex); + } + } + + public static String getSignedInEmail() { + return currentUser != null ? currentUser.getPrimaryEmail() : ""; + } + + public static boolean isSignedIn() { + return currentUser != null && currentUser.isHasVerifiedEmail(); + } + + public static void authenticate(Context context) { + User user = fetchCurrentUser(context); + if (user != null) { + currentUser = user; + if (context != null) { + context.sendBroadcast(new Intent(MainActivity.ACTION_USER_AUTHENTICATION_SUCCESS)); + } + } else { + if (context != null) { + context.sendBroadcast(new Intent(MainActivity.ACTION_USER_AUTHENTICATION_FAILED)); + } + } + } + + public static void loadExchangeRate() { + try { + JSONObject response = (JSONObject) parseResponse(Lbryio.call("lbc", "exchange_rate", null)); + LBCUSDRate = Helper.getJSONDouble("lbc_usd", 0, response); + } catch (LbryioResponseException | LbryioRequestException | ClassCastException ex) { + + } + } + + private static void broadcastAuthTokenGenerated(Context context) { + try { + if (context != null) { + String encryptedAuthToken = Utils.encrypt(AUTH_TOKEN.getBytes("UTF8"), context, Lbry.KEYSTORE); + Intent intent = new Intent(MainActivity.ACTION_AUTH_TOKEN_GENERATED); + intent.putExtra("authToken", encryptedAuthToken); + context.sendBroadcast(intent); + } + } catch (Exception ex) { + android.util.Log.e(TAG, "Error sending encrypted auth token action broadcast", ex); + // pass + } + } + + public static Map buildSingleParam(String key, String value) { + Map params = new HashMap<>(); + params.put(key, value); + return params; + } + + public static void setLastWalletSync(WalletSync walletSync) { + synchronized (lock) { + lastWalletSync = walletSync; + } + } + + public static void setLastRemoteHash(String hash) { + synchronized (lock) { + lastRemoteHash = hash; + } + } +} diff --git a/app/src/main/java/io/lbry/browser/utils/Lighthouse.java b/app/src/main/java/io/lbry/browser/utils/Lighthouse.java new file mode 100644 index 00000000..be277d8d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/Lighthouse.java @@ -0,0 +1,100 @@ +package io.lbry.browser.utils; + +import android.net.Uri; + +import org.json.JSONArray; +import org.json.JSONException; + +import java.io.IOException; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.exceptions.LbryRequestException; +import io.lbry.browser.exceptions.LbryResponseException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.UrlSuggestion; +import okhttp3.OkHttpClient; +import okhttp3.Request; +import okhttp3.Response; + +public class Lighthouse { + public static final String CONNECTION_STRING = "https://lighthouse.lbry.com"; + public static Map> autocompleteCache = new HashMap<>(); + + public static List search(String rawQuery, int size, int from, boolean nsfw, String relatedTo) throws LbryRequestException, LbryResponseException { + Uri.Builder uriBuilder = Uri.parse(String.format("%s/search", CONNECTION_STRING)).buildUpon(). + appendQueryParameter("s", rawQuery). + appendQueryParameter("resolve", "true"). + appendQueryParameter("nsfw", String.valueOf(nsfw).toLowerCase()). + appendQueryParameter("size", String.valueOf(size)). + appendQueryParameter("from", String.valueOf(from)); + if (!Helper.isNullOrEmpty(relatedTo)) { + uriBuilder.appendQueryParameter("related_to", relatedTo); + } + + List results = new ArrayList<>(); + Request request = new Request.Builder().url(uriBuilder.toString()).build(); + OkHttpClient client = new OkHttpClient(); + try { + Response response = client.newCall(request).execute(); + if (response.code() == 200) { + JSONArray array = new JSONArray(response.body().string()); + for (int i = 0; i < array.length(); i++) { + Claim claim = Claim.fromSearchJSONObject(array.getJSONObject(i)); + results.add(claim); + } + } else { + throw new LbryResponseException(response.message()); + } + } catch (IOException ex) { + throw new LbryRequestException(String.format("search request for '%s' failed", rawQuery), ex); + } catch (JSONException ex) { + throw new LbryResponseException(String.format("the search response for '%s' could not be parsed", rawQuery), ex); + } + + return results; + } + + public static List autocomplete(String text) throws LbryRequestException, LbryResponseException { + if (autocompleteCache.containsKey(text)) { + return autocompleteCache.get(text); + } + + List suggestions = new ArrayList<>(); + Uri.Builder uriBuilder = Uri.parse(String.format("%s/autocomplete", CONNECTION_STRING)).buildUpon(). + appendQueryParameter("s", text); + Request request = new Request.Builder().url(uriBuilder.toString()).build(); + OkHttpClient client = new OkHttpClient(); + try { + Response response = client.newCall(request).execute(); + if (response.code() == 200) { + JSONArray array = new JSONArray(response.body().string()); + for (int i = 0; i < array.length(); i++) { + String item = array.getString(i); + boolean isChannel = item.startsWith("@"); + LbryUri uri = new LbryUri(); + if (isChannel) { + uri.setChannelName(item); + } else { + uri.setStreamName(item); + } + UrlSuggestion suggestion = new UrlSuggestion(isChannel ? UrlSuggestion.TYPE_CHANNEL : UrlSuggestion.TYPE_FILE, item); + suggestion.setUri(uri); + suggestions.add(suggestion); + } + + autocompleteCache.put(text, suggestions); + } else { + throw new LbryResponseException(response.message()); + } + } catch (IOException ex) { + throw new LbryRequestException(String.format("autocomplete request for '%s' failed", text), ex); + } catch (JSONException ex) { + throw new LbryResponseException(String.format("the autocomplete response for '%s' could not be parsed", text), ex); + } + + return suggestions; + } +} diff --git a/app/src/main/res/drawable-anydpi/ic_about.xml b/app/src/main/res/drawable-anydpi/ic_about.xml new file mode 100644 index 00000000..596d6b73 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_about.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_arrow_dropdown.xml b/app/src/main/res/drawable-anydpi/ic_arrow_dropdown.xml new file mode 100644 index 00000000..d7c4a022 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_arrow_dropdown.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_arrow_dropup.xml b/app/src/main/res/drawable-anydpi/ic_arrow_dropup.xml new file mode 100644 index 00000000..16f39fa3 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_arrow_dropup.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_check.xml b/app/src/main/res/drawable-anydpi/ic_check.xml new file mode 100644 index 00000000..eb477701 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_check.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_close.xml b/app/src/main/res/drawable-anydpi/ic_close.xml new file mode 100644 index 00000000..443f138a --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_close.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_copy.xml b/app/src/main/res/drawable-anydpi/ic_copy.xml new file mode 100644 index 00000000..b11c7d4e --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_copy.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_download.xml b/app/src/main/res/drawable-anydpi/ic_download.xml new file mode 100644 index 00000000..4c5fde21 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_download.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_editors_choice.xml b/app/src/main/res/drawable-anydpi/ic_editors_choice.xml new file mode 100644 index 00000000..dd3c6372 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_editors_choice.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_file.xml b/app/src/main/res/drawable-anydpi/ic_file.xml new file mode 100644 index 00000000..c2db6b21 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_file.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_following.xml b/app/src/main/res/drawable-anydpi/ic_following.xml new file mode 100644 index 00000000..22ed2aad --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_following.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_invites.xml b/app/src/main/res/drawable-anydpi/ic_invites.xml new file mode 100644 index 00000000..4bc8e0a7 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_invites.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_library.xml b/app/src/main/res/drawable-anydpi/ic_library.xml new file mode 100644 index 00000000..4c5fde21 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_library.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_lock.xml b/app/src/main/res/drawable-anydpi/ic_lock.xml new file mode 100644 index 00000000..b6145be4 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_lock.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_new.xml b/app/src/main/res/drawable-anydpi/ic_new.xml new file mode 100644 index 00000000..11109de3 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_new.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_publish.xml b/app/src/main/res/drawable-anydpi/ic_publish.xml new file mode 100644 index 00000000..faebc4a5 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_publish.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_publishes.xml b/app/src/main/res/drawable-anydpi/ic_publishes.xml new file mode 100644 index 00000000..dcb5ebd6 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_publishes.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_report.xml b/app/src/main/res/drawable-anydpi/ic_report.xml new file mode 100644 index 00000000..1fe6ead6 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_report.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_search.xml b/app/src/main/res/drawable-anydpi/ic_search.xml new file mode 100644 index 00000000..afb0429d --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_search.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_settings.xml b/app/src/main/res/drawable-anydpi/ic_settings.xml new file mode 100644 index 00000000..01b2fba0 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_settings.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_share.xml b/app/src/main/res/drawable-anydpi/ic_share.xml new file mode 100644 index 00000000..ba1ae20a --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_share.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_top.xml b/app/src/main/res/drawable-anydpi/ic_top.xml new file mode 100644 index 00000000..a6bbcd9b --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_top.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_trending.xml b/app/src/main/res/drawable-anydpi/ic_trending.xml new file mode 100644 index 00000000..0ba7252b --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_trending.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_wallet.xml b/app/src/main/res/drawable-anydpi/ic_wallet.xml new file mode 100644 index 00000000..722d0aa4 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_wallet.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-mdpi/src_assets_default_channel_cover.png b/app/src/main/res/drawable-hdpi/default_channel_cover.png similarity index 100% rename from app/src/main/res/drawable-mdpi/src_assets_default_channel_cover.png rename to app/src/main/res/drawable-hdpi/default_channel_cover.png diff --git a/app/src/main/res/drawable-mdpi/src_assets_gerbilhappy.png b/app/src/main/res/drawable-hdpi/gerbil_happy.png similarity index 100% rename from app/src/main/res/drawable-mdpi/src_assets_gerbilhappy.png rename to app/src/main/res/drawable-hdpi/gerbil_happy.png diff --git a/app/src/main/res/drawable-hdpi/gerbil_sad.png b/app/src/main/res/drawable-hdpi/gerbil_sad.png new file mode 100644 index 00000000..153d4adf Binary files /dev/null and b/app/src/main/res/drawable-hdpi/gerbil_sad.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_about.png b/app/src/main/res/drawable-hdpi/ic_about.png new file mode 100644 index 00000000..8613df07 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_about.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_arrow_dropdown.png b/app/src/main/res/drawable-hdpi/ic_arrow_dropdown.png new file mode 100644 index 00000000..f7e23b6d Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_arrow_dropdown.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_arrow_dropup.png b/app/src/main/res/drawable-hdpi/ic_arrow_dropup.png new file mode 100644 index 00000000..60113caa Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_arrow_dropup.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_check.png b/app/src/main/res/drawable-hdpi/ic_check.png new file mode 100644 index 00000000..92a31879 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_check.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_close.png b/app/src/main/res/drawable-hdpi/ic_close.png new file mode 100644 index 00000000..c661d153 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_close.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_copy.png b/app/src/main/res/drawable-hdpi/ic_copy.png new file mode 100644 index 00000000..7821875d Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_copy.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_download.png b/app/src/main/res/drawable-hdpi/ic_download.png new file mode 100644 index 00000000..6f4cf5db Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_download.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_editors_choice.png b/app/src/main/res/drawable-hdpi/ic_editors_choice.png new file mode 100644 index 00000000..bc2c85d3 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_editors_choice.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_file.png b/app/src/main/res/drawable-hdpi/ic_file.png new file mode 100644 index 00000000..a2b608dc Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_file.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_following.png b/app/src/main/res/drawable-hdpi/ic_following.png new file mode 100644 index 00000000..f39b95f1 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_following.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_invites.png b/app/src/main/res/drawable-hdpi/ic_invites.png new file mode 100644 index 00000000..07bab014 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_invites.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_launcher.png b/app/src/main/res/drawable-hdpi/ic_launcher.png deleted file mode 100644 index 6c7126b0..00000000 Binary files a/app/src/main/res/drawable-hdpi/ic_launcher.png and /dev/null differ diff --git a/app/src/main/res/drawable-hdpi/ic_lbry.png b/app/src/main/res/drawable-hdpi/ic_lbry.png deleted file mode 100644 index b23a7bea..00000000 Binary files a/app/src/main/res/drawable-hdpi/ic_lbry.png and /dev/null differ diff --git a/app/src/main/res/drawable-hdpi/ic_library.png b/app/src/main/res/drawable-hdpi/ic_library.png new file mode 100644 index 00000000..6f4cf5db Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_library.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_lock.png b/app/src/main/res/drawable-hdpi/ic_lock.png new file mode 100644 index 00000000..b5e83c73 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_lock.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_new.png b/app/src/main/res/drawable-hdpi/ic_new.png new file mode 100644 index 00000000..5e2d353e Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_new.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_publish.png b/app/src/main/res/drawable-hdpi/ic_publish.png new file mode 100644 index 00000000..6edf2fd8 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_publish.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_publishes.png b/app/src/main/res/drawable-hdpi/ic_publishes.png new file mode 100644 index 00000000..2dec2b6a Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_publishes.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_report.png b/app/src/main/res/drawable-hdpi/ic_report.png new file mode 100644 index 00000000..48967e15 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_report.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_search.png b/app/src/main/res/drawable-hdpi/ic_search.png new file mode 100644 index 00000000..f9c0afe1 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_search.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_settings.png b/app/src/main/res/drawable-hdpi/ic_settings.png new file mode 100644 index 00000000..a99894b8 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_settings.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_share.png b/app/src/main/res/drawable-hdpi/ic_share.png new file mode 100644 index 00000000..6131f566 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_share.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_top.png b/app/src/main/res/drawable-hdpi/ic_top.png new file mode 100644 index 00000000..0b8356af Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_top.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_trending.png b/app/src/main/res/drawable-hdpi/ic_trending.png new file mode 100644 index 00000000..546e723a Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_trending.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_wallet.png b/app/src/main/res/drawable-hdpi/ic_wallet.png new file mode 100644 index 00000000..7930303c Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_wallet.png differ diff --git a/app/src/main/res/drawable-hdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png b/app/src/main/res/drawable-hdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png deleted file mode 100644 index ad03a63b..00000000 Binary files a/app/src/main/res/drawable-hdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-hdpi/node_modules_reactnavigation_src_views_assets_backicon.png b/app/src/main/res/drawable-hdpi/node_modules_reactnavigation_src_views_assets_backicon.png deleted file mode 100644 index ad03a63b..00000000 Binary files a/app/src/main/res/drawable-hdpi/node_modules_reactnavigation_src_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-hdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png b/app/src/main/res/drawable-hdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png deleted file mode 100644 index ad03a63b..00000000 Binary files a/app/src/main/res/drawable-hdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-hdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png b/app/src/main/res/drawable-hdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png deleted file mode 100644 index ad03a63b..00000000 Binary files a/app/src/main/res/drawable-hdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-hdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png b/app/src/main/res/drawable-hdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png deleted file mode 100644 index ad03a63b..00000000 Binary files a/app/src/main/res/drawable-hdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xhdpi/src_assets_stripe.png b/app/src/main/res/drawable-hdpi/stripe_2x.png similarity index 100% rename from app/src/main/res/drawable-xhdpi/src_assets_stripe.png rename to app/src/main/res/drawable-hdpi/stripe_2x.png diff --git a/app/src/main/res/drawable-mdpi/default_channel_cover.png b/app/src/main/res/drawable-mdpi/default_channel_cover.png new file mode 100644 index 00000000..9b49120a Binary files /dev/null and b/app/src/main/res/drawable-mdpi/default_channel_cover.png differ diff --git a/app/src/main/res/drawable-mdpi/gerbil_happy.png b/app/src/main/res/drawable-mdpi/gerbil_happy.png new file mode 100644 index 00000000..4247f831 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/gerbil_happy.png differ diff --git a/app/src/main/res/drawable-mdpi/gerbil_sad.png b/app/src/main/res/drawable-mdpi/gerbil_sad.png new file mode 100644 index 00000000..153d4adf Binary files /dev/null and b/app/src/main/res/drawable-mdpi/gerbil_sad.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_about.png b/app/src/main/res/drawable-mdpi/ic_about.png new file mode 100644 index 00000000..dd2973a3 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_about.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_arrow_dropdown.png b/app/src/main/res/drawable-mdpi/ic_arrow_dropdown.png new file mode 100644 index 00000000..5cd1ef44 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_arrow_dropdown.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_arrow_dropup.png b/app/src/main/res/drawable-mdpi/ic_arrow_dropup.png new file mode 100644 index 00000000..aed6fb86 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_arrow_dropup.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_check.png b/app/src/main/res/drawable-mdpi/ic_check.png new file mode 100644 index 00000000..038f4e35 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_check.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_close.png b/app/src/main/res/drawable-mdpi/ic_close.png new file mode 100644 index 00000000..6060ea25 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_close.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_copy.png b/app/src/main/res/drawable-mdpi/ic_copy.png new file mode 100644 index 00000000..91230913 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_copy.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_download.png b/app/src/main/res/drawable-mdpi/ic_download.png new file mode 100644 index 00000000..03968835 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_download.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_editors_choice.png b/app/src/main/res/drawable-mdpi/ic_editors_choice.png new file mode 100644 index 00000000..2a6b26f5 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_editors_choice.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_file.png b/app/src/main/res/drawable-mdpi/ic_file.png new file mode 100644 index 00000000..1167c3d5 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_file.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_following.png b/app/src/main/res/drawable-mdpi/ic_following.png new file mode 100644 index 00000000..fac666b5 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_following.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_invites.png b/app/src/main/res/drawable-mdpi/ic_invites.png new file mode 100644 index 00000000..72ef5fa2 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_invites.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_launcher.png b/app/src/main/res/drawable-mdpi/ic_launcher.png deleted file mode 100644 index 542c83bc..00000000 Binary files a/app/src/main/res/drawable-mdpi/ic_launcher.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/ic_lbry.png b/app/src/main/res/drawable-mdpi/ic_lbry.png deleted file mode 100644 index f59cbefa..00000000 Binary files a/app/src/main/res/drawable-mdpi/ic_lbry.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/ic_library.png b/app/src/main/res/drawable-mdpi/ic_library.png new file mode 100644 index 00000000..03968835 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_library.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_lock.png b/app/src/main/res/drawable-mdpi/ic_lock.png new file mode 100644 index 00000000..93eabc9f Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_lock.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_new.png b/app/src/main/res/drawable-mdpi/ic_new.png new file mode 100644 index 00000000..0ed37115 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_new.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_publish.png b/app/src/main/res/drawable-mdpi/ic_publish.png new file mode 100644 index 00000000..c914b144 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_publish.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_publishes.png b/app/src/main/res/drawable-mdpi/ic_publishes.png new file mode 100644 index 00000000..1c07f3dd Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_publishes.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_report.png b/app/src/main/res/drawable-mdpi/ic_report.png new file mode 100644 index 00000000..7d8ca772 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_report.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_search.png b/app/src/main/res/drawable-mdpi/ic_search.png new file mode 100644 index 00000000..6fd5a13c Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_search.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_settings.png b/app/src/main/res/drawable-mdpi/ic_settings.png new file mode 100644 index 00000000..69c67dd5 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_settings.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_share.png b/app/src/main/res/drawable-mdpi/ic_share.png new file mode 100644 index 00000000..c9e997d0 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_share.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_top.png b/app/src/main/res/drawable-mdpi/ic_top.png new file mode 100644 index 00000000..50babf24 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_top.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_trending.png b/app/src/main/res/drawable-mdpi/ic_trending.png new file mode 100644 index 00000000..9945e6d0 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_trending.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_wallet.png b/app/src/main/res/drawable-mdpi/ic_wallet.png new file mode 100644 index 00000000..387d84e7 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_wallet.png differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativecountrypickermodal_lib_assets_images_close.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativecountrypickermodal_lib_assets_images_close.png deleted file mode 100644 index f8a96b2a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativecountrypickermodal_lib_assets_images_close.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativecountrypickermodal_lib_assets_images_closeios.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativecountrypickermodal_lib_assets_images_closeios.png deleted file mode 100644 index e64d614e..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativecountrypickermodal_lib_assets_images_closeios.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativecountrypickermodal_src_androidclose.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativecountrypickermodal_src_androidclose.png deleted file mode 100644 index f8a96b2a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativecountrypickermodal_src_androidclose.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativepasswordstrengthmeter_src_images_eyeinvisible.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativepasswordstrengthmeter_src_images_eyeinvisible.png deleted file mode 100644 index e7d95e0a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativepasswordstrengthmeter_src_images_eyeinvisible.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativepasswordstrengthmeter_src_images_eyevisible.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativepasswordstrengthmeter_src_images_eyevisible.png deleted file mode 100644 index 8588c6f3..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativepasswordstrengthmeter_src_images_eyevisible.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ad.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ad.png deleted file mode 100644 index 886752f6..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ad.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ae.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ae.png deleted file mode 100644 index a253cd2d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ae.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_af.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_af.png deleted file mode 100644 index 6ae08810..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_af.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ag.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ag.png deleted file mode 100644 index ee529d2b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ag.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ai.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ai.png deleted file mode 100644 index a598c566..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ai.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_al.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_al.png deleted file mode 100644 index 4b59dfbd..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_al.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_am.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_am.png deleted file mode 100644 index 41b497a3..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_am.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ao.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ao.png deleted file mode 100644 index f5ff2374..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ao.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ar.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ar.png deleted file mode 100644 index 0b25d9cb..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ar.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_as.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_as.png deleted file mode 100644 index 1b571007..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_as.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_at.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_at.png deleted file mode 100644 index 75646bab..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_at.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_au.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_au.png deleted file mode 100644 index f2572d72..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_au.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_aw.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_aw.png deleted file mode 100644 index a72bfddc..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_aw.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ax.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ax.png deleted file mode 100644 index d78ff2e3..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ax.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_az.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_az.png deleted file mode 100644 index f639aefd..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_az.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ba.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ba.png deleted file mode 100644 index a4ac356d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ba.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bb.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bb.png deleted file mode 100644 index 2bf58e69..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bb.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bd.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bd.png deleted file mode 100644 index e9872d15..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bd.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_be.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_be.png deleted file mode 100644 index 5d1b8325..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_be.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bf.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bf.png deleted file mode 100644 index 5172dbfb..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bf.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bg.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bg.png deleted file mode 100644 index d78308df..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bg.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bh.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bh.png deleted file mode 100644 index 5e247e7a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bh.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bi.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bi.png deleted file mode 100644 index 26186437..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bi.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bj.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bj.png deleted file mode 100644 index 20e281f2..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bj.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bl.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bl.png deleted file mode 100644 index a4938528..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bl.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bm.png deleted file mode 100644 index 29a8532a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bn.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bn.png deleted file mode 100644 index b4a3e60e..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bn.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bo.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bo.png deleted file mode 100644 index 342267cb..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bo.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bq.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bq.png deleted file mode 100644 index 0386cc3e..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bq.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_br.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_br.png deleted file mode 100644 index 43725657..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_br.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bs.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bs.png deleted file mode 100644 index 1bbb1d8f..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bs.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bt.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bt.png deleted file mode 100644 index cd4c8539..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bt.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bw.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bw.png deleted file mode 100644 index 555d80b3..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bw.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_by.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_by.png deleted file mode 100644 index 0dc31020..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_by.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bz.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bz.png deleted file mode 100644 index 3b6c39e6..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_bz.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ca.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ca.png deleted file mode 100644 index c939b041..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ca.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cc.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cc.png deleted file mode 100644 index fa52dba8..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cc.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cd.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cd.png deleted file mode 100644 index 44043fac..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cd.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cf.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cf.png deleted file mode 100644 index 5b7cb225..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cf.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cg.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cg.png deleted file mode 100644 index 2d7ce4c0..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cg.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ch.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ch.png deleted file mode 100644 index 5fe151ca..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ch.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ci.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ci.png deleted file mode 100644 index 0534124c..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ci.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ck.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ck.png deleted file mode 100644 index e45cb390..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ck.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cl.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cl.png deleted file mode 100644 index af74ffc9..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cl.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cm.png deleted file mode 100644 index b33c8115..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cn.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cn.png deleted file mode 100644 index d31bab71..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cn.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_co.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_co.png deleted file mode 100644 index b6aae55c..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_co.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cr.png deleted file mode 100644 index 9c92f6de..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cu.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cu.png deleted file mode 100644 index f21090e2..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cu.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cv.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cv.png deleted file mode 100644 index 6eeae62b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cv.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cw.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cw.png deleted file mode 100644 index c38132ce..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cw.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cx.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cx.png deleted file mode 100644 index 9a3e367c..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cx.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cy.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cy.png deleted file mode 100644 index 55446041..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cy.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cz.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cz.png deleted file mode 100644 index 4fc3adb5..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_cz.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_de.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_de.png deleted file mode 100644 index eea2e58b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_de.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dj.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dj.png deleted file mode 100644 index dbc95d77..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dj.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dk.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dk.png deleted file mode 100644 index e3471d34..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dk.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dm.png deleted file mode 100644 index a158c88f..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_do.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_do.png deleted file mode 100644 index 81fa5e8b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_do.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dz.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dz.png deleted file mode 100644 index b2768bcc..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_dz.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ec.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ec.png deleted file mode 100644 index 27fe8115..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ec.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ee.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ee.png deleted file mode 100644 index 21b4b72d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ee.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_eg.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_eg.png deleted file mode 100644 index d98e5d3a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_eg.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_eh.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_eh.png deleted file mode 100644 index cf451799..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_eh.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_er.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_er.png deleted file mode 100644 index 3f88fc52..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_er.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_es.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_es.png deleted file mode 100644 index f589a835..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_es.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_et.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_et.png deleted file mode 100644 index d759c2fd..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_et.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fi.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fi.png deleted file mode 100644 index 2bcb6a55..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fi.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fj.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fj.png deleted file mode 100644 index 7aef415f..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fj.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fk.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fk.png deleted file mode 100644 index 7b59f8c1..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fk.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fm.png deleted file mode 100644 index 1dfbdffe..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fo.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fo.png deleted file mode 100644 index e2ca151a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fo.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fr.png deleted file mode 100644 index fcfa7caf..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_fr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ga.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ga.png deleted file mode 100644 index 2dc5f0fc..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ga.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gb.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gb.png deleted file mode 100644 index f1e0e126..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gb.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gd.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gd.png deleted file mode 100644 index 5e3ed13b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gd.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ge.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ge.png deleted file mode 100644 index cd5b75de..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ge.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gf.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gf.png deleted file mode 100644 index fb15b809..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gf.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gg.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gg.png deleted file mode 100644 index 1cf113d4..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gg.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gh.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gh.png deleted file mode 100644 index a7b60ce7..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gh.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gi.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gi.png deleted file mode 100644 index 74fae09c..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gi.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gm.png deleted file mode 100644 index ca440bb6..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gn.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gn.png deleted file mode 100644 index 0740a3fc..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gn.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gp.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gp.png deleted file mode 100644 index 3223f1a3..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gp.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gq.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gq.png deleted file mode 100644 index bc9c8c46..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gq.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gr.png deleted file mode 100644 index ec65864a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gt.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gt.png deleted file mode 100644 index 3c7cee7d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gt.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gu.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gu.png deleted file mode 100644 index c7e586f3..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gu.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gw.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gw.png deleted file mode 100644 index 515d457f..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gw.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gy.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gy.png deleted file mode 100644 index 6c3e6733..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_gy.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hk.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hk.png deleted file mode 100644 index d7bbe5a2..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hk.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hn.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hn.png deleted file mode 100644 index ee1d1028..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hn.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hr.png deleted file mode 100644 index 2dae8a8a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ht.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ht.png deleted file mode 100644 index 2e15f899..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ht.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hu.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hu.png deleted file mode 100644 index c1c028ec..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_hu.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_id.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_id.png deleted file mode 100644 index 619215da..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_id.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ie.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ie.png deleted file mode 100644 index 3881ba34..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ie.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_il.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_il.png deleted file mode 100644 index 33fc90c2..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_il.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_im.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_im.png deleted file mode 100644 index a7a52cfa..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_im.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_in.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_in.png deleted file mode 100644 index 2f06567b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_in.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_io.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_io.png deleted file mode 100644 index 58a4b9be..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_io.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_iq.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_iq.png deleted file mode 100644 index 6b5eb22a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_iq.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ir.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ir.png deleted file mode 100644 index 36f7ec83..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ir.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_is.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_is.png deleted file mode 100644 index 74fef41d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_is.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_it.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_it.png deleted file mode 100644 index ff7ed317..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_it.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_je.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_je.png deleted file mode 100644 index dced1b0b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_je.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_jm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_jm.png deleted file mode 100644 index 68e58fee..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_jm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_jo.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_jo.png deleted file mode 100644 index 57bd76a6..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_jo.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_jp.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_jp.png deleted file mode 100644 index 33f3a757..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_jp.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ke.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ke.png deleted file mode 100644 index 9e8373fd..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ke.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kg.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kg.png deleted file mode 100644 index 3e7d6611..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kg.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kh.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kh.png deleted file mode 100644 index cf76786b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kh.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ki.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ki.png deleted file mode 100644 index ff8e470d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ki.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_km.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_km.png deleted file mode 100644 index cbd5e1b5..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_km.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kn.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kn.png deleted file mode 100644 index fed64fc0..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kn.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kp.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kp.png deleted file mode 100644 index b25aadc3..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kp.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kr.png deleted file mode 100644 index d035cab9..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ks.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ks.png deleted file mode 100644 index 942e1b58..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ks.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kw.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kw.png deleted file mode 100644 index 8c01668d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kw.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ky.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ky.png deleted file mode 100644 index 80bf785d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ky.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kz.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kz.png deleted file mode 100644 index 436ac8a1..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_kz.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_la.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_la.png deleted file mode 100644 index 87d7fb3c..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_la.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lb.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lb.png deleted file mode 100644 index 7d3659ab..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lb.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lc.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lc.png deleted file mode 100644 index 4bb0487c..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lc.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_li.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_li.png deleted file mode 100644 index b68b433a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_li.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lk.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lk.png deleted file mode 100644 index 15e45c81..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lk.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lr.png deleted file mode 100644 index 36948eef..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ls.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ls.png deleted file mode 100644 index 70cab723..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ls.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lt.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lt.png deleted file mode 100644 index 80bc5805..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lt.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lu.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lu.png deleted file mode 100644 index c5c2246c..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lu.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lv.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lv.png deleted file mode 100644 index 75431d19..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_lv.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ly.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ly.png deleted file mode 100644 index 2914da29..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ly.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ma.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ma.png deleted file mode 100644 index 0f751a1c..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ma.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mc.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mc.png deleted file mode 100644 index 3f8311b2..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mc.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_md.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_md.png deleted file mode 100644 index 4645ae10..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_md.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_me.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_me.png deleted file mode 100644 index 941d51d4..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_me.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mf.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mf.png deleted file mode 100644 index fcfa7caf..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mf.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mg.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mg.png deleted file mode 100644 index 43922054..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mg.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mh.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mh.png deleted file mode 100644 index 8438bfa3..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mh.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mk.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mk.png deleted file mode 100644 index 3c08615b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mk.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ml.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ml.png deleted file mode 100644 index ce81958a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ml.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mm.png deleted file mode 100644 index 3c1c0856..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mn.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mn.png deleted file mode 100644 index 2771b270..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mn.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mo.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mo.png deleted file mode 100644 index 2e62a9d8..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mo.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mp.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mp.png deleted file mode 100644 index 98ce37bd..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mp.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mq.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mq.png deleted file mode 100644 index 06466b3b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mq.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mr.png deleted file mode 100644 index f4dcf1d2..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ms.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ms.png deleted file mode 100644 index 163f5996..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ms.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mt.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mt.png deleted file mode 100644 index 950502ab..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mt.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mu.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mu.png deleted file mode 100644 index a6349637..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mu.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mv.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mv.png deleted file mode 100644 index 565a4083..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mv.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mw.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mw.png deleted file mode 100644 index 442dbc58..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mw.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mx.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mx.png deleted file mode 100644 index 666424d1..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mx.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_my.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_my.png deleted file mode 100644 index 215448cd..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_my.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mz.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mz.png deleted file mode 100644 index 18e2a949..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_mz.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_na.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_na.png deleted file mode 100644 index ca31b5d2..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_na.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nc.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nc.png deleted file mode 100644 index a55d0374..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nc.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ne.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ne.png deleted file mode 100644 index e0097297..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ne.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nf.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nf.png deleted file mode 100644 index 8a83dbf4..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nf.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ng.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ng.png deleted file mode 100644 index ee5775a8..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ng.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ni.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ni.png deleted file mode 100644 index 2ebe882a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ni.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nl.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nl.png deleted file mode 100644 index 0386cc3e..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nl.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_no.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_no.png deleted file mode 100644 index bb2f806b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_no.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_np.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_np.png deleted file mode 100644 index 726500cc..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_np.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nr.png deleted file mode 100644 index 65b58110..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nu.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nu.png deleted file mode 100644 index 4bc2ad23..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nu.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nz.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nz.png deleted file mode 100644 index abe4acf6..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_nz.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_om.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_om.png deleted file mode 100644 index 86812676..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_om.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pa.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pa.png deleted file mode 100644 index e821dee8..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pa.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pe.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pe.png deleted file mode 100644 index 5af51ad7..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pe.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pf.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pf.png deleted file mode 100644 index 4ecb31d9..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pf.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pg.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pg.png deleted file mode 100644 index 14818457..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pg.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ph.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ph.png deleted file mode 100644 index ffa33a92..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ph.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pk.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pk.png deleted file mode 100644 index 645971c5..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pk.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pl.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pl.png deleted file mode 100644 index 9d4e6925..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pl.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pm.png deleted file mode 100644 index 336cb210..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pr.png deleted file mode 100644 index 3fc7a074..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ps.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ps.png deleted file mode 100644 index ffc7621a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ps.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pt.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pt.png deleted file mode 100644 index 6526f8c1..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pt.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pw.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pw.png deleted file mode 100644 index 0a91ea56..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_pw.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_py.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_py.png deleted file mode 100644 index 40dffa49..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_py.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_qa.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_qa.png deleted file mode 100644 index 9cf00683..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_qa.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_re.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_re.png deleted file mode 100644 index 1dc648e7..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_re.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ro.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ro.png deleted file mode 100644 index 0bee8d1a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ro.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_rs.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_rs.png deleted file mode 100644 index 19fd38a6..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_rs.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ru.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ru.png deleted file mode 100644 index 66741a4d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ru.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_rw.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_rw.png deleted file mode 100644 index 24080d6d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_rw.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sa.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sa.png deleted file mode 100644 index 66dadb5b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sa.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sb.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sb.png deleted file mode 100644 index 97e0fc7c..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sb.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sc.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sc.png deleted file mode 100644 index 76863735..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sc.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sd.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sd.png deleted file mode 100644 index 9a6f886d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sd.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_se.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_se.png deleted file mode 100644 index 59595199..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_se.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sg.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sg.png deleted file mode 100644 index 8ba42209..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sg.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sh.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sh.png deleted file mode 100644 index d4c97406..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sh.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_si.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_si.png deleted file mode 100644 index 3b751344..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_si.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sj.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sj.png deleted file mode 100644 index bb2f806b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sj.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sk.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sk.png deleted file mode 100644 index 0769397a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sk.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sl.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sl.png deleted file mode 100644 index 96cddd4f..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sl.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sm.png deleted file mode 100644 index 4ee071c2..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sn.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sn.png deleted file mode 100644 index 9415c60e..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sn.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_so.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_so.png deleted file mode 100644 index 93a7fdc9..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_so.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sr.png deleted file mode 100644 index 47092d9e..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ss.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ss.png deleted file mode 100644 index e5f2259d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ss.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_st.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_st.png deleted file mode 100644 index 85f7d386..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_st.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sv.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sv.png deleted file mode 100644 index 97795729..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sv.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sx.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sx.png deleted file mode 100644 index ec17c244..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sx.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sy.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sy.png deleted file mode 100644 index a80b6b11..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sy.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sz.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sz.png deleted file mode 100644 index 89337677..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_sz.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tc.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tc.png deleted file mode 100644 index 5f5c2449..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tc.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_td.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_td.png deleted file mode 100644 index 41f123b5..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_td.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tg.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tg.png deleted file mode 100644 index a4a1d9f9..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tg.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_th.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_th.png deleted file mode 100644 index f0f7207d..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_th.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tj.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tj.png deleted file mode 100644 index 682b5e0f..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tj.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tk.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tk.png deleted file mode 100644 index 24b93302..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tk.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tl.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tl.png deleted file mode 100644 index 8a98e900..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tl.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tm.png deleted file mode 100644 index 58567c81..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tn.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tn.png deleted file mode 100644 index db4951a6..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tn.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_to.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_to.png deleted file mode 100644 index 95b78ce2..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_to.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tr.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tr.png deleted file mode 100644 index 95d0c871..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tr.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tt.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tt.png deleted file mode 100644 index 39a4af42..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tt.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tv.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tv.png deleted file mode 100644 index 6bfe412e..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tv.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tw.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tw.png deleted file mode 100644 index 80e07d81..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tw.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tz.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tz.png deleted file mode 100644 index 446ecb4f..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_tz.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ua.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ua.png deleted file mode 100644 index 00234794..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ua.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ug.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ug.png deleted file mode 100644 index cdcab6a1..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ug.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_us.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_us.png deleted file mode 100644 index 5b96ff24..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_us.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_uy.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_uy.png deleted file mode 100644 index 219ef44a..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_uy.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_uz.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_uz.png deleted file mode 100644 index 80e0a446..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_uz.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_va.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_va.png deleted file mode 100644 index c94c81dd..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_va.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vc.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vc.png deleted file mode 100644 index 77196ed9..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vc.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ve.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ve.png deleted file mode 100644 index 40ae68eb..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ve.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vg.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vg.png deleted file mode 100644 index 4de2078b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vg.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vi.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vi.png deleted file mode 100644 index a4bd67cb..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vi.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vn.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vn.png deleted file mode 100644 index d6838523..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vn.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vu.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vu.png deleted file mode 100644 index e1ad764e..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_vu.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_wf.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_wf.png deleted file mode 100644 index c3c5a9e8..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_wf.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ws.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ws.png deleted file mode 100644 index 71db01fa..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ws.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ye.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ye.png deleted file mode 100644 index 3a2e0a2b..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_ye.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_yt.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_yt.png deleted file mode 100644 index fcfa7caf..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_yt.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_za.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_za.png deleted file mode 100644 index 535fe710..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_za.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_zm.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_zm.png deleted file mode 100644 index 7b0246a8..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_zm.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_zw.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_zw.png deleted file mode 100644 index abf13869..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativephoneinput_lib_resources_flags_images_zw.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnativespeedometer_images_speedometerneedle.png b/app/src/main/res/drawable-mdpi/node_modules_reactnativespeedometer_images_speedometerneedle.png deleted file mode 100644 index 42373f85..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnativespeedometer_images_speedometerneedle.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png b/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png deleted file mode 100644 index 083db295..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backiconmask.png b/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backiconmask.png deleted file mode 100644 index dbddbdff..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backiconmask.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_src_views_assets_backicon.png b/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_src_views_assets_backicon.png deleted file mode 100644 index 083db295..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_src_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_src_views_assets_backiconmask.png b/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_src_views_assets_backiconmask.png deleted file mode 100644 index 5d7df0c0..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnavigation_src_views_assets_backiconmask.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png b/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png deleted file mode 100644 index 083db295..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_dist_views_assets_backiconmask.png b/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_dist_views_assets_backiconmask.png deleted file mode 100644 index 5fa299b7..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_dist_views_assets_backiconmask.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png b/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png deleted file mode 100644 index 083db295..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_lib_module_views_assets_backiconmask.png b/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_lib_module_views_assets_backiconmask.png deleted file mode 100644 index dbddbdff..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_lib_module_views_assets_backiconmask.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png b/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png deleted file mode 100644 index 083db295..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_src_views_assets_backiconmask.png b/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_src_views_assets_backiconmask.png deleted file mode 100644 index dbddbdff..00000000 Binary files a/app/src/main/res/drawable-mdpi/node_modules_reactnavigationstack_src_views_assets_backiconmask.png and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/src_assets_default_avatar.jpg b/app/src/main/res/drawable-mdpi/src_assets_default_avatar.jpg deleted file mode 100644 index 2cffd3af..00000000 Binary files a/app/src/main/res/drawable-mdpi/src_assets_default_avatar.jpg and /dev/null differ diff --git a/app/src/main/res/drawable-mdpi/stripe_2x.png b/app/src/main/res/drawable-mdpi/stripe_2x.png new file mode 100644 index 00000000..e377b672 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/stripe_2x.png differ diff --git a/app/src/main/res/drawable-v24/ic_launcher_foreground.xml b/app/src/main/res/drawable-v24/ic_launcher_foreground.xml new file mode 100644 index 00000000..2b068d11 --- /dev/null +++ b/app/src/main/res/drawable-v24/ic_launcher_foreground.xml @@ -0,0 +1,30 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable-xhdpi/default_channel_cover.png b/app/src/main/res/drawable-xhdpi/default_channel_cover.png new file mode 100644 index 00000000..9b49120a Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/default_channel_cover.png differ diff --git a/app/src/main/res/drawable-xhdpi/gerbil_happy.png b/app/src/main/res/drawable-xhdpi/gerbil_happy.png new file mode 100644 index 00000000..4247f831 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/gerbil_happy.png differ diff --git a/app/src/main/res/drawable-xhdpi/gerbil_sad.png b/app/src/main/res/drawable-xhdpi/gerbil_sad.png new file mode 100644 index 00000000..153d4adf Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/gerbil_sad.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_about.png b/app/src/main/res/drawable-xhdpi/ic_about.png new file mode 100644 index 00000000..6729f52e Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_about.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_arrow_dropdown.png b/app/src/main/res/drawable-xhdpi/ic_arrow_dropdown.png new file mode 100644 index 00000000..979ee5d4 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_arrow_dropdown.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_arrow_dropup.png b/app/src/main/res/drawable-xhdpi/ic_arrow_dropup.png new file mode 100644 index 00000000..0ea7d6db Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_arrow_dropup.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_check.png b/app/src/main/res/drawable-xhdpi/ic_check.png new file mode 100644 index 00000000..d9b2249a Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_check.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_close.png b/app/src/main/res/drawable-xhdpi/ic_close.png new file mode 100644 index 00000000..91f6c7a8 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_close.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_copy.png b/app/src/main/res/drawable-xhdpi/ic_copy.png new file mode 100644 index 00000000..a15eb3da Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_copy.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_download.png b/app/src/main/res/drawable-xhdpi/ic_download.png new file mode 100644 index 00000000..e311de88 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_download.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_editors_choice.png b/app/src/main/res/drawable-xhdpi/ic_editors_choice.png new file mode 100644 index 00000000..a955fba4 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_editors_choice.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_file.png b/app/src/main/res/drawable-xhdpi/ic_file.png new file mode 100644 index 00000000..15c258b0 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_file.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_following.png b/app/src/main/res/drawable-xhdpi/ic_following.png new file mode 100644 index 00000000..8209422e Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_following.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_invites.png b/app/src/main/res/drawable-xhdpi/ic_invites.png new file mode 100644 index 00000000..075db8d0 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_invites.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_launcher.png b/app/src/main/res/drawable-xhdpi/ic_launcher.png deleted file mode 100644 index df80b81c..00000000 Binary files a/app/src/main/res/drawable-xhdpi/ic_launcher.png and /dev/null differ diff --git a/app/src/main/res/drawable-xhdpi/ic_lbry.png b/app/src/main/res/drawable-xhdpi/ic_lbry.png deleted file mode 100644 index 872021ae..00000000 Binary files a/app/src/main/res/drawable-xhdpi/ic_lbry.png and /dev/null differ diff --git a/app/src/main/res/drawable-xhdpi/ic_library.png b/app/src/main/res/drawable-xhdpi/ic_library.png new file mode 100644 index 00000000..e311de88 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_library.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_lock.png b/app/src/main/res/drawable-xhdpi/ic_lock.png new file mode 100644 index 00000000..ecc037ca Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_lock.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_new.png b/app/src/main/res/drawable-xhdpi/ic_new.png new file mode 100644 index 00000000..9507ba46 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_new.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_publish.png b/app/src/main/res/drawable-xhdpi/ic_publish.png new file mode 100644 index 00000000..904b34ea Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_publish.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_publishes.png b/app/src/main/res/drawable-xhdpi/ic_publishes.png new file mode 100644 index 00000000..3d88ad33 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_publishes.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_report.png b/app/src/main/res/drawable-xhdpi/ic_report.png new file mode 100644 index 00000000..841d4ae2 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_report.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_search.png b/app/src/main/res/drawable-xhdpi/ic_search.png new file mode 100644 index 00000000..aad212cc Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_search.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_settings.png b/app/src/main/res/drawable-xhdpi/ic_settings.png new file mode 100644 index 00000000..3d88a9ea Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_settings.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_share.png b/app/src/main/res/drawable-xhdpi/ic_share.png new file mode 100644 index 00000000..c77d4eb3 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_share.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_top.png b/app/src/main/res/drawable-xhdpi/ic_top.png new file mode 100644 index 00000000..8789cd5f Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_top.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_trending.png b/app/src/main/res/drawable-xhdpi/ic_trending.png new file mode 100644 index 00000000..e7e6274b Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_trending.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_wallet.png b/app/src/main/res/drawable-xhdpi/ic_wallet.png new file mode 100644 index 00000000..69eba060 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_wallet.png differ diff --git a/app/src/main/res/drawable-xhdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png b/app/src/main/res/drawable-xhdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png deleted file mode 100644 index 6de0a1cb..00000000 Binary files a/app/src/main/res/drawable-xhdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xhdpi/node_modules_reactnavigation_src_views_assets_backicon.png b/app/src/main/res/drawable-xhdpi/node_modules_reactnavigation_src_views_assets_backicon.png deleted file mode 100644 index 6de0a1cb..00000000 Binary files a/app/src/main/res/drawable-xhdpi/node_modules_reactnavigation_src_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xhdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png b/app/src/main/res/drawable-xhdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png deleted file mode 100644 index 6de0a1cb..00000000 Binary files a/app/src/main/res/drawable-xhdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xhdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png b/app/src/main/res/drawable-xhdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png deleted file mode 100644 index 6de0a1cb..00000000 Binary files a/app/src/main/res/drawable-xhdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xhdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png b/app/src/main/res/drawable-xhdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png deleted file mode 100644 index 6de0a1cb..00000000 Binary files a/app/src/main/res/drawable-xhdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xhdpi/stripe_2x.png b/app/src/main/res/drawable-xhdpi/stripe_2x.png new file mode 100644 index 00000000..e377b672 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/stripe_2x.png differ diff --git a/app/src/main/res/drawable-xxhdpi/default_channel_cover.png b/app/src/main/res/drawable-xxhdpi/default_channel_cover.png new file mode 100644 index 00000000..9b49120a Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/default_channel_cover.png differ diff --git a/app/src/main/res/drawable-xxhdpi/gerbil_happy.png b/app/src/main/res/drawable-xxhdpi/gerbil_happy.png new file mode 100644 index 00000000..4247f831 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/gerbil_happy.png differ diff --git a/app/src/main/res/drawable-xxhdpi/gerbil_sad.png b/app/src/main/res/drawable-xxhdpi/gerbil_sad.png new file mode 100644 index 00000000..153d4adf Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/gerbil_sad.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_about.png b/app/src/main/res/drawable-xxhdpi/ic_about.png new file mode 100644 index 00000000..85336c2d Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_about.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_arrow_dropdown.png b/app/src/main/res/drawable-xxhdpi/ic_arrow_dropdown.png new file mode 100644 index 00000000..2a7e5766 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_arrow_dropdown.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_arrow_dropup.png b/app/src/main/res/drawable-xxhdpi/ic_arrow_dropup.png new file mode 100644 index 00000000..8c868014 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_arrow_dropup.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_check.png b/app/src/main/res/drawable-xxhdpi/ic_check.png new file mode 100644 index 00000000..506b931d Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_check.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_close.png b/app/src/main/res/drawable-xxhdpi/ic_close.png new file mode 100644 index 00000000..1e41c557 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_close.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_copy.png b/app/src/main/res/drawable-xxhdpi/ic_copy.png new file mode 100644 index 00000000..bfc0c258 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_copy.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_download.png b/app/src/main/res/drawable-xxhdpi/ic_download.png new file mode 100644 index 00000000..f0df0523 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_download.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_editors_choice.png b/app/src/main/res/drawable-xxhdpi/ic_editors_choice.png new file mode 100644 index 00000000..bbf3481a Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_editors_choice.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_file.png b/app/src/main/res/drawable-xxhdpi/ic_file.png new file mode 100644 index 00000000..be649527 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_file.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_following.png b/app/src/main/res/drawable-xxhdpi/ic_following.png new file mode 100644 index 00000000..b7a0f318 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_following.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_invites.png b/app/src/main/res/drawable-xxhdpi/ic_invites.png new file mode 100644 index 00000000..5224ca25 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_invites.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_launcher.png b/app/src/main/res/drawable-xxhdpi/ic_launcher.png deleted file mode 100644 index d5c9438b..00000000 Binary files a/app/src/main/res/drawable-xxhdpi/ic_launcher.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_lbry.png b/app/src/main/res/drawable-xxhdpi/ic_lbry.png deleted file mode 100644 index 048f3bbe..00000000 Binary files a/app/src/main/res/drawable-xxhdpi/ic_lbry.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_library.png b/app/src/main/res/drawable-xxhdpi/ic_library.png new file mode 100644 index 00000000..f0df0523 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_library.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_lock.png b/app/src/main/res/drawable-xxhdpi/ic_lock.png new file mode 100644 index 00000000..ea7292b5 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_lock.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_new.png b/app/src/main/res/drawable-xxhdpi/ic_new.png new file mode 100644 index 00000000..e10a418b Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_new.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_publish.png b/app/src/main/res/drawable-xxhdpi/ic_publish.png new file mode 100644 index 00000000..44167620 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_publish.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_publishes.png b/app/src/main/res/drawable-xxhdpi/ic_publishes.png new file mode 100644 index 00000000..ab71c39e Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_publishes.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_report.png b/app/src/main/res/drawable-xxhdpi/ic_report.png new file mode 100644 index 00000000..4eb94dd2 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_report.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_search.png b/app/src/main/res/drawable-xxhdpi/ic_search.png new file mode 100644 index 00000000..64b54caf Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_search.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_settings.png b/app/src/main/res/drawable-xxhdpi/ic_settings.png new file mode 100644 index 00000000..80e49bb7 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_settings.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_share.png b/app/src/main/res/drawable-xxhdpi/ic_share.png new file mode 100644 index 00000000..edd9c312 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_share.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_top.png b/app/src/main/res/drawable-xxhdpi/ic_top.png new file mode 100644 index 00000000..3e289889 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_top.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_trending.png b/app/src/main/res/drawable-xxhdpi/ic_trending.png new file mode 100644 index 00000000..b4e6af6f Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_trending.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_wallet.png b/app/src/main/res/drawable-xxhdpi/ic_wallet.png new file mode 100644 index 00000000..c0d7666a Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_wallet.png differ diff --git a/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png b/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png deleted file mode 100644 index 15a983a6..00000000 Binary files a/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigation_src_views_assets_backicon.png b/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigation_src_views_assets_backicon.png deleted file mode 100644 index 15a983a6..00000000 Binary files a/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigation_src_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png b/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png deleted file mode 100644 index 15a983a6..00000000 Binary files a/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png b/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png deleted file mode 100644 index 15a983a6..00000000 Binary files a/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png b/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png deleted file mode 100644 index 15a983a6..00000000 Binary files a/app/src/main/res/drawable-xxhdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxhdpi/stripe_2x.png b/app/src/main/res/drawable-xxhdpi/stripe_2x.png new file mode 100644 index 00000000..e377b672 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/stripe_2x.png differ diff --git a/app/src/main/res/drawable-xxxhdpi/ic_lbry.png b/app/src/main/res/drawable-xxxhdpi/ic_lbry.png deleted file mode 100644 index c5ff4133..00000000 Binary files a/app/src/main/res/drawable-xxxhdpi/ic_lbry.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png b/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png deleted file mode 100644 index 17e52e85..00000000 Binary files a/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigation_node_modules_reactnavigationstack_lib_module_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigation_src_views_assets_backicon.png b/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigation_src_views_assets_backicon.png deleted file mode 100644 index 17e52e85..00000000 Binary files a/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigation_src_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png b/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png deleted file mode 100644 index 17e52e85..00000000 Binary files a/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigationstack_dist_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png b/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png deleted file mode 100644 index 17e52e85..00000000 Binary files a/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigationstack_lib_module_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png b/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png deleted file mode 100644 index 17e52e85..00000000 Binary files a/app/src/main/res/drawable-xxxhdpi/node_modules_reactnavigationstack_src_views_assets_backicon.png and /dev/null differ diff --git a/app/src/main/res/drawable/.gitkeep b/app/src/main/res/drawable/.gitkeep deleted file mode 100644 index e69de29b..00000000 diff --git a/app/src/main/res/drawable/bg_channel_icon.xml b/app/src/main/res/drawable/bg_channel_icon.xml new file mode 100644 index 00000000..57eb9801 --- /dev/null +++ b/app/src/main/res/drawable/bg_channel_icon.xml @@ -0,0 +1,2 @@ + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_selected_item.xml b/app/src/main/res/drawable/bg_selected_item.xml new file mode 100644 index 00000000..7e782807 --- /dev/null +++ b/app/src/main/res/drawable/bg_selected_item.xml @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/app/src/main/res/drawable/bg_selected_nav_item.xml b/app/src/main/res/drawable/bg_selected_nav_item.xml new file mode 100644 index 00000000..8da95b8e --- /dev/null +++ b/app/src/main/res/drawable/bg_selected_nav_item.xml @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/app/src/main/res/drawable/bg_tag.xml b/app/src/main/res/drawable/bg_tag.xml new file mode 100644 index 00000000..aa70d0db --- /dev/null +++ b/app/src/main/res/drawable/bg_tag.xml @@ -0,0 +1,6 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_tag_mature.xml b/app/src/main/res/drawable/bg_tag_mature.xml new file mode 100644 index 00000000..418e1053 --- /dev/null +++ b/app/src/main/res/drawable/bg_tag_mature.xml @@ -0,0 +1,6 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_thumbnail_icon.xml b/app/src/main/res/drawable/bg_thumbnail_icon.xml new file mode 100644 index 00000000..17877cf9 --- /dev/null +++ b/app/src/main/res/drawable/bg_thumbnail_icon.xml @@ -0,0 +1,2 @@ + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_thumbnail_placeholder.xml b/app/src/main/res/drawable/bg_thumbnail_placeholder.xml new file mode 100644 index 00000000..41ab809d --- /dev/null +++ b/app/src/main/res/drawable/bg_thumbnail_placeholder.xml @@ -0,0 +1,4 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_transparent.xml b/app/src/main/res/drawable/bg_transparent.xml new file mode 100644 index 00000000..ed2c1e08 --- /dev/null +++ b/app/src/main/res/drawable/bg_transparent.xml @@ -0,0 +1,4 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_wallet_address.xml b/app/src/main/res/drawable/bg_wallet_address.xml new file mode 100644 index 00000000..baac4c1f --- /dev/null +++ b/app/src/main/res/drawable/bg_wallet_address.xml @@ -0,0 +1,13 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/fg_nav_menu_item.xml b/app/src/main/res/drawable/fg_nav_menu_item.xml new file mode 100644 index 00000000..ee7d0e64 --- /dev/null +++ b/app/src/main/res/drawable/fg_nav_menu_item.xml @@ -0,0 +1,5 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_launcher_background.xml b/app/src/main/res/drawable/ic_launcher_background.xml new file mode 100644 index 00000000..ca3826a4 --- /dev/null +++ b/app/src/main/res/drawable/ic_launcher_background.xml @@ -0,0 +1,74 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/drawable/ic_menu_camera.xml b/app/src/main/res/drawable/ic_menu_camera.xml new file mode 100644 index 00000000..634fe922 --- /dev/null +++ b/app/src/main/res/drawable/ic_menu_camera.xml @@ -0,0 +1,12 @@ + + + + diff --git a/app/src/main/res/drawable/ic_menu_gallery.xml b/app/src/main/res/drawable/ic_menu_gallery.xml new file mode 100644 index 00000000..03c77099 --- /dev/null +++ b/app/src/main/res/drawable/ic_menu_gallery.xml @@ -0,0 +1,9 @@ + + + diff --git a/app/src/main/res/drawable/ic_menu_slideshow.xml b/app/src/main/res/drawable/ic_menu_slideshow.xml new file mode 100644 index 00000000..5e9e163a --- /dev/null +++ b/app/src/main/res/drawable/ic_menu_slideshow.xml @@ -0,0 +1,9 @@ + + + diff --git a/app/src/main/res/drawable/icon.png b/app/src/main/res/drawable/icon.png deleted file mode 100644 index fb78a16b..00000000 Binary files a/app/src/main/res/drawable/icon.png and /dev/null differ diff --git a/app/src/main/res/drawable/side_nav_bar.xml b/app/src/main/res/drawable/side_nav_bar.xml new file mode 100644 index 00000000..6d81870b --- /dev/null +++ b/app/src/main/res/drawable/side_nav_bar.xml @@ -0,0 +1,9 @@ + + + \ No newline at end of file diff --git a/app/src/main/res/font/inter.xml b/app/src/main/res/font/inter.xml new file mode 100644 index 00000000..7a51878a --- /dev/null +++ b/app/src/main/res/font/inter.xml @@ -0,0 +1,61 @@ + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/assets/fonts/Inter-Bold.otf b/app/src/main/res/font/inter_bold.otf similarity index 62% rename from app/src/main/assets/fonts/Inter-Bold.otf rename to app/src/main/res/font/inter_bold.otf index 0dbf67b7..43b12eab 100644 Binary files a/app/src/main/assets/fonts/Inter-Bold.otf and b/app/src/main/res/font/inter_bold.otf differ diff --git a/app/src/main/res/font/inter_bolditalic.otf b/app/src/main/res/font/inter_bolditalic.otf new file mode 100644 index 00000000..a07fdd57 Binary files /dev/null and b/app/src/main/res/font/inter_bolditalic.otf differ diff --git a/app/src/main/res/font/inter_italic.otf b/app/src/main/res/font/inter_italic.otf new file mode 100644 index 00000000..7b63db5e Binary files /dev/null and b/app/src/main/res/font/inter_italic.otf differ diff --git a/app/src/main/res/font/inter_light.otf b/app/src/main/res/font/inter_light.otf new file mode 100644 index 00000000..d4f94c37 Binary files /dev/null and b/app/src/main/res/font/inter_light.otf differ diff --git a/app/src/main/res/font/inter_lightitalic.otf b/app/src/main/res/font/inter_lightitalic.otf new file mode 100644 index 00000000..9a4b7a03 Binary files /dev/null and b/app/src/main/res/font/inter_lightitalic.otf differ diff --git a/app/src/main/assets/fonts/Inter-Regular.otf b/app/src/main/res/font/inter_regular.otf similarity index 57% rename from app/src/main/assets/fonts/Inter-Regular.otf rename to app/src/main/res/font/inter_regular.otf index 6724353f..c0dab1bf 100644 Binary files a/app/src/main/assets/fonts/Inter-Regular.otf and b/app/src/main/res/font/inter_regular.otf differ diff --git a/app/src/main/assets/fonts/Inter-SemiBold.otf b/app/src/main/res/font/inter_semibold.otf similarity index 63% rename from app/src/main/assets/fonts/Inter-SemiBold.otf rename to app/src/main/res/font/inter_semibold.otf index a6f9b2dc..4c238105 100644 Binary files a/app/src/main/assets/fonts/Inter-SemiBold.otf and b/app/src/main/res/font/inter_semibold.otf differ diff --git a/app/src/main/res/layout/activity_file_view.xml b/app/src/main/res/layout/activity_file_view.xml new file mode 100644 index 00000000..3157c206 --- /dev/null +++ b/app/src/main/res/layout/activity_file_view.xml @@ -0,0 +1,387 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/activity_first_run.xml b/app/src/main/res/layout/activity_first_run.xml new file mode 100644 index 00000000..13ced125 --- /dev/null +++ b/app/src/main/res/layout/activity_first_run.xml @@ -0,0 +1,99 @@ + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/activity_main.xml b/app/src/main/res/layout/activity_main.xml new file mode 100644 index 00000000..85f39dd0 --- /dev/null +++ b/app/src/main/res/layout/activity_main.xml @@ -0,0 +1,34 @@ + + + + + + + + + + + + diff --git a/app/src/main/res/layout/activity_verification.xml b/app/src/main/res/layout/activity_verification.xml new file mode 100644 index 00000000..9714897a --- /dev/null +++ b/app/src/main/res/layout/activity_verification.xml @@ -0,0 +1,37 @@ + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/activity_video.xml b/app/src/main/res/layout/activity_video.xml new file mode 100644 index 00000000..3425222b --- /dev/null +++ b/app/src/main/res/layout/activity_video.xml @@ -0,0 +1,13 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/app_bar_main.xml b/app/src/main/res/layout/app_bar_main.xml new file mode 100644 index 00000000..f2141a5d --- /dev/null +++ b/app/src/main/res/layout/app_bar_main.xml @@ -0,0 +1,94 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/card_wallet_balance.xml b/app/src/main/res/layout/card_wallet_balance.xml new file mode 100644 index 00000000..06536782 --- /dev/null +++ b/app/src/main/res/layout/card_wallet_balance.xml @@ -0,0 +1,293 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/card_wallet_receive_credits.xml b/app/src/main/res/layout/card_wallet_receive_credits.xml new file mode 100644 index 00000000..880a55e7 --- /dev/null +++ b/app/src/main/res/layout/card_wallet_receive_credits.xml @@ -0,0 +1,82 @@ + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/card_wallet_recent_transactions.xml b/app/src/main/res/layout/card_wallet_recent_transactions.xml new file mode 100644 index 00000000..916bf3c8 --- /dev/null +++ b/app/src/main/res/layout/card_wallet_recent_transactions.xml @@ -0,0 +1,90 @@ + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/card_wallet_send_credits.xml b/app/src/main/res/layout/card_wallet_send_credits.xml new file mode 100644 index 00000000..ec65e681 --- /dev/null +++ b/app/src/main/res/layout/card_wallet_send_credits.xml @@ -0,0 +1,105 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/card_wallet_sync.xml b/app/src/main/res/layout/card_wallet_sync.xml new file mode 100644 index 00000000..a3c46c55 --- /dev/null +++ b/app/src/main/res/layout/card_wallet_sync.xml @@ -0,0 +1,128 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/content_main.xml b/app/src/main/res/layout/content_main.xml new file mode 100644 index 00000000..81f12902 --- /dev/null +++ b/app/src/main/res/layout/content_main.xml @@ -0,0 +1,106 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/dialog_content_from.xml b/app/src/main/res/layout/dialog_content_from.xml new file mode 100644 index 00000000..cf99a858 --- /dev/null +++ b/app/src/main/res/layout/dialog_content_from.xml @@ -0,0 +1,154 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/dialog_content_sort.xml b/app/src/main/res/layout/dialog_content_sort.xml new file mode 100644 index 00000000..593bfd67 --- /dev/null +++ b/app/src/main/res/layout/dialog_content_sort.xml @@ -0,0 +1,129 @@ + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_all_content.xml b/app/src/main/res/layout/fragment_all_content.xml new file mode 100644 index 00000000..dc1100f8 --- /dev/null +++ b/app/src/main/res/layout/fragment_all_content.xml @@ -0,0 +1,190 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_channel.xml b/app/src/main/res/layout/fragment_channel.xml new file mode 100644 index 00000000..3bd2b4a2 --- /dev/null +++ b/app/src/main/res/layout/fragment_channel.xml @@ -0,0 +1,136 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_channel_about.xml b/app/src/main/res/layout/fragment_channel_about.xml new file mode 100644 index 00000000..d8e7e04d --- /dev/null +++ b/app/src/main/res/layout/fragment_channel_about.xml @@ -0,0 +1,96 @@ + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_channel_content.xml b/app/src/main/res/layout/fragment_channel_content.xml new file mode 100644 index 00000000..62ae50d2 --- /dev/null +++ b/app/src/main/res/layout/fragment_channel_content.xml @@ -0,0 +1,108 @@ + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_following.xml b/app/src/main/res/layout/fragment_following.xml new file mode 100644 index 00000000..03bb8a80 --- /dev/null +++ b/app/src/main/res/layout/fragment_following.xml @@ -0,0 +1,193 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_gallery.xml b/app/src/main/res/layout/fragment_gallery.xml new file mode 100644 index 00000000..643fe254 --- /dev/null +++ b/app/src/main/res/layout/fragment_gallery.xml @@ -0,0 +1,22 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_home.xml b/app/src/main/res/layout/fragment_home.xml new file mode 100644 index 00000000..f3d9b08f --- /dev/null +++ b/app/src/main/res/layout/fragment_home.xml @@ -0,0 +1,22 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_search.xml b/app/src/main/res/layout/fragment_search.xml new file mode 100644 index 00000000..b8126321 --- /dev/null +++ b/app/src/main/res/layout/fragment_search.xml @@ -0,0 +1,49 @@ + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_slideshow.xml b/app/src/main/res/layout/fragment_slideshow.xml new file mode 100644 index 00000000..2141a333 --- /dev/null +++ b/app/src/main/res/layout/fragment_slideshow.xml @@ -0,0 +1,22 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_verification_email.xml b/app/src/main/res/layout/fragment_verification_email.xml new file mode 100644 index 00000000..42663abf --- /dev/null +++ b/app/src/main/res/layout/fragment_verification_email.xml @@ -0,0 +1,156 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_verification_wallet.xml b/app/src/main/res/layout/fragment_verification_wallet.xml new file mode 100644 index 00000000..273903e0 --- /dev/null +++ b/app/src/main/res/layout/fragment_verification_wallet.xml @@ -0,0 +1,111 @@ + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_wallet.xml b/app/src/main/res/layout/fragment_wallet.xml new file mode 100644 index 00000000..bd64fb1d --- /dev/null +++ b/app/src/main/res/layout/fragment_wallet.xml @@ -0,0 +1,128 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_channel.xml b/app/src/main/res/layout/list_item_channel.xml new file mode 100644 index 00000000..3e9c4c5d --- /dev/null +++ b/app/src/main/res/layout/list_item_channel.xml @@ -0,0 +1,78 @@ + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_channel_filter.xml b/app/src/main/res/layout/list_item_channel_filter.xml new file mode 100644 index 00000000..a8ad20f3 --- /dev/null +++ b/app/src/main/res/layout/list_item_channel_filter.xml @@ -0,0 +1,53 @@ + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_featured_search_result.xml b/app/src/main/res/layout/list_item_featured_search_result.xml new file mode 100644 index 00000000..019c22f8 --- /dev/null +++ b/app/src/main/res/layout/list_item_featured_search_result.xml @@ -0,0 +1,100 @@ + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_nav_menu_group.xml b/app/src/main/res/layout/list_item_nav_menu_group.xml new file mode 100644 index 00000000..fb402dea --- /dev/null +++ b/app/src/main/res/layout/list_item_nav_menu_group.xml @@ -0,0 +1,18 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_nav_menu_item.xml b/app/src/main/res/layout/list_item_nav_menu_item.xml new file mode 100644 index 00000000..19fe7943 --- /dev/null +++ b/app/src/main/res/layout/list_item_nav_menu_item.xml @@ -0,0 +1,31 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_stream.xml b/app/src/main/res/layout/list_item_stream.xml new file mode 100644 index 00000000..8d611504 --- /dev/null +++ b/app/src/main/res/layout/list_item_stream.xml @@ -0,0 +1,95 @@ + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_suggested_channel.xml b/app/src/main/res/layout/list_item_suggested_channel.xml new file mode 100644 index 00000000..3c527c63 --- /dev/null +++ b/app/src/main/res/layout/list_item_suggested_channel.xml @@ -0,0 +1,67 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_tag.xml b/app/src/main/res/layout/list_item_tag.xml new file mode 100644 index 00000000..63530d5b --- /dev/null +++ b/app/src/main/res/layout/list_item_tag.xml @@ -0,0 +1,21 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_transaction.xml b/app/src/main/res/layout/list_item_transaction.xml new file mode 100644 index 00000000..14e1ccf7 --- /dev/null +++ b/app/src/main/res/layout/list_item_transaction.xml @@ -0,0 +1,107 @@ + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_url_suggestion.xml b/app/src/main/res/layout/list_item_url_suggestion.xml new file mode 100644 index 00000000..335c865e --- /dev/null +++ b/app/src/main/res/layout/list_item_url_suggestion.xml @@ -0,0 +1,44 @@ + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/nav_header_main.xml b/app/src/main/res/layout/nav_header_main.xml new file mode 100644 index 00000000..dbdc2cde --- /dev/null +++ b/app/src/main/res/layout/nav_header_main.xml @@ -0,0 +1,60 @@ + + + + + + + + + + + + + diff --git a/app/src/main/res/menu/activity_main_drawer.xml b/app/src/main/res/menu/activity_main_drawer.xml new file mode 100644 index 00000000..b700902f --- /dev/null +++ b/app/src/main/res/menu/activity_main_drawer.xml @@ -0,0 +1,20 @@ + + + + + + + + + diff --git a/app/src/main/res/menu/main.xml b/app/src/main/res/menu/main.xml new file mode 100644 index 00000000..a2411e31 --- /dev/null +++ b/app/src/main/res/menu/main.xml @@ -0,0 +1,9 @@ + + + + diff --git a/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml b/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml new file mode 100644 index 00000000..c4a603d4 --- /dev/null +++ b/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml b/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml new file mode 100644 index 00000000..c4a603d4 --- /dev/null +++ b/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/mipmap-hdpi/ic_launcher.png b/app/src/main/res/mipmap-hdpi/ic_launcher.png index a2f59082..f7533d43 100644 Binary files a/app/src/main/res/mipmap-hdpi/ic_launcher.png and b/app/src/main/res/mipmap-hdpi/ic_launcher.png differ diff --git a/app/src/main/res/mipmap-hdpi/ic_launcher_foreground.png b/app/src/main/res/mipmap-hdpi/ic_launcher_foreground.png new file mode 100644 index 00000000..15eb8049 Binary files /dev/null and b/app/src/main/res/mipmap-hdpi/ic_launcher_foreground.png differ diff --git a/app/src/main/res/mipmap-hdpi/ic_launcher_round.png b/app/src/main/res/mipmap-hdpi/ic_launcher_round.png index 1b523998..4319062c 100644 Binary files a/app/src/main/res/mipmap-hdpi/ic_launcher_round.png and b/app/src/main/res/mipmap-hdpi/ic_launcher_round.png differ diff --git a/app/src/main/res/mipmap-mdpi/ic_launcher.png b/app/src/main/res/mipmap-mdpi/ic_launcher.png index ff10afd6..b465e06a 100644 Binary files a/app/src/main/res/mipmap-mdpi/ic_launcher.png and b/app/src/main/res/mipmap-mdpi/ic_launcher.png differ diff --git a/app/src/main/res/mipmap-mdpi/ic_launcher_foreground.png b/app/src/main/res/mipmap-mdpi/ic_launcher_foreground.png new file mode 100644 index 00000000..87146093 Binary files /dev/null and b/app/src/main/res/mipmap-mdpi/ic_launcher_foreground.png differ diff --git a/app/src/main/res/mipmap-mdpi/ic_launcher_round.png b/app/src/main/res/mipmap-mdpi/ic_launcher_round.png index 115a4c76..93807367 100644 Binary files a/app/src/main/res/mipmap-mdpi/ic_launcher_round.png and b/app/src/main/res/mipmap-mdpi/ic_launcher_round.png differ diff --git a/app/src/main/res/mipmap-xhdpi/ic_launcher.png b/app/src/main/res/mipmap-xhdpi/ic_launcher.png index dcd3cd80..55844606 100644 Binary files a/app/src/main/res/mipmap-xhdpi/ic_launcher.png and b/app/src/main/res/mipmap-xhdpi/ic_launcher.png differ diff --git a/app/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png b/app/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png new file mode 100644 index 00000000..a2c14e06 Binary files /dev/null and b/app/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png differ diff --git a/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png b/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png index 459ca609..b2a63b40 100644 Binary files a/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png and b/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png differ diff --git a/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/app/src/main/res/mipmap-xxhdpi/ic_launcher.png index 8ca12fe0..6370e492 100644 Binary files a/app/src/main/res/mipmap-xxhdpi/ic_launcher.png and b/app/src/main/res/mipmap-xxhdpi/ic_launcher.png differ diff --git a/app/src/main/res/mipmap-xxhdpi/ic_launcher_foreground.png b/app/src/main/res/mipmap-xxhdpi/ic_launcher_foreground.png new file mode 100644 index 00000000..7e344d2b Binary files /dev/null and b/app/src/main/res/mipmap-xxhdpi/ic_launcher_foreground.png differ diff --git a/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png b/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png index 8e19b410..896fc998 100644 Binary files a/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png and b/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png differ diff --git a/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png index b824ebdd..c5112355 100644 Binary files a/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png and b/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png differ diff --git a/app/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png b/app/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png new file mode 100644 index 00000000..2d329001 Binary files /dev/null and b/app/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png differ diff --git a/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png b/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png index 4c19a13c..6707ab40 100644 Binary files a/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png and b/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png differ diff --git a/app/src/main/res/raw/node_modules_nodeemoji_lib_emoji.json b/app/src/main/res/raw/node_modules_nodeemoji_lib_emoji.json deleted file mode 100644 index 4fbb1c4a..00000000 --- a/app/src/main/res/raw/node_modules_nodeemoji_lib_emoji.json +++ /dev/null @@ -1 +0,0 @@ -{"100":"💯","1234":"🔢","umbrella_with_rain_drops":"☔","coffee":"☕","aries":"♈","taurus":"♉","sagittarius":"♐","capricorn":"♑","aquarius":"♒","pisces":"♓","anchor":"⚓","white_check_mark":"✅","sparkles":"✨","question":"❓","grey_question":"❔","grey_exclamation":"❕","exclamation":"❗","heavy_exclamation_mark":"❗","heavy_plus_sign":"➕","heavy_minus_sign":"➖","heavy_division_sign":"➗","hash":"#️⃣","keycap_star":"*️⃣","zero":"0️⃣","one":"1️⃣","two":"2️⃣","three":"3️⃣","four":"4️⃣","five":"5️⃣","six":"6️⃣","seven":"7️⃣","eight":"8️⃣","nine":"9️⃣","copyright":"©️","registered":"®️","mahjong":"🀄","black_joker":"🃏","a":"🅰️","b":"🅱️","o2":"🅾️","parking":"🅿️","ab":"🆎","cl":"🆑","cool":"🆒","free":"🆓","id":"🆔","new":"🆕","ng":"🆖","ok":"🆗","sos":"🆘","up":"🆙","vs":"🆚","flag-ac":"🇦🇨","flag-ad":"🇦🇩","flag-ae":"🇦🇪","flag-af":"🇦🇫","flag-ag":"🇦🇬","flag-ai":"🇦🇮","flag-al":"🇦🇱","flag-am":"🇦🇲","flag-ao":"🇦🇴","flag-aq":"🇦🇶","flag-ar":"🇦🇷","flag-as":"🇦🇸","flag-at":"🇦🇹","flag-au":"🇦🇺","flag-aw":"🇦🇼","flag-ax":"🇦🇽","flag-az":"🇦🇿","flag-ba":"🇧🇦","flag-bb":"🇧🇧","flag-bd":"🇧🇩","flag-be":"🇧🇪","flag-bf":"🇧🇫","flag-bg":"🇧🇬","flag-bh":"🇧🇭","flag-bi":"🇧🇮","flag-bj":"🇧🇯","flag-bl":"🇧🇱","flag-bm":"🇧🇲","flag-bn":"🇧🇳","flag-bo":"🇧🇴","flag-bq":"🇧🇶","flag-br":"🇧🇷","flag-bs":"🇧🇸","flag-bt":"🇧🇹","flag-bv":"🇧🇻","flag-bw":"🇧🇼","flag-by":"🇧🇾","flag-bz":"🇧🇿","flag-ca":"🇨🇦","flag-cc":"🇨🇨","flag-cd":"🇨🇩","flag-cf":"🇨🇫","flag-cg":"🇨🇬","flag-ch":"🇨🇭","flag-ci":"🇨🇮","flag-ck":"🇨🇰","flag-cl":"🇨🇱","flag-cm":"🇨🇲","cn":"🇨🇳","flag-cn":"🇨🇳","flag-co":"🇨🇴","flag-cp":"🇨🇵","flag-cr":"🇨🇷","flag-cu":"🇨🇺","flag-cv":"🇨🇻","flag-cw":"🇨🇼","flag-cx":"🇨🇽","flag-cy":"🇨🇾","flag-cz":"🇨🇿","de":"🇩🇪","flag-de":"🇩🇪","flag-dg":"🇩🇬","flag-dj":"🇩🇯","flag-dk":"🇩🇰","flag-dm":"🇩🇲","flag-do":"🇩🇴","flag-dz":"🇩🇿","flag-ea":"🇪🇦","flag-ec":"🇪🇨","flag-ee":"🇪🇪","flag-eg":"🇪🇬","flag-eh":"🇪🇭","flag-er":"🇪🇷","es":"🇪🇸","flag-es":"🇪🇸","flag-et":"🇪🇹","flag-eu":"🇪🇺","flag-fi":"🇫🇮","flag-fj":"🇫🇯","flag-fk":"🇫🇰","flag-fm":"🇫🇲","flag-fo":"🇫🇴","fr":"🇫🇷","flag-fr":"🇫🇷","flag-ga":"🇬🇦","gb":"🇬🇧","uk":"🇬🇧","flag-gb":"🇬🇧","flag-gd":"🇬🇩","flag-ge":"🇬🇪","flag-gf":"🇬🇫","flag-gg":"🇬🇬","flag-gh":"🇬🇭","flag-gi":"🇬🇮","flag-gl":"🇬🇱","flag-gm":"🇬🇲","flag-gn":"🇬🇳","flag-gp":"🇬🇵","flag-gq":"🇬🇶","flag-gr":"🇬🇷","flag-gs":"🇬🇸","flag-gt":"🇬🇹","flag-gu":"🇬🇺","flag-gw":"🇬🇼","flag-gy":"🇬🇾","flag-hk":"🇭🇰","flag-hm":"🇭🇲","flag-hn":"🇭🇳","flag-hr":"🇭🇷","flag-ht":"🇭🇹","flag-hu":"🇭🇺","flag-ic":"🇮🇨","flag-id":"🇮🇩","flag-ie":"🇮🇪","flag-il":"🇮🇱","flag-im":"🇮🇲","flag-in":"🇮🇳","flag-io":"🇮🇴","flag-iq":"🇮🇶","flag-ir":"🇮🇷","flag-is":"🇮🇸","it":"🇮🇹","flag-it":"🇮🇹","flag-je":"🇯🇪","flag-jm":"🇯🇲","flag-jo":"🇯🇴","jp":"🇯🇵","flag-jp":"🇯🇵","flag-ke":"🇰🇪","flag-kg":"🇰🇬","flag-kh":"🇰🇭","flag-ki":"🇰🇮","flag-km":"🇰🇲","flag-kn":"🇰🇳","flag-kp":"🇰🇵","kr":"🇰🇷","flag-kr":"🇰🇷","flag-kw":"🇰🇼","flag-ky":"🇰🇾","flag-kz":"🇰🇿","flag-la":"🇱🇦","flag-lb":"🇱🇧","flag-lc":"🇱🇨","flag-li":"🇱🇮","flag-lk":"🇱🇰","flag-lr":"🇱🇷","flag-ls":"🇱🇸","flag-lt":"🇱🇹","flag-lu":"🇱🇺","flag-lv":"🇱🇻","flag-ly":"🇱🇾","flag-ma":"🇲🇦","flag-mc":"🇲🇨","flag-md":"🇲🇩","flag-me":"🇲🇪","flag-mf":"🇲🇫","flag-mg":"🇲🇬","flag-mh":"🇲🇭","flag-mk":"🇲🇰","flag-ml":"🇲🇱","flag-mm":"🇲🇲","flag-mn":"🇲🇳","flag-mo":"🇲🇴","flag-mp":"🇲🇵","flag-mq":"🇲🇶","flag-mr":"🇲🇷","flag-ms":"🇲🇸","flag-mt":"🇲🇹","flag-mu":"🇲🇺","flag-mv":"🇲🇻","flag-mw":"🇲🇼","flag-mx":"🇲🇽","flag-my":"🇲🇾","flag-mz":"🇲🇿","flag-na":"🇳🇦","flag-nc":"🇳🇨","flag-ne":"🇳🇪","flag-nf":"🇳🇫","flag-ng":"🇳🇬","flag-ni":"🇳🇮","flag-nl":"🇳🇱","flag-no":"🇳🇴","flag-np":"🇳🇵","flag-nr":"🇳🇷","flag-nu":"🇳🇺","flag-nz":"🇳🇿","flag-om":"🇴🇲","flag-pa":"🇵🇦","flag-pe":"🇵🇪","flag-pf":"🇵🇫","flag-pg":"🇵🇬","flag-ph":"🇵🇭","flag-pk":"🇵🇰","flag-pl":"🇵🇱","flag-pm":"🇵🇲","flag-pn":"🇵🇳","flag-pr":"🇵🇷","flag-ps":"🇵🇸","flag-pt":"🇵🇹","flag-pw":"🇵🇼","flag-py":"🇵🇾","flag-qa":"🇶🇦","flag-re":"🇷🇪","flag-ro":"🇷🇴","flag-rs":"🇷🇸","ru":"🇷🇺","flag-ru":"🇷🇺","flag-rw":"🇷🇼","flag-sa":"🇸🇦","flag-sb":"🇸🇧","flag-sc":"🇸🇨","flag-sd":"🇸🇩","flag-se":"🇸🇪","flag-sg":"🇸🇬","flag-sh":"🇸🇭","flag-si":"🇸🇮","flag-sj":"🇸🇯","flag-sk":"🇸🇰","flag-sl":"🇸🇱","flag-sm":"🇸🇲","flag-sn":"🇸🇳","flag-so":"🇸🇴","flag-sr":"🇸🇷","flag-ss":"🇸🇸","flag-st":"🇸🇹","flag-sv":"🇸🇻","flag-sx":"🇸🇽","flag-sy":"🇸🇾","flag-sz":"🇸🇿","flag-ta":"🇹🇦","flag-tc":"🇹🇨","flag-td":"🇹🇩","flag-tf":"🇹🇫","flag-tg":"🇹🇬","flag-th":"🇹🇭","flag-tj":"🇹🇯","flag-tk":"🇹🇰","flag-tl":"🇹🇱","flag-tm":"🇹🇲","flag-tn":"🇹🇳","flag-to":"🇹🇴","flag-tr":"🇹🇷","flag-tt":"🇹🇹","flag-tv":"🇹🇻","flag-tw":"🇹🇼","flag-tz":"🇹🇿","flag-ua":"🇺🇦","flag-ug":"🇺🇬","flag-um":"🇺🇲","flag-un":"🇺🇳","us":"🇺🇸","flag-us":"🇺🇸","flag-uy":"🇺🇾","flag-uz":"🇺🇿","flag-va":"🇻🇦","flag-vc":"🇻🇨","flag-ve":"🇻🇪","flag-vg":"🇻🇬","flag-vi":"🇻🇮","flag-vn":"🇻🇳","flag-vu":"🇻🇺","flag-wf":"🇼🇫","flag-ws":"🇼🇸","flag-xk":"🇽🇰","flag-ye":"🇾🇪","flag-yt":"🇾🇹","flag-za":"🇿🇦","flag-zm":"🇿🇲","flag-zw":"🇿🇼","koko":"🈁","sa":"🈂️","u7121":"🈚","u6307":"🈯","u7981":"🈲","u7a7a":"🈳","u5408":"🈴","u6e80":"🈵","u6709":"🈶","u6708":"🈷️","u7533":"🈸","u5272":"🈹","u55b6":"🈺","ideograph_advantage":"🉐","accept":"🉑","cyclone":"🌀","foggy":"🌁","closed_umbrella":"🌂","night_with_stars":"🌃","sunrise_over_mountains":"🌄","sunrise":"🌅","city_sunset":"🌆","city_sunrise":"🌇","rainbow":"🌈","bridge_at_night":"🌉","ocean":"🌊","volcano":"🌋","milky_way":"🌌","earth_africa":"🌍","earth_americas":"🌎","earth_asia":"🌏","globe_with_meridians":"🌐","new_moon":"🌑","waxing_crescent_moon":"🌒","first_quarter_moon":"🌓","moon":"🌔","waxing_gibbous_moon":"🌔","full_moon":"🌕","waning_gibbous_moon":"🌖","last_quarter_moon":"🌗","waning_crescent_moon":"🌘","crescent_moon":"🌙","new_moon_with_face":"🌚","first_quarter_moon_with_face":"🌛","last_quarter_moon_with_face":"🌜","full_moon_with_face":"🌝","sun_with_face":"🌞","star2":"🌟","stars":"🌠","thermometer":"🌡️","mostly_sunny":"🌤️","sun_small_cloud":"🌤️","barely_sunny":"🌥️","sun_behind_cloud":"🌥️","partly_sunny_rain":"🌦️","sun_behind_rain_cloud":"🌦️","rain_cloud":"🌧️","snow_cloud":"🌨️","lightning":"🌩️","lightning_cloud":"🌩️","tornado":"🌪️","tornado_cloud":"🌪️","fog":"🌫️","wind_blowing_face":"🌬️","hotdog":"🌭","taco":"🌮","burrito":"🌯","chestnut":"🌰","seedling":"🌱","evergreen_tree":"🌲","deciduous_tree":"🌳","palm_tree":"🌴","cactus":"🌵","hot_pepper":"🌶️","tulip":"🌷","cherry_blossom":"🌸","rose":"🌹","hibiscus":"🌺","sunflower":"🌻","blossom":"🌼","corn":"🌽","ear_of_rice":"🌾","herb":"🌿","four_leaf_clover":"🍀","maple_leaf":"🍁","fallen_leaf":"🍂","leaves":"🍃","mushroom":"🍄","tomato":"🍅","eggplant":"🍆","grapes":"🍇","melon":"🍈","watermelon":"🍉","tangerine":"🍊","lemon":"🍋","banana":"🍌","pineapple":"🍍","apple":"🍎","green_apple":"🍏","pear":"🍐","peach":"🍑","cherries":"🍒","strawberry":"🍓","hamburger":"🍔","pizza":"🍕","meat_on_bone":"🍖","poultry_leg":"🍗","rice_cracker":"🍘","rice_ball":"🍙","rice":"🍚","curry":"🍛","ramen":"🍜","spaghetti":"🍝","bread":"🍞","fries":"🍟","sweet_potato":"🍠","dango":"🍡","oden":"🍢","sushi":"🍣","fried_shrimp":"🍤","fish_cake":"🍥","icecream":"🍦","shaved_ice":"🍧","ice_cream":"🍨","doughnut":"🍩","cookie":"🍪","chocolate_bar":"🍫","candy":"🍬","lollipop":"🍭","custard":"🍮","honey_pot":"🍯","cake":"🍰","bento":"🍱","stew":"🍲","fried_egg":"🍳","cooking":"🍳","fork_and_knife":"🍴","tea":"🍵","sake":"🍶","wine_glass":"🍷","cocktail":"🍸","tropical_drink":"🍹","beer":"🍺","beers":"🍻","baby_bottle":"🍼","knife_fork_plate":"🍽️","champagne":"🍾","popcorn":"🍿","ribbon":"🎀","gift":"🎁","birthday":"🎂","jack_o_lantern":"🎃","christmas_tree":"🎄","santa":"🎅","fireworks":"🎆","sparkler":"🎇","balloon":"🎈","tada":"🎉","confetti_ball":"🎊","tanabata_tree":"🎋","crossed_flags":"🎌","bamboo":"🎍","dolls":"🎎","flags":"🎏","wind_chime":"🎐","rice_scene":"🎑","school_satchel":"🎒","mortar_board":"🎓","medal":"🎖️","reminder_ribbon":"🎗️","studio_microphone":"🎙️","level_slider":"🎚️","control_knobs":"🎛️","film_frames":"🎞️","admission_tickets":"🎟️","carousel_horse":"🎠","ferris_wheel":"🎡","roller_coaster":"🎢","fishing_pole_and_fish":"🎣","microphone":"🎤","movie_camera":"🎥","cinema":"🎦","headphones":"🎧","art":"🎨","tophat":"🎩","circus_tent":"🎪","ticket":"🎫","clapper":"🎬","performing_arts":"🎭","video_game":"🎮","dart":"🎯","slot_machine":"🎰","8ball":"🎱","game_die":"🎲","bowling":"🎳","flower_playing_cards":"🎴","musical_note":"🎵","notes":"🎶","saxophone":"🎷","guitar":"🎸","musical_keyboard":"🎹","trumpet":"🎺","violin":"🎻","musical_score":"🎼","running_shirt_with_sash":"🎽","tennis":"🎾","ski":"🎿","basketball":"🏀","checkered_flag":"🏁","snowboarder":"🏂","woman-running":"🏃‍♀️","man-running":"🏃‍♂️","runner":"🏃‍♂️","running":"🏃‍♂️","woman-surfing":"🏄‍♀️","man-surfing":"🏄‍♂️","surfer":"🏄‍♂️","sports_medal":"🏅","trophy":"🏆","horse_racing":"🏇","football":"🏈","rugby_football":"🏉","woman-swimming":"🏊‍♀️","man-swimming":"🏊‍♂️","swimmer":"🏊‍♂️","woman-lifting-weights":"🏋️‍♀️","man-lifting-weights":"🏋️‍♂️","weight_lifter":"🏋️‍♂️","woman-golfing":"🏌️‍♀️","man-golfing":"🏌️‍♂️","golfer":"🏌️‍♂️","racing_motorcycle":"🏍️","racing_car":"🏎️","cricket_bat_and_ball":"🏏","volleyball":"🏐","field_hockey_stick_and_ball":"🏑","ice_hockey_stick_and_puck":"🏒","table_tennis_paddle_and_ball":"🏓","snow_capped_mountain":"🏔️","camping":"🏕️","beach_with_umbrella":"🏖️","building_construction":"🏗️","house_buildings":"🏘️","cityscape":"🏙️","derelict_house_building":"🏚️","classical_building":"🏛️","desert":"🏜️","desert_island":"🏝️","national_park":"🏞️","stadium":"🏟️","house":"🏠","house_with_garden":"🏡","office":"🏢","post_office":"🏣","european_post_office":"🏤","hospital":"🏥","bank":"🏦","atm":"🏧","hotel":"🏨","love_hotel":"🏩","convenience_store":"🏪","school":"🏫","department_store":"🏬","factory":"🏭","izakaya_lantern":"🏮","lantern":"🏮","japanese_castle":"🏯","european_castle":"🏰","rainbow-flag":"🏳️‍🌈","waving_white_flag":"🏳️","flag-england":"🏴󠁧󠁢󠁥󠁮󠁧󠁿","flag-scotland":"🏴󠁧󠁢󠁳󠁣󠁴󠁿","flag-wales":"🏴󠁧󠁢󠁷󠁬󠁳󠁿","waving_black_flag":"🏴","rosette":"🏵️","label":"🏷️","badminton_racquet_and_shuttlecock":"🏸","bow_and_arrow":"🏹","amphora":"🏺","skin-tone-2":"🏻","skin-tone-3":"🏼","skin-tone-4":"🏽","skin-tone-5":"🏾","skin-tone-6":"🏿","rat":"🐀","mouse2":"🐁","ox":"🐂","water_buffalo":"🐃","cow2":"🐄","tiger2":"🐅","leopard":"🐆","rabbit2":"🐇","cat2":"🐈","dragon":"🐉","crocodile":"🐊","whale2":"🐋","snail":"🐌","snake":"🐍","racehorse":"🐎","ram":"🐏","goat":"🐐","sheep":"🐑","monkey":"🐒","rooster":"🐓","chicken":"🐔","dog2":"🐕","pig2":"🐖","boar":"🐗","elephant":"🐘","octopus":"🐙","shell":"🐚","bug":"🐛","ant":"🐜","bee":"🐝","honeybee":"🐝","beetle":"🐞","fish":"🐟","tropical_fish":"🐠","blowfish":"🐡","turtle":"🐢","hatching_chick":"🐣","baby_chick":"🐤","hatched_chick":"🐥","bird":"🐦","penguin":"🐧","koala":"🐨","poodle":"🐩","dromedary_camel":"🐪","camel":"🐫","dolphin":"🐬","flipper":"🐬","mouse":"🐭","cow":"🐮","tiger":"🐯","rabbit":"🐰","cat":"🐱","dragon_face":"🐲","whale":"🐳","horse":"🐴","monkey_face":"🐵","dog":"🐶","pig":"🐷","frog":"🐸","hamster":"🐹","wolf":"🐺","bear":"🐻","panda_face":"🐼","pig_nose":"🐽","feet":"🐾","paw_prints":"🐾","chipmunk":"🐿️","eyes":"👀","eye-in-speech-bubble":"👁️‍🗨️","eye":"👁️","ear":"👂","nose":"👃","lips":"👄","tongue":"👅","point_up_2":"👆","point_down":"👇","point_left":"👈","point_right":"👉","facepunch":"👊","punch":"👊","wave":"👋","ok_hand":"👌","+1":"👍","thumbsup":"👍","-1":"👎","thumbsdown":"👎","clap":"👏","open_hands":"👐","crown":"👑","womans_hat":"👒","eyeglasses":"👓","necktie":"👔","shirt":"👕","tshirt":"👕","jeans":"👖","dress":"👗","kimono":"👘","bikini":"👙","womans_clothes":"👚","purse":"👛","handbag":"👜","pouch":"👝","mans_shoe":"👞","shoe":"👞","athletic_shoe":"👟","high_heel":"👠","sandal":"👡","boot":"👢","footprints":"👣","bust_in_silhouette":"👤","busts_in_silhouette":"👥","boy":"👦","girl":"👧","male-farmer":"👨‍🌾","male-cook":"👨‍🍳","male-student":"👨‍🎓","male-singer":"👨‍🎤","male-artist":"👨‍🎨","male-teacher":"👨‍🏫","male-factory-worker":"👨‍🏭","man-boy-boy":"👨‍👦‍👦","man-boy":"👨‍👦","man-girl-boy":"👨‍👧‍👦","man-girl-girl":"👨‍👧‍👧","man-girl":"👨‍👧","man-man-boy":"👨‍👨‍👦","man-man-boy-boy":"👨‍👨‍👦‍👦","man-man-girl":"👨‍👨‍👧","man-man-girl-boy":"👨‍👨‍👧‍👦","man-man-girl-girl":"👨‍👨‍👧‍👧","man-woman-boy":"👨‍👩‍👦","family":"👨‍👩‍👦","man-woman-boy-boy":"👨‍👩‍👦‍👦","man-woman-girl":"👨‍👩‍👧","man-woman-girl-boy":"👨‍👩‍👧‍👦","man-woman-girl-girl":"👨‍👩‍👧‍👧","male-technologist":"👨‍💻","male-office-worker":"👨‍💼","male-mechanic":"👨‍🔧","male-scientist":"👨‍🔬","male-astronaut":"👨‍🚀","male-firefighter":"👨‍🚒","male-doctor":"👨‍⚕️","male-judge":"👨‍⚖️","male-pilot":"👨‍✈️","man-heart-man":"👨‍❤️‍👨","man-kiss-man":"👨‍❤️‍💋‍👨","man":"👨","female-farmer":"👩‍🌾","female-cook":"👩‍🍳","female-student":"👩‍🎓","female-singer":"👩‍🎤","female-artist":"👩‍🎨","female-teacher":"👩‍🏫","female-factory-worker":"👩‍🏭","woman-boy-boy":"👩‍👦‍👦","woman-boy":"👩‍👦","woman-girl-boy":"👩‍👧‍👦","woman-girl-girl":"👩‍👧‍👧","woman-girl":"👩‍👧","woman-woman-boy":"👩‍👩‍👦","woman-woman-boy-boy":"👩‍👩‍👦‍👦","woman-woman-girl":"👩‍👩‍👧","woman-woman-girl-boy":"👩‍👩‍👧‍👦","woman-woman-girl-girl":"👩‍👩‍👧‍👧","female-technologist":"👩‍💻","female-office-worker":"👩‍💼","female-mechanic":"👩‍🔧","female-scientist":"👩‍🔬","female-astronaut":"👩‍🚀","female-firefighter":"👩‍🚒","female-doctor":"👩‍⚕️","female-judge":"👩‍⚖️","female-pilot":"👩‍✈️","woman-heart-man":"👩‍❤️‍👨","couple_with_heart":"👩‍❤️‍👨","woman-heart-woman":"👩‍❤️‍👩","woman-kiss-man":"👩‍❤️‍💋‍👨","couplekiss":"👩‍❤️‍💋‍👨","woman-kiss-woman":"👩‍❤️‍💋‍👩","woman":"👩","couple":"👫","man_and_woman_holding_hands":"👫","two_men_holding_hands":"👬","two_women_holding_hands":"👭","female-police-officer":"👮‍♀️","male-police-officer":"👮‍♂️","cop":"👮‍♂️","woman-with-bunny-ears-partying":"👯‍♀️","dancers":"👯‍♀️","man-with-bunny-ears-partying":"👯‍♂️","bride_with_veil":"👰","blond-haired-woman":"👱‍♀️","blond-haired-man":"👱‍♂️","person_with_blond_hair":"👱‍♂️","man_with_gua_pi_mao":"👲","woman-wearing-turban":"👳‍♀️","man-wearing-turban":"👳‍♂️","man_with_turban":"👳‍♂️","older_man":"👴","older_woman":"👵","baby":"👶","female-construction-worker":"👷‍♀️","male-construction-worker":"👷‍♂️","construction_worker":"👷‍♂️","princess":"👸","japanese_ogre":"👹","japanese_goblin":"👺","ghost":"👻","angel":"👼","alien":"👽","space_invader":"👾","imp":"👿","skull":"💀","woman-tipping-hand":"💁‍♀️","information_desk_person":"💁‍♀️","man-tipping-hand":"💁‍♂️","female-guard":"💂‍♀️","male-guard":"💂‍♂️","guardsman":"💂‍♂️","dancer":"💃","lipstick":"💄","nail_care":"💅","woman-getting-massage":"💆‍♀️","massage":"💆‍♀️","man-getting-massage":"💆‍♂️","woman-getting-haircut":"💇‍♀️","haircut":"💇‍♀️","man-getting-haircut":"💇‍♂️","barber":"💈","syringe":"💉","pill":"💊","kiss":"💋","love_letter":"💌","ring":"💍","gem":"💎","bouquet":"💐","wedding":"💒","heartbeat":"💓","broken_heart":"💔","two_hearts":"💕","sparkling_heart":"💖","heartpulse":"💗","cupid":"💘","blue_heart":"💙","green_heart":"💚","yellow_heart":"💛","purple_heart":"💜","gift_heart":"💝","revolving_hearts":"💞","heart_decoration":"💟","diamond_shape_with_a_dot_inside":"💠","bulb":"💡","anger":"💢","bomb":"💣","zzz":"💤","boom":"💥","collision":"💥","sweat_drops":"💦","droplet":"💧","dash":"💨","hankey":"💩","poop":"💩","shit":"💩","muscle":"💪","dizzy":"💫","speech_balloon":"💬","thought_balloon":"💭","white_flower":"💮","moneybag":"💰","currency_exchange":"💱","heavy_dollar_sign":"💲","credit_card":"💳","yen":"💴","dollar":"💵","euro":"💶","pound":"💷","money_with_wings":"💸","chart":"💹","seat":"💺","computer":"💻","briefcase":"💼","minidisc":"💽","floppy_disk":"💾","cd":"💿","dvd":"📀","file_folder":"📁","open_file_folder":"📂","page_with_curl":"📃","page_facing_up":"📄","date":"📅","calendar":"📆","card_index":"📇","chart_with_upwards_trend":"📈","chart_with_downwards_trend":"📉","bar_chart":"📊","clipboard":"📋","pushpin":"📌","round_pushpin":"📍","paperclip":"📎","straight_ruler":"📏","triangular_ruler":"📐","bookmark_tabs":"📑","ledger":"📒","notebook":"📓","notebook_with_decorative_cover":"📔","closed_book":"📕","book":"📖","open_book":"📖","green_book":"📗","blue_book":"📘","orange_book":"📙","books":"📚","name_badge":"📛","scroll":"📜","memo":"📝","pencil":"📝","telephone_receiver":"📞","pager":"📟","fax":"📠","satellite_antenna":"📡","loudspeaker":"📢","mega":"📣","outbox_tray":"📤","inbox_tray":"📥","package":"📦","e-mail":"📧","incoming_envelope":"📨","envelope_with_arrow":"📩","mailbox_closed":"📪","mailbox":"📫","mailbox_with_mail":"📬","mailbox_with_no_mail":"📭","postbox":"📮","postal_horn":"📯","newspaper":"📰","iphone":"📱","calling":"📲","vibration_mode":"📳","mobile_phone_off":"📴","no_mobile_phones":"📵","signal_strength":"📶","camera":"📷","camera_with_flash":"📸","video_camera":"📹","tv":"📺","radio":"📻","vhs":"📼","film_projector":"📽️","prayer_beads":"📿","twisted_rightwards_arrows":"🔀","repeat":"🔁","repeat_one":"🔂","arrows_clockwise":"🔃","arrows_counterclockwise":"🔄","low_brightness":"🔅","high_brightness":"🔆","mute":"🔇","speaker":"🔈","sound":"🔉","loud_sound":"🔊","battery":"🔋","electric_plug":"🔌","mag":"🔍","mag_right":"🔎","lock_with_ink_pen":"🔏","closed_lock_with_key":"🔐","key":"🔑","lock":"🔒","unlock":"🔓","bell":"🔔","no_bell":"🔕","bookmark":"🔖","link":"🔗","radio_button":"🔘","back":"🔙","end":"🔚","on":"🔛","soon":"🔜","top":"🔝","underage":"🔞","keycap_ten":"🔟","capital_abcd":"🔠","abcd":"🔡","symbols":"🔣","abc":"🔤","fire":"🔥","flashlight":"🔦","wrench":"🔧","hammer":"🔨","nut_and_bolt":"🔩","hocho":"🔪","knife":"🔪","gun":"🔫","microscope":"🔬","telescope":"🔭","crystal_ball":"🔮","six_pointed_star":"🔯","beginner":"🔰","trident":"🔱","black_square_button":"🔲","white_square_button":"🔳","red_circle":"🔴","large_blue_circle":"🔵","large_orange_diamond":"🔶","large_blue_diamond":"🔷","small_orange_diamond":"🔸","small_blue_diamond":"🔹","small_red_triangle":"🔺","small_red_triangle_down":"🔻","arrow_up_small":"🔼","arrow_down_small":"🔽","om_symbol":"🕉️","dove_of_peace":"🕊️","kaaba":"🕋","mosque":"🕌","synagogue":"🕍","menorah_with_nine_branches":"🕎","clock1":"🕐","clock2":"🕑","clock3":"🕒","clock4":"🕓","clock5":"🕔","clock6":"🕕","clock7":"🕖","clock8":"🕗","clock9":"🕘","clock10":"🕙","clock11":"🕚","clock12":"🕛","clock130":"🕜","clock230":"🕝","clock330":"🕞","clock430":"🕟","clock530":"🕠","clock630":"🕡","clock730":"🕢","clock830":"🕣","clock930":"🕤","clock1030":"🕥","clock1130":"🕦","clock1230":"🕧","candle":"🕯️","mantelpiece_clock":"🕰️","hole":"🕳️","man_in_business_suit_levitating":"🕴️","female-detective":"🕵️‍♀️","male-detective":"🕵️‍♂️","sleuth_or_spy":"🕵️‍♂️","dark_sunglasses":"🕶️","spider":"🕷️","spider_web":"🕸️","joystick":"🕹️","man_dancing":"🕺","linked_paperclips":"🖇️","lower_left_ballpoint_pen":"🖊️","lower_left_fountain_pen":"🖋️","lower_left_paintbrush":"🖌️","lower_left_crayon":"🖍️","raised_hand_with_fingers_splayed":"🖐️","middle_finger":"🖕","reversed_hand_with_middle_finger_extended":"🖕","spock-hand":"🖖","black_heart":"🖤","desktop_computer":"🖥️","printer":"🖨️","three_button_mouse":"🖱️","trackball":"🖲️","frame_with_picture":"🖼️","card_index_dividers":"🗂️","card_file_box":"🗃️","file_cabinet":"🗄️","wastebasket":"🗑️","spiral_note_pad":"🗒️","spiral_calendar_pad":"🗓️","compression":"🗜️","old_key":"🗝️","rolled_up_newspaper":"🗞️","dagger_knife":"🗡️","speaking_head_in_silhouette":"🗣️","left_speech_bubble":"🗨️","right_anger_bubble":"🗯️","ballot_box_with_ballot":"🗳️","world_map":"🗺️","mount_fuji":"🗻","tokyo_tower":"🗼","statue_of_liberty":"🗽","japan":"🗾","moyai":"🗿","grinning":"😀","grin":"😁","joy":"😂","smiley":"😃","smile":"😄","sweat_smile":"😅","laughing":"😆","satisfied":"😆","innocent":"😇","smiling_imp":"😈","wink":"😉","blush":"😊","yum":"😋","relieved":"😌","heart_eyes":"😍","sunglasses":"😎","smirk":"😏","neutral_face":"😐","expressionless":"😑","unamused":"😒","sweat":"😓","pensive":"😔","confused":"😕","confounded":"😖","kissing":"😗","kissing_heart":"😘","kissing_smiling_eyes":"😙","kissing_closed_eyes":"😚","stuck_out_tongue":"😛","stuck_out_tongue_winking_eye":"😜","stuck_out_tongue_closed_eyes":"😝","disappointed":"😞","worried":"😟","angry":"😠","rage":"😡","cry":"😢","persevere":"😣","triumph":"😤","disappointed_relieved":"😥","frowning":"😦","anguished":"😧","fearful":"😨","weary":"😩","sleepy":"😪","tired_face":"😫","grimacing":"😬","sob":"😭","open_mouth":"😮","hushed":"😯","cold_sweat":"😰","scream":"😱","astonished":"😲","flushed":"😳","sleeping":"😴","dizzy_face":"😵","no_mouth":"😶","mask":"😷","smile_cat":"😸","joy_cat":"😹","smiley_cat":"😺","heart_eyes_cat":"😻","smirk_cat":"😼","kissing_cat":"😽","pouting_cat":"😾","crying_cat_face":"😿","scream_cat":"🙀","slightly_frowning_face":"🙁","slightly_smiling_face":"🙂","upside_down_face":"🙃","face_with_rolling_eyes":"🙄","woman-gesturing-no":"🙅‍♀️","no_good":"🙅‍♀️","man-gesturing-no":"🙅‍♂️","woman-gesturing-ok":"🙆‍♀️","ok_woman":"🙆‍♀️","man-gesturing-ok":"🙆‍♂️","woman-bowing":"🙇‍♀️","man-bowing":"🙇‍♂️","bow":"🙇‍♂️","see_no_evil":"🙈","hear_no_evil":"🙉","speak_no_evil":"🙊","woman-raising-hand":"🙋‍♀️","raising_hand":"🙋‍♀️","man-raising-hand":"🙋‍♂️","raised_hands":"🙌","woman-frowning":"🙍‍♀️","person_frowning":"🙍‍♀️","man-frowning":"🙍‍♂️","woman-pouting":"🙎‍♀️","person_with_pouting_face":"🙎‍♀️","man-pouting":"🙎‍♂️","pray":"🙏","rocket":"🚀","helicopter":"🚁","steam_locomotive":"🚂","railway_car":"🚃","bullettrain_side":"🚄","bullettrain_front":"🚅","train2":"🚆","metro":"🚇","light_rail":"🚈","station":"🚉","tram":"🚊","train":"🚋","bus":"🚌","oncoming_bus":"🚍","trolleybus":"🚎","busstop":"🚏","minibus":"🚐","ambulance":"🚑","fire_engine":"🚒","police_car":"🚓","oncoming_police_car":"🚔","taxi":"🚕","oncoming_taxi":"🚖","car":"🚗","red_car":"🚗","oncoming_automobile":"🚘","blue_car":"🚙","truck":"🚚","articulated_lorry":"🚛","tractor":"🚜","monorail":"🚝","mountain_railway":"🚞","suspension_railway":"🚟","mountain_cableway":"🚠","aerial_tramway":"🚡","ship":"🚢","woman-rowing-boat":"🚣‍♀️","man-rowing-boat":"🚣‍♂️","rowboat":"🚣‍♂️","speedboat":"🚤","traffic_light":"🚥","vertical_traffic_light":"🚦","construction":"🚧","rotating_light":"🚨","triangular_flag_on_post":"🚩","door":"🚪","no_entry_sign":"🚫","smoking":"🚬","no_smoking":"🚭","put_litter_in_its_place":"🚮","do_not_litter":"🚯","potable_water":"🚰","non-potable_water":"🚱","bike":"🚲","no_bicycles":"🚳","woman-biking":"🚴‍♀️","man-biking":"🚴‍♂️","bicyclist":"🚴‍♂️","woman-mountain-biking":"🚵‍♀️","man-mountain-biking":"🚵‍♂️","mountain_bicyclist":"🚵‍♂️","woman-walking":"🚶‍♀️","man-walking":"🚶‍♂️","walking":"🚶‍♂️","no_pedestrians":"🚷","children_crossing":"🚸","mens":"🚹","womens":"🚺","restroom":"🚻","baby_symbol":"🚼","toilet":"🚽","wc":"🚾","shower":"🚿","bath":"🛀","bathtub":"🛁","passport_control":"🛂","customs":"🛃","baggage_claim":"🛄","left_luggage":"🛅","couch_and_lamp":"🛋️","sleeping_accommodation":"🛌","shopping_bags":"🛍️","bellhop_bell":"🛎️","bed":"🛏️","place_of_worship":"🛐","octagonal_sign":"🛑","shopping_trolley":"🛒","hammer_and_wrench":"🛠️","shield":"🛡️","oil_drum":"🛢️","motorway":"🛣️","railway_track":"🛤️","motor_boat":"🛥️","small_airplane":"🛩️","airplane_departure":"🛫","airplane_arriving":"🛬","satellite":"🛰️","passenger_ship":"🛳️","scooter":"🛴","motor_scooter":"🛵","canoe":"🛶","sled":"🛷","flying_saucer":"🛸","zipper_mouth_face":"🤐","money_mouth_face":"🤑","face_with_thermometer":"🤒","nerd_face":"🤓","thinking_face":"🤔","face_with_head_bandage":"🤕","robot_face":"🤖","hugging_face":"🤗","the_horns":"🤘","sign_of_the_horns":"🤘","call_me_hand":"🤙","raised_back_of_hand":"🤚","left-facing_fist":"🤛","right-facing_fist":"🤜","handshake":"🤝","crossed_fingers":"🤞","hand_with_index_and_middle_fingers_crossed":"🤞","i_love_you_hand_sign":"🤟","face_with_cowboy_hat":"🤠","clown_face":"🤡","nauseated_face":"🤢","rolling_on_the_floor_laughing":"🤣","drooling_face":"🤤","lying_face":"🤥","woman-facepalming":"🤦‍♀️","man-facepalming":"🤦‍♂️","face_palm":"🤦","sneezing_face":"🤧","face_with_raised_eyebrow":"🤨","face_with_one_eyebrow_raised":"🤨","star-struck":"🤩","grinning_face_with_star_eyes":"🤩","zany_face":"🤪","grinning_face_with_one_large_and_one_small_eye":"🤪","shushing_face":"🤫","face_with_finger_covering_closed_lips":"🤫","face_with_symbols_on_mouth":"🤬","serious_face_with_symbols_covering_mouth":"🤬","face_with_hand_over_mouth":"🤭","smiling_face_with_smiling_eyes_and_hand_covering_mouth":"🤭","face_vomiting":"🤮","face_with_open_mouth_vomiting":"🤮","exploding_head":"🤯","shocked_face_with_exploding_head":"🤯","pregnant_woman":"🤰","breast-feeding":"🤱","palms_up_together":"🤲","selfie":"🤳","prince":"🤴","man_in_tuxedo":"🤵","mrs_claus":"🤶","mother_christmas":"🤶","woman-shrugging":"🤷‍♀️","man-shrugging":"🤷‍♂️","shrug":"🤷","woman-cartwheeling":"🤸‍♀️","man-cartwheeling":"🤸‍♂️","person_doing_cartwheel":"🤸","woman-juggling":"🤹‍♀️","man-juggling":"🤹‍♂️","juggling":"🤹","fencer":"🤺","woman-wrestling":"🤼‍♀️","man-wrestling":"🤼‍♂️","wrestlers":"🤼","woman-playing-water-polo":"🤽‍♀️","man-playing-water-polo":"🤽‍♂️","water_polo":"🤽","woman-playing-handball":"🤾‍♀️","man-playing-handball":"🤾‍♂️","handball":"🤾","wilted_flower":"🥀","drum_with_drumsticks":"🥁","clinking_glasses":"🥂","tumbler_glass":"🥃","spoon":"🥄","goal_net":"🥅","first_place_medal":"🥇","second_place_medal":"🥈","third_place_medal":"🥉","boxing_glove":"🥊","martial_arts_uniform":"🥋","curling_stone":"🥌","croissant":"🥐","avocado":"🥑","cucumber":"🥒","bacon":"🥓","potato":"🥔","carrot":"🥕","baguette_bread":"🥖","green_salad":"🥗","shallow_pan_of_food":"🥘","stuffed_flatbread":"🥙","egg":"🥚","glass_of_milk":"🥛","peanuts":"🥜","kiwifruit":"🥝","pancakes":"🥞","dumpling":"🥟","fortune_cookie":"🥠","takeout_box":"🥡","chopsticks":"🥢","bowl_with_spoon":"🥣","cup_with_straw":"🥤","coconut":"🥥","broccoli":"🥦","pie":"🥧","pretzel":"🥨","cut_of_meat":"🥩","sandwich":"🥪","canned_food":"🥫","crab":"🦀","lion_face":"🦁","scorpion":"🦂","turkey":"🦃","unicorn_face":"🦄","eagle":"🦅","duck":"🦆","bat":"🦇","shark":"🦈","owl":"🦉","fox_face":"🦊","butterfly":"🦋","deer":"🦌","gorilla":"🦍","lizard":"🦎","rhinoceros":"🦏","shrimp":"🦐","squid":"🦑","giraffe_face":"🦒","zebra_face":"🦓","hedgehog":"🦔","sauropod":"🦕","t-rex":"🦖","cricket":"🦗","cheese_wedge":"🧀","face_with_monocle":"🧐","adult":"🧑","child":"🧒","older_adult":"🧓","bearded_person":"🧔","person_with_headscarf":"🧕","woman_in_steamy_room":"🧖‍♀️","man_in_steamy_room":"🧖‍♂️","person_in_steamy_room":"🧖‍♂️","woman_climbing":"🧗‍♀️","person_climbing":"🧗‍♀️","man_climbing":"🧗‍♂️","woman_in_lotus_position":"🧘‍♀️","person_in_lotus_position":"🧘‍♀️","man_in_lotus_position":"🧘‍♂️","female_mage":"🧙‍♀️","mage":"🧙‍♀️","male_mage":"🧙‍♂️","female_fairy":"🧚‍♀️","fairy":"🧚‍♀️","male_fairy":"🧚‍♂️","female_vampire":"🧛‍♀️","vampire":"🧛‍♀️","male_vampire":"🧛‍♂️","mermaid":"🧜‍♀️","merman":"🧜‍♂️","merperson":"🧜‍♂️","female_elf":"🧝‍♀️","male_elf":"🧝‍♂️","elf":"🧝‍♂️","female_genie":"🧞‍♀️","male_genie":"🧞‍♂️","genie":"🧞‍♂️","female_zombie":"🧟‍♀️","male_zombie":"🧟‍♂️","zombie":"🧟‍♂️","brain":"🧠","orange_heart":"🧡","billed_cap":"🧢","scarf":"🧣","gloves":"🧤","coat":"🧥","socks":"🧦","bangbang":"‼️","interrobang":"⁉️","tm":"™️","information_source":"ℹ️","left_right_arrow":"↔️","arrow_up_down":"↕️","arrow_upper_left":"↖️","arrow_upper_right":"↗️","arrow_lower_right":"↘️","arrow_lower_left":"↙️","leftwards_arrow_with_hook":"↩️","arrow_right_hook":"↪️","watch":"⌚","hourglass":"⌛","keyboard":"⌨️","eject":"⏏️","fast_forward":"⏩","rewind":"⏪","arrow_double_up":"⏫","arrow_double_down":"⏬","black_right_pointing_double_triangle_with_vertical_bar":"⏭️","black_left_pointing_double_triangle_with_vertical_bar":"⏮️","black_right_pointing_triangle_with_double_vertical_bar":"⏯️","alarm_clock":"⏰","stopwatch":"⏱️","timer_clock":"⏲️","hourglass_flowing_sand":"⏳","double_vertical_bar":"⏸️","black_square_for_stop":"⏹️","black_circle_for_record":"⏺️","m":"Ⓜ️","black_small_square":"▪️","white_small_square":"▫️","arrow_forward":"▶️","arrow_backward":"◀️","white_medium_square":"◻️","black_medium_square":"◼️","white_medium_small_square":"◽","black_medium_small_square":"◾","sunny":"☀️","cloud":"☁️","umbrella":"☂️","snowman":"☃️","comet":"☄️","phone":"☎️","telephone":"☎️","ballot_box_with_check":"☑️","shamrock":"☘️","point_up":"☝️","skull_and_crossbones":"☠️","radioactive_sign":"☢️","biohazard_sign":"☣️","orthodox_cross":"☦️","star_and_crescent":"☪️","peace_symbol":"☮️","yin_yang":"☯️","wheel_of_dharma":"☸️","white_frowning_face":"☹️","relaxed":"☺️","female_sign":"♀️","male_sign":"♂️","gemini":"♊","cancer":"♋","leo":"♌","virgo":"♍","libra":"♎","scorpius":"♏","spades":"♠️","clubs":"♣️","hearts":"♥️","diamonds":"♦️","hotsprings":"♨️","recycle":"♻️","wheelchair":"♿","hammer_and_pick":"⚒️","crossed_swords":"⚔️","medical_symbol":"⚕️","staff_of_aesculapius":"⚕️","scales":"⚖️","alembic":"⚗️","gear":"⚙️","atom_symbol":"⚛️","fleur_de_lis":"⚜️","warning":"⚠️","zap":"⚡","white_circle":"⚪","black_circle":"⚫","coffin":"⚰️","funeral_urn":"⚱️","soccer":"⚽","baseball":"⚾","snowman_without_snow":"⛄","partly_sunny":"⛅","thunder_cloud_and_rain":"⛈️","ophiuchus":"⛎","pick":"⛏️","helmet_with_white_cross":"⛑️","chains":"⛓️","no_entry":"⛔","shinto_shrine":"⛩️","church":"⛪","mountain":"⛰️","umbrella_on_ground":"⛱️","fountain":"⛲","golf":"⛳","ferry":"⛴️","boat":"⛵","sailboat":"⛵","skier":"⛷️","ice_skate":"⛸️","woman-bouncing-ball":"⛹️‍♀️","man-bouncing-ball":"⛹️‍♂️","person_with_ball":"⛹️‍♂️","tent":"⛺","fuelpump":"⛽","scissors":"✂️","airplane":"✈️","email":"✉️","envelope":"✉️","fist":"✊","hand":"✋","raised_hand":"✋","v":"✌️","writing_hand":"✍️","pencil2":"✏️","black_nib":"✒️","heavy_check_mark":"✔️","heavy_multiplication_x":"✖️","latin_cross":"✝️","star_of_david":"✡️","eight_spoked_asterisk":"✳️","eight_pointed_black_star":"✴️","snowflake":"❄️","sparkle":"❇️","x":"❌","negative_squared_cross_mark":"❎","heavy_heart_exclamation_mark_ornament":"❣️","heart":"❤️","arrow_right":"➡️","curly_loop":"➰","loop":"➿","arrow_heading_up":"⤴️","arrow_heading_down":"⤵️","arrow_left":"⬅️","arrow_up":"⬆️","arrow_down":"⬇️","black_large_square":"⬛","white_large_square":"⬜","star":"⭐","o":"⭕","wavy_dash":"〰️","part_alternation_mark":"〽️","congratulations":"㊗️","secret":"㊙️"} \ No newline at end of file diff --git a/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_data_cca2.json b/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_data_cca2.json deleted file mode 100644 index 39e3f0e2..00000000 --- a/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_data_cca2.json +++ /dev/null @@ -1 +0,0 @@ -["AF","AL","DZ","AS","AD","AO","AI","AQ","AG","AR","AM","AW","AU","AT","AZ","BS","BH","BD","BB","BY","BE","BZ","BJ","BM","BT","BO","BA","BW","BV","BR","IO","VG","BN","BG","BF","BI","KH","CM","CA","CV","KY","CF","TD","CL","CN","CX","CC","CO","KM","CK","CR","HR","CU","CW","CY","CZ","CD","DK","DJ","DM","DO","EC","EG","SV","GQ","ER","EE","ET","FK","FO","FJ","FI","FR","GF","PF","TF","GA","GM","GE","DE","GH","GI","GR","GL","GD","GP","GU","GT","GG","GN","GW","GY","HT","HM","HN","HK","HU","IS","IN","ID","IR","IQ","IE","IM","IL","IT","CI","JM","JP","JE","JO","KZ","KE","KI","XK","KW","KG","LA","LV","LB","LS","LR","LY","LI","LT","LU","MO","MK","MG","MW","MY","MV","ML","MT","MH","MQ","MR","MU","YT","MX","FM","MD","MC","MN","ME","MS","MA","MZ","MM","NA","NR","NP","NL","NC","NZ","NI","NE","NG","NU","NF","KP","MP","NO","OM","PK","PW","PS","PA","PG","PY","PE","PH","PN","PL","PT","PR","QA","CG","RO","RU","RW","RE","BL","KN","LC","MF","PM","VC","WS","SM","SA","SN","RS","SC","SL","SG","SX","SK","SI","SB","SO","ZA","GS","KR","SS","ES","LK","SD","SR","SJ","SZ","SE","CH","SY","ST","TW","TJ","TZ","TH","TL","TG","TK","TO","TT","TN","TR","TM","TC","TV","UG","UA","AE","GB","US","UM","VI","UY","UZ","VU","VA","VE","VN","WF","EH","YE","ZM","ZW","AX"] diff --git a/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_data_countries.json b/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_data_countries.json deleted file mode 100644 index a7472ace..00000000 --- a/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_data_countries.json +++ /dev/null @@ -1 +0,0 @@ -{"AF": {"currency": "AFN", "callingCode": "93", "flag": "", "name": {"common": "Afghanistan", "cym": "Affganistan", "deu": "Afghanistan", "fra": "Afghanistan", "hrv": "Afganistan", "ita": "Afghanistan", "jpn": "アフガニスタン", "nld": "Afghanistan", "por": "Afeganistão", "rus": "Афганистан", "spa": "Afganistán", "svk": "Afganistan", "fin": "Afganistan", "zho": "阿富汗", "isr": "אפגניסטן"}}, "AL": {"currency": "ALL", "callingCode": "355", "flag": "", "name": {"common": "Albania", "cym": "Albania", "deu": "Albanien", "fra": "Albanie", "hrv": "Albanija", "ita": "Albania", "jpn": "アルバニア", "nld": "Albanië", "por": "Albânia", "rus": "Албания", "spa": "Albania", "svk": "Albánsko", "fin": "Albania", "zho": "阿尔巴尼亚", "isr": "אלבניה"}}, "DZ": {"currency": "DZD", "callingCode": "213", "flag": "", "name": {"common": "Algeria", "cym": "Algeria", "deu": "Algerien", "fra": "Algérie", "hrv": "Alžir", "ita": "Algeria", "jpn": "アルジェリア", "nld": "Algerije", "por": "Argélia", "rus": "Алжир", "spa": "Argelia", "svk": "Alžírsko", "fin": "Algeria", "zho": "阿尔及利亚", "isr": "אלג׳יריה"}}, "AS": {"currency": "USD", "callingCode": "1684", "flag": "", "name": {"common": "American Samoa", "deu": "Amerikanisch-Samoa", "fra": "Samoa américaines", "hrv": "Američka Samoa", "ita": "Samoa Americane", "jpn": "アメリカ領サモア", "nld": "Amerikaans Samoa", "por": "Samoa Americana", "rus": "Американское Самоа", "spa": "Samoa Americana", "svk": "Americká Samoa", "fin": "Amerikan Samoa", "zho": "美属萨摩亚", "isr": "סמואה האמריקנית"}}, "AD": {"currency": "EUR", "callingCode": "376", "flag": "", "name": {"common": "Andorra", "cym": "Andorra", "deu": "Andorra", "fra": "Andorre", "hrv": "Andora", "ita": "Andorra", "jpn": "アンドラ", "nld": "Andorra", "por": "Andorra", "rus": "Андорра", "spa": "Andorra", "svk": "Andorra", "fin": "Andorra", "zho": "安道尔", "isr": "אנדורה"}}, "AO": {"currency": "AOA", "callingCode": "244", "flag": "", "name": {"common": "Angola", "cym": "Angola", "deu": "Angola", "fra": "Angola", "hrv": "Angola", "ita": "Angola", "jpn": "アンゴラ", "nld": "Angola", "por": "Angola", "rus": "Ангола", "spa": "Angola", "svk": "Angola", "fin": "Angola", "zho": "安哥拉", "isr": "אנגולה"}}, "AI": {"currency": "XCD", "callingCode": "1264", "flag": "", "name": {"common": "Anguilla", "deu": "Anguilla", "fra": "Anguilla", "hrv": "Angvila", "ita": "Anguilla", "jpn": "アンギラ", "nld": "Anguilla", "por": "Anguilla", "rus": "Ангилья", "spa": "Anguilla", "svk": "Anguilla", "fin": "Anguilla", "zho": "安圭拉", "isr": "אנגילה"}}, "AQ": {"flag": "", "name": {"common": "Antarctica", "cym": "Antarctica", "deu": "Antarktis", "fra": "Antarctique", "hrv": "Antarktika", "ita": "Antartide", "jpn": "南極", "nld": "Antarctica", "por": "Antártida", "rus": "Антарктида", "spa": "Antártida", "svk": "Antarktída", "fin": "Etelämanner", "zho": "南极洲", "isr": "אנטארקטיקה"}}, "AG": {"currency": "XCD", "callingCode": "1268", "flag": "", "name": {"common": "Antigua and Barbuda", "cym": "Antigwa a Barbiwda", "deu": "Antigua und Barbuda", "fra": "Antigua-et-Barbuda", "hrv": "Antigva i Barbuda", "ita": "Antigua e Barbuda", "jpn": "アンティグア・バーブーダ", "nld": "Antigua en Barbuda", "por": "Antígua e Barbuda", "rus": "Антигуа и Барбуда", "spa": "Antigua y Barbuda", "svk": "Antigua a Barbuda", "fin": "Antigua ja Barbuda", "zho": "安提瓜和巴布达", "isr": "אנטיגואה וברבודה"}}, "AR": {"currency": "ARS", "callingCode": "54", "flag": "", "name": {"common": "Argentina", "cym": "Ariannin", "deu": "Argentinien", "fra": "Argentine", "hrv": "Argentina", "ita": "Argentina", "jpn": "アルゼンチン", "nld": "Argentinië", "por": "Argentina", "rus": "Аргентина", "spa": "Argentina", "svk": "Argentína", "fin": "Argentiina", "zho": "阿根廷", "isr": "ארגנטינה"}}, "AM": {"currency": "AMD", "callingCode": "374", "flag": "", "name": {"common": "Armenia", "cym": "Armenia", "deu": "Armenien", "fra": "Arménie", "hrv": "Armenija", "ita": "Armenia", "jpn": "アルメニア", "nld": "Armenië", "por": "Arménia", "rus": "Армения", "spa": "Armenia", "svk": "Arménsko", "fin": "Armenia", "zho": "亚美尼亚", "isr": "ארמניה"}}, "AW": {"currency": "AWG", "callingCode": "297", "flag": "", "name": {"common": "Aruba", "deu": "Aruba", "fra": "Aruba", "hrv": "Aruba", "ita": "Aruba", "jpn": "アルバ", "nld": "Aruba", "por": "Aruba", "rus": "Аруба", "spa": "Aruba", "svk": "Aruba", "fin": "Aruba", "zho": "阿鲁巴", "isr": "ארובה"}}, "AU": {"currency": "AUD", "callingCode": "61", "flag": "", "name": {"common": "Australia", "cym": "Awstralia", "deu": "Australien", "fra": "Australie", "hrv": "Australija", "ita": "Australia", "jpn": "オーストラリア", "nld": "Australië", "por": "Austrália", "rus": "Австралия", "spa": "Australia", "svk": "Austrália", "fin": "Australia", "zho": "澳大利亚", "isr": "אוסטרליה"}}, "AT": {"currency": "EUR", "callingCode": "43", "flag": "", "name": {"common": "Austria", "cym": "Awstria", "deu": "Österreich", "fra": "Autriche", "hrv": "Austrija", "ita": "Austria", "jpn": "オーストリア", "nld": "Oostenrijk", "por": "Áustria", "rus": "Австрия", "spa": "Austria", "svk": "Rakúsko", "fin": "Itävalta", "zho": "奥地利", "isr": "אוסטריה"}}, "AZ": {"currency": "AZN", "callingCode": "994", "flag": "", "name": {"common": "Azerbaijan", "cym": "Aserbaijan", "deu": "Aserbaidschan", "fra": "Azerbaïdjan", "hrv": "Azerbajdžan", "ita": "Azerbaijan", "jpn": "アゼルバイジャン", "nld": "Azerbeidzjan", "por": "Azerbeijão", "rus": "Азербайджан", "spa": "Azerbaiyán", "svk": "Azerbajdžan", "fin": "Azerbaidzan", "zho": "阿塞拜疆", "isr": "אזרבייג׳ן"}}, "BS": {"currency": "BSD", "callingCode": "1242", "flag": "", "name": {"common": "Bahamas", "cym": "Bahamas", "deu": "Bahamas", "fra": "Bahamas", "hrv": "Bahami", "ita": "Bahamas", "jpn": "バハマ", "nld": "Bahama’s", "por": "Bahamas", "rus": "Багамские Острова", "spa": "Bahamas", "svk": "Bahamy", "fin": "Bahamasaaret", "zho": "巴哈马", "isr": "איי בהאמה"}}, "BH": {"currency": "BHD", "callingCode": "973", "flag": "", "name": {"common": "Bahrain", "cym": "Bahrain", "deu": "Bahrain", "fra": "Bahreïn", "hrv": "Bahrein", "ita": "Bahrein", "jpn": "バーレーン", "nld": "Bahrein", "por": "Bahrein", "rus": "Бахрейн", "spa": "Bahrein", "svk": "Bahrajn", "fin": "Bahrain", "zho": "巴林", "isr": "בחריין"}}, "BD": {"currency": "BDT", "callingCode": "880", "flag": "", "name": {"common": "Bangladesh", "cym": "Bangladesh", "deu": "Bangladesch", "fra": "Bangladesh", "hrv": "Bangladeš", "ita": "Bangladesh", "jpn": "バングラデシュ", "nld": "Bangladesh", "por": "Bangladesh", "rus": "Бангладеш", "spa": "Bangladesh", "svk": "Bangladéš", "fin": "Bangladesh", "zho": "孟加拉国", "isr": "בנגלדש"}}, "BB": {"currency": "BBD", "callingCode": "1246", "flag": "", "name": {"common": "Barbados", "cym": "Barbados", "deu": "Barbados", "fra": "Barbade", "hrv": "Barbados", "ita": "Barbados", "jpn": "バルバドス", "nld": "Barbados", "por": "Barbados", "rus": "Барбадос", "spa": "Barbados", "svk": "Barbados", "fin": "Barbados", "zho": "巴巴多斯", "isr": "ברבדוס"}}, "BY": {"currency": "BYR", "callingCode": "375", "flag": "", "name": {"common": "Belarus", "cym": "Belarws", "deu": "Weißrussland", "fra": "Biélorussie", "hrv": "Bjelorusija", "ita": "Bielorussia", "jpn": "ベラルーシ", "nld": "Wit-Rusland", "por": "Bielorússia", "rus": "Белоруссия", "spa": "Bielorrusia", "svk": "Bielorusko", "fin": "Valko-Venäjä", "zho": "白俄罗斯", "isr": "בלארוס"}}, "BE": {"currency": "EUR", "callingCode": "32", "flag": "", "name": {"common": "Belgium", "cym": "Gwlad Belg", "deu": "Belgien", "fra": "Belgique", "hrv": "Belgija", "ita": "Belgio", "jpn": "ベルギー", "nld": "België", "por": "Bélgica", "rus": "Бельгия", "spa": "Bélgica", "svk": "Belgicko", "fin": "Belgia", "zho": "比利时", "isr": "בלגיה"}}, "BZ": {"currency": "BZD", "callingCode": "501", "flag": "", "name": {"common": "Belize", "cym": "Belize", "deu": "Belize", "fra": "Belize", "hrv": "Belize", "ita": "Belize", "jpn": "ベリーズ", "nld": "Belize", "por": "Belize", "rus": "Белиз", "spa": "Belice", "svk": "Belize", "fin": "Belize", "zho": "伯利兹", "isr": "בליז"}}, "BJ": {"currency": "XOF", "callingCode": "229", "flag": "", "name": {"common": "Benin", "cym": "Benin", "deu": "Benin", "fra": "Bénin", "hrv": "Benin", "ita": "Benin", "jpn": "ベナン", "nld": "Benin", "por": "Benin", "rus": "Бенин", "spa": "Benín", "svk": "Benin", "fin": "Benin", "zho": "贝宁", "isr": "בנין"}}, "BM": {"currency": "BMD", "callingCode": "1441", "flag": "", "name": {"common": "Bermuda", "cym": "Bermiwda", "deu": "Bermuda", "fra": "Bermudes", "hrv": "Bermudi", "ita": "Bermuda", "jpn": "バミューダ", "nld": "Bermuda", "por": "Bermudas", "rus": "Бермудские Острова", "spa": "Bermudas", "svk": "Bermudy", "fin": "Bermuda", "zho": "百慕大", "isr": "ברמודה"}}, "BT": {"currency": "BTN", "callingCode": "975", "flag": "", "name": {"common": "Bhutan", "cym": "Bhwtan", "deu": "Bhutan", "fra": "Bhoutan", "hrv": "Butan", "ita": "Bhutan", "jpn": "ブータン", "nld": "Bhutan", "por": "Butão", "rus": "Бутан", "spa": "Bután", "svk": "Bhután", "fin": "Bhutan", "zho": "不丹", "isr": "בהוטן"}}, "BO": {"currency": "BOB", "callingCode": "591", "flag": "", "name": {"common": "Bolivia", "cym": "Bolifia", "deu": "Bolivien", "fra": "Bolivie", "hrv": "Bolivija", "ita": "Bolivia", "jpn": "ボリビア多民族国", "nld": "Bolivia", "por": "Bolívia", "rus": "Боливия", "spa": "Bolivia", "svk": "Bolívija", "fin": "Bolivia", "zho": "玻利维亚", "isr": "בוליביה"}}, "BA": {"currency": "BAM", "callingCode": "387", "flag": "", "name": {"common": "Bosnia and Herzegovina", "cym": "Bosnia a Hercegovina", "deu": "Bosnien und Herzegowina", "fra": "Bosnie-Herzégovine", "hrv": "Bosna i Hercegovina", "ita": "Bosnia ed Erzegovina", "jpn": "ボスニア・ヘルツェゴビナ", "nld": "Bosnië en Herzegovina", "por": "Bósnia e Herzegovina", "rus": "Босния и Герцеговина", "spa": "Bosnia y Herzegovina", "svk": "Bosna a Hercegovina", "fin": "Bosnia ja Hertsegovina", "zho": "波斯尼亚和黑塞哥维那", "isr": "בוסניה והרצגובינה"}}, "BW": {"currency": "BWP", "callingCode": "267", "flag": "", "name": {"common": "Botswana", "deu": "Botswana", "fra": "Botswana", "hrv": "Bocvana", "ita": "Botswana", "jpn": "ボツワナ", "nld": "Botswana", "por": "Botswana", "rus": "Ботсвана", "spa": "Botswana", "svk": "Botswana", "fin": "Botswana", "zho": "博茨瓦纳", "isr": "בוצוואנה"}}, "BV": {"currency": "NOK", "flag": "", "name": {"common": "Bouvet Island", "deu": "Bouvetinsel", "fra": "Île Bouvet", "hrv": "Otok Bouvet", "ita": "Isola Bouvet", "jpn": "ブーベ島", "nld": "Bouveteiland", "por": "Ilha Bouvet", "rus": "Остров Буве", "spa": "Isla Bouvet", "svk": "Bouvetov ostrov", "fin": "Bouvet'nsaari", "zho": "布维岛", "isr": "איי בובה"}}, "BR": {"currency": "BRL", "callingCode": "55", "flag": "", "name": {"common": "Brazil", "cym": "Brasil", "deu": "Brasilien", "fra": "Brésil", "hrv": "Brazil", "ita": "Brasile", "jpn": "ブラジル", "nld": "Brazilië", "por": "Brasil", "rus": "Бразилия", "spa": "Brasil", "svk": "Brazília", "fin": "Brasilia", "zho": "巴西", "isr": "ברזיל"}}, "IO": {"currency": "USD", "callingCode": "246", "flag": "", "name": {"common": "British Indian Ocean Territory", "cym": "Tiriogaeth Brydeinig Cefnfor India", "deu": "Britisches Territorium im Indischen Ozean", "fra": "Territoire britannique de l'océan Indien", "hrv": "Britanski Indijskooceanski teritorij", "ita": "Territorio britannico dell'oceano indiano", "jpn": "イギリス領インド洋地域", "nld": "Britse Gebieden in de Indische Oceaan", "por": "Território Britânico do Oceano Índico", "rus": "Британская территория в Индийском океане", "spa": "Territorio Británico del Océano Índico", "svk": "Britské indickooceánske územie", "fin": "Brittiläinen Intian valtameren alue", "zho": "英属印度洋领地", "isr": "הטריטוריה הבריטית באוקיינוס ההודי"}}, "VG": {"currency": "USD", "callingCode": "1284", "flag": "", "name": {"common": "British Virgin Islands", "deu": "Britische Jungferninseln", "fra": "Îles Vierges britanniques", "hrv": "Britanski Djevičanski Otoci", "ita": "Isole Vergini Britanniche", "jpn": "イギリス領ヴァージン諸島", "nld": "Britse Maagdeneilanden", "por": "Ilhas Virgens", "rus": "Британские Виргинские острова", "spa": "Islas Vírgenes del Reino Unido", "svk": "Panenské ostrovy", "fin": "Neitsytsaaret", "zho": "英属维尔京群岛", "isr": "איי הבתולה הבריטיים"}}, "BN": {"currency": "BND", "callingCode": "673", "flag": "", "name": {"common": "Brunei", "cym": "Brunei", "deu": "Brunei", "fra": "Brunei", "hrv": "Brunej", "ita": "Brunei", "jpn": "ブルネイ・ダルサラーム", "nld": "Brunei", "por": "Brunei", "rus": "Бруней", "spa": "Brunei", "svk": "Brunej", "fin": "Brunei", "zho": "文莱", "isr": "ברוניי"}}, "BG": {"currency": "BGN", "callingCode": "359", "flag": "", "name": {"common": "Bulgaria", "cym": "Bwlgaria", "deu": "Bulgarien", "fra": "Bulgarie", "hrv": "Bugarska", "ita": "Bulgaria", "jpn": "ブルガリア", "nld": "Bulgarije", "por": "Bulgária", "rus": "Болгария", "spa": "Bulgaria", "svk": "Bulharsko", "fin": "Bulgaria", "zho": "保加利亚", "isr": "בולגריה"}}, "BF": {"currency": "XOF", "callingCode": "226", "flag": "", "name": {"common": "Burkina Faso", "cym": "Burkina Faso", "deu": "Burkina Faso", "fra": "Burkina Faso", "hrv": "Burkina Faso", "ita": "Burkina Faso", "jpn": "ブルキナファソ", "nld": "Burkina Faso", "por": "Burkina Faso", "rus": "Буркина-Фасо", "spa": "Burkina Faso", "svk": "Burkina Faso", "fin": "Burkina Faso", "zho": "布基纳法索", "isr": "בורקינה פאסו"}}, "BI": {"currency": "BIF", "callingCode": "257", "flag": "", "name": {"common": "Burundi", "cym": "Bwrwndi", "deu": "Burundi", "fra": "Burundi", "hrv": "Burundi", "ita": "Burundi", "jpn": "ブルンジ", "nld": "Burundi", "por": "Burundi", "rus": "Бурунди", "spa": "Burundi", "svk": "Burundi", "fin": "Burundi", "zho": "布隆迪", "isr": "בורונדי"}}, "KH": {"currency": "KHR", "callingCode": "855", "flag": "", "name": {"common": "Cambodia", "cym": "Cambodia", "deu": "Kambodscha", "fra": "Cambodge", "hrv": "Kambodža", "ita": "Cambogia", "jpn": "カンボジア", "nld": "Cambodja", "por": "Camboja", "rus": "Камбоджа", "spa": "Camboya", "svk": "Kambodža", "fin": "Kambodža", "zho": "柬埔寨", "isr": "קמבודיה"}}, "CM": {"currency": "XAF", "callingCode": "237", "flag": "", "name": {"common": "Cameroon", "cym": "Camerŵn", "deu": "Kamerun", "fra": "Cameroun", "hrv": "Kamerun", "ita": "Camerun", "jpn": "カメルーン", "nld": "Kameroen", "por": "Camarões", "rus": "Камерун", "spa": "Camerún", "svk": "Kamerun", "fin": "Kamerun", "zho": "喀麦隆", "isr": "קמרון"}}, "CA": {"currency": "CAD", "callingCode": "1", "flag": "", "name": {"common": "Canada", "cym": "Canada", "deu": "Kanada", "fra": "Canada", "hrv": "Kanada", "ita": "Canada", "jpn": "カナダ", "nld": "Canada", "por": "Canadá", "rus": "Канада", "spa": "Canadá", "svk": "Kanada", "fin": "Kanada", "zho": "加拿大", "isr": "קנדה"}}, "CV": {"currency": "CVE", "callingCode": "238", "flag": "", "name": {"common": "Cape Verde", "cym": "Cape Verde", "deu": "Kap Verde", "fra": "Îles du Cap-Vert", "hrv": "Zelenortska Republika", "ita": "Capo Verde", "jpn": "カーボベルデ", "nld": "Kaapverdië", "por": "Cabo Verde", "rus": "Кабо-Верде", "spa": "Cabo Verde", "svk": "Kapverdy", "fin": "Kap Verde", "zho": "佛得角", "isr": "כף ורדה"}}, "KY": {"currency": "KYD", "callingCode": "1345", "flag": "", "name": {"common": "Cayman Islands", "cym": "Ynysoedd_Cayman", "deu": "Kaimaninseln", "fra": "Îles Caïmans", "hrv": "Kajmanski otoci", "ita": "Isole Cayman", "jpn": "ケイマン諸島", "nld": "Caymaneilanden", "por": "Ilhas Caimão", "rus": "Каймановы острова", "spa": "Islas Caimán", "svk": "Kajmanie ostrovy", "fin": "Caymansaaret", "zho": "开曼群岛", "isr": "איי קיימן"}}, "CF": {"currency": "XAF", "callingCode": "236", "flag": "", "name": {"common": "Central African Republic", "cym": "Gweriniaeth Canolbarth Affrica", "deu": "Zentralafrikanische Republik", "fra": "République centrafricaine", "hrv": "Srednjoafrička Republika", "ita": "Repubblica Centrafricana", "jpn": "中央アフリカ共和国", "nld": "Centraal-Afrikaanse Republiek", "por": "República Centro-Africana", "rus": "Центральноафриканская Республика", "spa": "República Centroafricana", "svk": "Stredoafrická republika", "fin": "Keski-Afrikan tasavalta", "zho": "中非共和国", "isr": "הרפובליקה של מרכז אפריקה"}}, "TD": {"currency": "XAF", "callingCode": "235", "flag": "", "name": {"common": "Chad", "cym": "Tsiad", "deu": "Tschad", "fra": "Tchad", "hrv": "Čad", "ita": "Ciad", "jpn": "チャド", "nld": "Tsjaad", "por": "Chade", "rus": "Чад", "spa": "Chad", "svk": "Čad", "fin": "Tšad", "zho": "乍得", "isr": "צ׳אד"}}, "CL": {"currency": "CLF", "callingCode": "56", "flag": "", "name": {"common": "Chile", "cym": "Chile", "deu": "Chile", "fra": "Chili", "hrv": "Čile", "ita": "Cile", "jpn": "チリ", "nld": "Chili", "por": "Chile", "rus": "Чили", "spa": "Chile", "svk": "Čile", "fin": "Chile", "zho": "智利", "isr": "צ׳ילה"}}, "CN": {"currency": "CNY", "callingCode": "86", "flag": "", "name": {"common": "China", "cym": "Tsieina", "deu": "China", "fra": "Chine", "hrv": "Kina", "ita": "Cina", "jpn": "中国", "nld": "China", "por": "China", "rus": "Китай", "spa": "China", "svk": "Čína", "fin": "Kiina", "isr": "סין"}}, "CX": {"currency": "AUD", "callingCode": "61", "flag": "", "name": {"common": "Christmas Island", "cym": "Ynys y Nadolig", "deu": "Weihnachtsinsel", "fra": "Île Christmas", "hrv": "Božićni otok", "ita": "Isola di Natale", "jpn": "クリスマス島", "nld": "Christmaseiland", "por": "Ilha do Natal", "rus": "Остров Рождества", "spa": "Isla de Navidad", "svk": "Vianočnú ostrov", "fin": "Joulusaari", "zho": "圣诞岛", "isr": "האי כריסטמס"}}, "CC": {"currency": "AUD", "callingCode": "61", "flag": "", "name": {"common": "Cocos (Keeling) Islands", "cym": "Ynysoedd Cocos", "deu": "Kokosinseln", "fra": "Îles Cocos", "hrv": "Kokosovi Otoci", "ita": "Isole Cocos e Keeling", "jpn": "ココス(キーリング)諸島", "nld": "Cocoseilanden", "por": "Ilhas Cocos (Keeling)", "rus": "Кокосовые острова", "spa": "Islas Cocos o Islas Keeling", "svk": "Kokosové ostrovy", "fin": "Kookossaaret", "zho": "科科斯", "isr": "איי קוקוס (קילינג)"}}, "CO": {"currency": "COP", "callingCode": "57", "flag": "", "name": {"common": "Colombia", "cym": "Colombia", "deu": "Kolumbien", "fra": "Colombie", "hrv": "Kolumbija", "ita": "Colombia", "jpn": "コロンビア", "nld": "Colombia", "por": "Colômbia", "rus": "Колумбия", "spa": "Colombia", "svk": "Kolumbia", "fin": "Kolumbia", "zho": "哥伦比亚", "isr": "קולומביה"}}, "KM": {"currency": "KMF", "callingCode": "269", "flag": "", "name": {"common": "Comoros", "cym": "Comoros", "deu": "Union der Komoren", "fra": "Comores", "hrv": "Komori", "ita": "Comore", "jpn": "コモロ", "nld": "Comoren", "por": "Comores", "rus": "Коморы", "spa": "Comoras", "svk": "Komory", "fin": "Komorit", "zho": "科摩罗", "isr": "קומורו"}}, "CK": {"currency": "NZD", "callingCode": "682", "flag": "", "name": {"common": "Cook Islands", "cym": "Ynysoedd Cook", "deu": "Cookinseln", "fra": "Îles Cook", "hrv": "Cookovo Otočje", "ita": "Isole Cook", "jpn": "クック諸島", "nld": "Cookeilanden", "por": "Ilhas Cook", "rus": "Острова Кука", "spa": "Islas Cook", "svk": "Cookove ostrovy", "fin": "Cookinsaaret", "zho": "库克群岛", "isr": "איי קוק"}}, "CR": {"currency": "CRC", "callingCode": "506", "flag": "", "name": {"common": "Costa Rica", "cym": "Costa Rica", "deu": "Costa Rica", "fra": "Costa Rica", "hrv": "Kostarika", "ita": "Costa Rica", "jpn": "コスタリカ", "nld": "Costa Rica", "por": "Costa Rica", "rus": "Коста-Рика", "spa": "Costa Rica", "svk": "Kostarika", "fin": "Costa Rica", "zho": "哥斯达黎加", "isr": "קוסטה ריקה"}}, "HR": {"currency": "HRK", "callingCode": "385", "flag": "", "name": {"common": "Croatia", "cym": "Croatia", "deu": "Kroatien", "fra": "Croatie", "hrv": "Hrvatska", "ita": "Croazia", "jpn": "クロアチア", "nld": "Kroatië", "por": "Croácia", "rus": "Хорватия", "spa": "Croacia", "svk": "Chorvátsko", "fin": "Kroatia", "zho": "克罗地亚", "isr": "קרואטיה"}}, "CU": {"currency": "CUC", "callingCode": "53", "flag": "", "name": {"common": "Cuba", "cym": "Ciwba", "deu": "Kuba", "fra": "Cuba", "hrv": "Kuba", "ita": "Cuba", "jpn": "キューバ", "nld": "Cuba", "por": "Cuba", "rus": "Куба", "spa": "Cuba", "svk": "Kuba", "fin": "Kuuba", "zho": "古巴", "isr": "קובה"}}, "CW": {"currency": "ANG", "callingCode": "5999", "flag": "", "name": {"common": "Curaçao", "deu": "Curaçao", "fra": "Curaçao", "nld": "Curaçao", "por": "ilha da Curação", "rus": "Кюрасао", "spa": "Curazao", "svk": "CuraÇao", "fin": "Curaçao", "zho": "库拉索", "isr": "קוראסאו"}}, "CY": {"currency": "EUR", "callingCode": "357", "flag": "", "name": {"common": "Cyprus", "cym": "Cyprus", "deu": "Zypern", "fra": "Chypre", "hrv": "Cipar", "ita": "Cipro", "jpn": "キプロス", "nld": "Cyprus", "por": "Chipre", "rus": "Кипр", "spa": "Chipre", "svk": "Cyprus", "fin": "Kypros", "zho": "塞浦路斯", "isr": "קפריסין"}}, "CZ": {"currency": "CZK", "callingCode": "420", "flag": "", "name": {"common": "Czech Republic", "cym": "Y Weriniaeth Tsiec", "deu": "Tschechische Republik", "fra": "République tchèque", "hrv": "Češka", "ita": "Repubblica Ceca", "jpn": "チェコ", "nld": "Tsjechië", "por": "República Checa", "rus": "Чехия", "spa": "República Checa", "svk": "Česko", "fin": "Tšekki", "zho": "捷克", "isr": "צ׳כיה"}}, "CD": {"currency": "CDF", "callingCode": "243", "flag": "", "name": {"common": "DR Congo", "cym": "Gweriniaeth Ddemocrataidd Congo", "deu": "Kongo (Dem. Rep.)", "fra": "Congo (Rép. dém.)", "hrv": "Kongo, Demokratska Republika", "ita": "Congo (Rep. Dem.)", "jpn": "コンゴ民主共和国", "nld": "Congo (DRC)", "por": "República Democrática do Congo", "rus": "Демократическая Республика Конго", "spa": "Congo (Rep. Dem.)", "svk": "Kongo", "fin": "Kongon demokraattinen tasavalta", "zho": "民主刚果", "isr": "קונגו - קינשאסה"}}, "DK": {"currency": "DKK", "callingCode": "45", "flag": "", "name": {"common": "Denmark", "cym": "Denmarc", "deu": "Dänemark", "fra": "Danemark", "hrv": "Danska", "ita": "Danimarca", "jpn": "デンマーク", "nld": "Denemarken", "por": "Dinamarca", "rus": "Дания", "spa": "Dinamarca", "svk": "Dánsko", "fin": "Tanska", "zho": "丹麦", "isr": "דנמרק"}}, "DJ": {"currency": "DJF", "callingCode": "253", "flag": "", "name": {"common": "Djibouti", "cym": "Djibouti", "deu": "Dschibuti", "fra": "Djibouti", "hrv": "Džibuti", "ita": "Gibuti", "jpn": "ジブチ", "nld": "Djibouti", "por": "Djibouti", "rus": "Джибути", "spa": "Djibouti", "svk": "Džibutsko", "fin": "Dijibouti", "zho": "吉布提", "isr": "ג׳יבוטי"}}, "DM": {"currency": "XCD", "callingCode": "1767", "flag": "", "name": {"common": "Dominica", "cym": "Dominica", "deu": "Dominica", "fra": "Dominique", "hrv": "Dominika", "ita": "Dominica", "jpn": "ドミニカ国", "nld": "Dominica", "por": "Dominica", "rus": "Доминика", "spa": "Dominica", "svk": "Dominika", "fin": "Dominica", "zho": "多米尼加", "isr": "דומיניקה"}}, "DO": {"currency": "DOP", "callingCode": "1809", "flag": "", "name": {"common": "Dominican Republic", "cym": "Gweriniaeth_Dominica", "deu": "Dominikanische Republik", "fra": "République dominicaine", "hrv": "Dominikanska Republika", "ita": "Repubblica Dominicana", "jpn": "ドミニカ共和国", "nld": "Dominicaanse Republiek", "por": "República Dominicana", "rus": "Доминиканская Республика", "spa": "República Dominicana", "svk": "Dominikánska republika", "fin": "Dominikaaninen tasavalta", "zho": "多明尼加", "isr": "הרפובליקה הדומיניקנית"}}, "EC": {"currency": "USD", "callingCode": "593", "flag": "", "name": {"common": "Ecuador", "cym": "Ecwador", "deu": "Ecuador", "fra": "Équateur", "hrv": "Ekvador", "ita": "Ecuador", "jpn": "エクアドル", "nld": "Ecuador", "por": "Equador", "rus": "Эквадор", "spa": "Ecuador", "svk": "Ekvádor", "fin": "Ecuador", "zho": "厄瓜多尔", "isr": "אקוודור"}}, "EG": {"currency": "EGP", "callingCode": "20", "flag": "", "name": {"common": "Egypt", "cym": "Yr Aifft", "deu": "Ägypten", "fra": "Égypte", "hrv": "Egipat", "ita": "Egitto", "jpn": "エジプト", "nld": "Egypte", "por": "Egito", "rus": "Египет", "spa": "Egipto", "svk": "Egypt", "fin": "Egypti", "zho": "埃及", "isr": "מצרים"}}, "SV": {"currency": "SVC", "callingCode": "503", "flag": "", "name": {"common": "El Salvador", "cym": "El Salvador", "deu": "El Salvador", "fra": "Salvador", "hrv": "Salvador", "ita": "El Salvador", "jpn": "エルサルバドル", "nld": "El Salvador", "por": "El Salvador", "rus": "Сальвадор", "spa": "El Salvador", "svk": "Salvádor", "fin": "El Salvador", "zho": "萨尔瓦多", "isr": "אל סלבדור"}}, "GQ": {"currency": "XAF", "callingCode": "240", "flag": "", "name": {"common": "Equatorial Guinea", "cym": "Gini Gyhydeddol", "deu": "Äquatorialguinea", "fra": "Guinée équatoriale", "hrv": "Ekvatorijalna Gvineja", "ita": "Guinea Equatoriale", "jpn": "赤道ギニア", "nld": "Equatoriaal-Guinea", "por": "Guiné Equatorial", "rus": "Экваториальная Гвинея", "spa": "Guinea Ecuatorial", "svk": "Rovníková Guinea", "fin": "Päiväntasaajan Guinea", "zho": "赤道几内亚", "isr": "גינאה המשוונית"}}, "ER": {"currency": "ERN", "callingCode": "291", "flag": "", "name": {"common": "Eritrea", "cym": "Eritrea", "deu": "Eritrea", "fra": "Érythrée", "hrv": "Eritreja", "ita": "Eritrea", "jpn": "エリトリア", "nld": "Eritrea", "por": "Eritreia", "rus": "Эритрея", "spa": "Eritrea", "svk": "Eritrea", "fin": "Eritrea", "zho": "厄立特里亚", "isr": "אריתריאה"}}, "EE": {"currency": "EUR", "callingCode": "372", "flag": "", "name": {"common": "Estonia", "cym": "Estonia", "deu": "Estland", "fra": "Estonie", "hrv": "Estonija", "ita": "Estonia", "jpn": "エストニア", "nld": "Estland", "por": "Estónia", "rus": "Эстония", "spa": "Estonia", "svk": "Estónsko", "fin": "Viro", "zho": "爱沙尼亚", "isr": "אסטוניה"}}, "ET": {"currency": "ETB", "callingCode": "251", "flag": "", "name": {"common": "Ethiopia", "cym": "Ethiopia", "deu": "Äthiopien", "fra": "Éthiopie", "hrv": "Etiopija", "ita": "Etiopia", "jpn": "エチオピア", "nld": "Ethiopië", "por": "Etiópia", "rus": "Эфиопия", "spa": "Etiopía", "svk": "Etiópia", "fin": "Etiopia", "zho": "埃塞俄比亚", "isr": "אתיופיה"}}, "FK": {"currency": "FKP", "callingCode": "500", "flag": "", "name": {"common": "Falkland Islands", "deu": "Falklandinseln", "fra": "Îles Malouines", "hrv": "Falklandski Otoci", "ita": "Isole Falkland o Isole Malvine", "jpn": "フォークランド(マルビナス)諸島", "nld": "Falklandeilanden", "por": "Ilhas Malvinas", "rus": "Фолклендские острова", "spa": "Islas Malvinas", "svk": "Falklandy", "fin": "Falkandinsaaret", "zho": "福克兰群岛", "isr": "איי פוקלנד"}}, "FO": {"currency": "DKK", "callingCode": "298", "flag": "", "name": {"common": "Faroe Islands", "deu": "Färöer-Inseln", "fra": "Îles Féroé", "hrv": "Farski Otoci", "ita": "Isole Far Oer", "jpn": "フェロー諸島", "nld": "Faeröer", "por": "Ilhas Faroé", "rus": "Фарерские острова", "spa": "Islas Faroe", "svk": "Faerské ostrovy", "fin": "Färsaaret", "zho": "法罗群岛", "isr": "איי פארו"}}, "FJ": {"currency": "FJD", "callingCode": "679", "flag": "", "name": {"common": "Fiji", "deu": "Fidschi", "fra": "Fidji", "hrv": "Fiđi", "ita": "Figi", "jpn": "フィジー", "nld": "Fiji", "por": "Fiji", "rus": "Фиджи", "spa": "Fiyi", "svk": "Fidži", "fin": "Fidži", "zho": "斐济", "isr": "פיג׳י"}}, "FI": {"currency": "EUR", "callingCode": "358", "flag": "", "name": {"common": "Finland", "deu": "Finnland", "fra": "Finlande", "hrv": "Finska", "ita": "Finlandia", "jpn": "フィンランド", "nld": "Finland", "por": "Finlândia", "rus": "Финляндия", "spa": "Finlandia", "svk": "Fínsko", "fin": "Suomi", "zho": "芬兰", "isr": "פינלנד"}}, "FR": {"currency": "EUR", "callingCode": "33", "flag": "", "name": {"common": "France", "deu": "Frankreich", "fra": "France", "hrv": "Francuska", "ita": "Francia", "jpn": "フランス", "nld": "Frankrijk", "por": "França", "rus": "Франция", "spa": "Francia", "svk": "Francúzsko", "fin": "Ranska", "zho": "法国", "isr": "צרפת"}}, "GF": {"currency": "EUR", "callingCode": "594", "flag": "", "name": {"common": "French Guiana", "deu": "Französisch Guyana", "fra": "Guyane", "hrv": "Francuska Gvajana", "ita": "Guyana francese", "jpn": "フランス領ギアナ", "nld": "Frans-Guyana", "por": "Guiana Francesa", "rus": "Французская Гвиана", "spa": "Guayana Francesa", "svk": "Guyana", "fin": "Ranskan Guayana", "zho": "法属圭亚那", "isr": "גיאנה הצרפתית"}}, "PF": {"currency": "XPF", "callingCode": "689", "flag": "", "name": {"common": "French Polynesia", "deu": "Französisch-Polynesien", "fra": "Polynésie française", "hrv": "Francuska Polinezija", "ita": "Polinesia Francese", "jpn": "フランス領ポリネシア", "nld": "Frans-Polynesië", "por": "Polinésia Francesa", "rus": "Французская Полинезия", "spa": "Polinesia Francesa", "svk": "Francúzska Polynézia", "fin": "Ranskan Polynesia", "zho": "法属波利尼西亚", "isr": "פולינזיה הצרפתית"}}, "TF": {"currency": "EUR", "flag": "", "name": {"common": "French Southern and Antarctic Lands", "deu": "Französische Süd-und Antarktisgebiete", "fra": "Terres australes et antarctiques françaises", "hrv": "Francuski južni i antarktički teritoriji", "ita": "Territori Francesi del Sud", "jpn": "フランス領南方・南極地域", "nld": "Franse Gebieden in de zuidelijke Indische Oceaan", "por": "Terras Austrais e Antárticas Francesas", "rus": "Французские Южные и Антарктические территории", "spa": "Tierras Australes y Antárticas Francesas", "svk": "Francúzske juŽné a antarktické územia", "fin": "Ranskan eteläiset ja antarktiset alueet", "zho": "法国南部和南极土地", "isr": "הטריטוריות הדרומיות של צרפת"}}, "GA": {"currency": "XAF", "callingCode": "241", "flag": "", "name": {"common": "Gabon", "deu": "Gabun", "fra": "Gabon", "hrv": "Gabon", "ita": "Gabon", "jpn": "ガボン", "nld": "Gabon", "por": "Gabão", "rus": "Габон", "spa": "Gabón", "svk": "Gabon", "fin": "Gabon", "zho": "加蓬", "isr": "גבון"}}, "GM": {"currency": "GMD", "callingCode": "220", "flag": "", "name": {"common": "Gambia", "deu": "Gambia", "fra": "Gambie", "hrv": "Gambija", "ita": "Gambia", "jpn": "ガンビア", "nld": "Gambia", "por": "Gâmbia", "rus": "Гамбия", "spa": "Gambia", "svk": "Gambia", "fin": "Gambia", "zho": "冈比亚", "isr": "גמביה"}}, "GE": {"currency": "GEL", "callingCode": "995", "flag": "", "name": {"common": "Georgia", "deu": "Georgien", "fra": "Géorgie", "hrv": "Gruzija", "ita": "Georgia", "jpn": "グルジア", "nld": "Georgië", "por": "Geórgia", "rus": "Грузия", "spa": "Georgia", "svk": "Gruzínsko", "fin": "Georgia", "zho": "格鲁吉亚", "isr": "גאורגיה"}}, "DE": {"currency": "EUR", "callingCode": "49", "flag": "", "name": {"common": "Germany", "deu": "Deutschland", "fra": "Allemagne", "hrv": "Njemačka", "ita": "Germania", "jpn": "ドイツ", "nld": "Duitsland", "por": "Alemanha", "rus": "Германия", "spa": "Alemania", "svk": "Nemecko", "fin": "Saksa", "zho": "德国", "isr": "גרמניה"}}, "GH": {"currency": "GHS", "callingCode": "233", "flag": "", "name": {"common": "Ghana", "deu": "Ghana", "fra": "Ghana", "hrv": "Gana", "ita": "Ghana", "jpn": "ガーナ", "nld": "Ghana", "por": "Gana", "rus": "Гана", "spa": "Ghana", "svk": "Ghana", "fin": "Ghana", "zho": "加纳", "isr": "גאנה"}}, "GI": {"currency": "GIP", "callingCode": "350", "flag": "", "name": {"common": "Gibraltar", "deu": "Gibraltar", "fra": "Gibraltar", "hrv": "Gibraltar", "ita": "Gibilterra", "jpn": "ジブラルタル", "nld": "Gibraltar", "por": "Gibraltar", "rus": "Гибралтар", "spa": "Gibraltar", "svk": "Gibraltár", "fin": "Gibraltar", "zho": "直布罗陀", "isr": "גיברלטר"}}, "GR": {"currency": "EUR", "callingCode": "30", "flag": "", "name": {"common": "Greece", "deu": "Griechenland", "fra": "Grèce", "hrv": "Grčka", "ita": "Grecia", "jpn": "ギリシャ", "nld": "Griekenland", "por": "Grécia", "rus": "Греция", "spa": "Grecia", "svk": "Greécko", "fin": "Kreikka", "zho": "希腊", "isr": "יוון"}}, "GL": {"currency": "DKK", "callingCode": "299", "flag": "", "name": {"common": "Greenland", "deu": "Grönland", "fra": "Groenland", "hrv": "Grenland", "ita": "Groenlandia", "jpn": "グリーンランド", "nld": "Groenland", "por": "Gronelândia", "rus": "Гренландия", "spa": "Groenlandia", "svk": "Grónsko", "fin": "Groönlanti", "zho": "格陵兰", "isr": "גרינלנד"}}, "GD": {"currency": "XCD", "callingCode": "1473", "flag": "", "name": {"common": "Grenada", "deu": "Grenada", "fra": "Grenade", "hrv": "Grenada", "ita": "Grenada", "jpn": "グレナダ", "nld": "Grenada", "por": "Granada", "rus": "Гренада", "spa": "Grenada", "svk": "Grenada", "fin": "Grenada", "zho": "格林纳达", "isr": "גרנדה"}}, "GP": {"currency": "EUR", "callingCode": "590", "flag": "", "name": {"common": "Guadeloupe", "deu": "Guadeloupe", "fra": "Guadeloupe", "hrv": "Gvadalupa", "ita": "Guadeloupa", "jpn": "グアドループ", "nld": "Guadeloupe", "por": "Guadalupe", "rus": "Гваделупа", "spa": "Guadalupe", "svk": "Guadeloupe", "fin": "Guadeloupe", "zho": "瓜德罗普岛", "isr": "גוואדלופ"}}, "GU": {"currency": "USD", "callingCode": "1671", "flag": "", "name": {"common": "Guam", "deu": "Guam", "fra": "Guam", "hrv": "Guam", "ita": "Guam", "jpn": "グアム", "nld": "Guam", "por": "Guam", "rus": "Гуам", "spa": "Guam", "svk": "Guam", "fin": "Guam", "zho": "关岛", "isr": "גואם"}}, "GT": {"currency": "GTQ", "callingCode": "502", "flag": "", "name": {"common": "Guatemala", "deu": "Guatemala", "fra": "Guatemala", "hrv": "Gvatemala", "ita": "Guatemala", "jpn": "グアテマラ", "nld": "Guatemala", "por": "Guatemala", "rus": "Гватемала", "spa": "Guatemala", "svk": "Guatemala", "fin": "Guatemala", "zho": "危地马拉", "isr": "גואטמלה"}}, "GG": {"currency": "GBP", "callingCode": "44", "flag": "", "name": {"common": "Guernsey", "deu": "Guernsey", "fra": "Guernesey", "hrv": "Guernsey", "ita": "Guernsey", "jpn": "ガーンジー", "nld": "Guernsey", "por": "Guernsey", "rus": "Гернси", "spa": "Guernsey", "svk": "Guernsey", "fin": "Guernsey", "zho": "根西岛", "isr": "גרנסי"}}, "GN": {"currency": "GNF", "callingCode": "224", "flag": "", "name": {"common": "Guinea", "deu": "Guinea", "fra": "Guinée", "hrv": "Gvineja", "ita": "Guinea", "jpn": "ギニア", "nld": "Guinee", "por": "Guiné", "rus": "Гвинея", "spa": "Guinea", "svk": "Guinea", "fin": "Guinea", "zho": "几内亚", "isr": "גינאה"}}, "GW": {"currency": "XOF", "callingCode": "245", "flag": "", "name": {"common": "Guinea-Bissau", "deu": "Guinea-Bissau", "fra": "Guinée-Bissau", "hrv": "Gvineja Bisau", "ita": "Guinea-Bissau", "jpn": "ギニアビサウ", "nld": "Guinee-Bissau", "por": "Guiné-Bissau", "rus": "Гвинея-Бисау", "spa": "Guinea-Bisáu", "svk": "Guinea-Bissau", "fin": "Guinea-Bissau", "zho": "几内亚比绍", "isr": "גינאה ביסאו"}}, "GY": {"currency": "GYD", "callingCode": "592", "flag": "", "name": {"common": "Guyana", "deu": "Guyana", "fra": "Guyana", "hrv": "Gvajana", "ita": "Guyana", "jpn": "ガイアナ", "nld": "Guyana", "por": "Guiana", "rus": "Гайана", "spa": "Guyana", "svk": "Guyana", "fin": "Guayana", "zho": "圭亚那", "isr": "גיאנה"}}, "HT": {"currency": "HTG", "callingCode": "509", "flag": "", "name": {"common": "Haiti", "deu": "Haiti", "fra": "Haïti", "hrv": "Haiti", "ita": "Haiti", "jpn": "ハイチ", "nld": "Haïti", "por": "Haiti", "rus": "Гаити", "spa": "Haiti", "svk": "Haiti", "fin": "Haiti", "zho": "海地", "isr": "האיטי"}}, "HM": {"currency": "AUD", "flag": "", "name": {"common": "Heard Island and McDonald Islands", "deu": "Heard und die McDonaldinseln", "fra": "Îles Heard-et-MacDonald", "hrv": "Otok Heard i otočje McDonald", "ita": "Isole Heard e McDonald", "jpn": "ハード島とマクドナルド諸島", "nld": "Heard-en McDonaldeilanden", "por": "Ilha Heard e Ilhas McDonald", "rus": "Остров Херд и острова Макдональд", "spa": "Islas Heard y McDonald", "svk": "Heardov ostrov", "fin": "Heard ja McDonaldinsaaret", "zho": "赫德岛和麦当劳群岛", "isr": "איי הרד ומקדונלד"}}, "HN": {"currency": "HNL", "callingCode": "504", "flag": "", "name": {"common": "Honduras", "deu": "Honduras", "fra": "Honduras", "hrv": "Honduras", "ita": "Honduras", "jpn": "ホンジュラス", "nld": "Honduras", "por": "Honduras", "rus": "Гондурас", "spa": "Honduras", "fin": "Honduras", "zho": "洪都拉斯", "isr": "הונדורס"}}, "HK": {"currency": "HKD", "callingCode": "852", "flag": "", "name": {"common": "Hong Kong", "deu": "Hongkong", "fra": "Hong Kong", "hrv": "Hong Kong", "ita": "Hong Kong", "jpn": "香港", "nld": "Hongkong", "por": "Hong Kong", "rus": "Гонконг", "spa": "Hong Kong", "svk": "Hongkong", "fin": "Hongkong", "isr": "הונג קונג (מחוז מנהלי מיוחד של סין)"}}, "HU": {"currency": "HUF", "callingCode": "36", "flag": "", "name": {"common": "Hungary", "deu": "Ungarn", "fra": "Hongrie", "hrv": "Mađarska", "ita": "Ungheria", "jpn": "ハンガリー", "nld": "Hongarije", "por": "Hungria", "rus": "Венгрия", "spa": "Hungría", "svk": "Maďarsko", "fin": "Unkari", "zho": "匈牙利", "isr": "הונגריה"}}, "IS": {"currency": "ISK", "callingCode": "354", "flag": "", "name": {"common": "Iceland", "deu": "Island", "fra": "Islande", "hrv": "Island", "ita": "Islanda", "jpn": "アイスランド", "nld": "IJsland", "por": "Islândia", "rus": "Исландия", "spa": "Islandia", "svk": "Island", "fin": "Islanti", "zho": "冰岛", "isr": "איסלנד"}}, "IN": {"currency": "INR", "callingCode": "91", "flag": "", "name": {"common": "India", "deu": "Indien", "fra": "Inde", "hrv": "Indija", "ita": "India", "jpn": "インド", "nld": "India", "por": "Índia", "rus": "Индия", "spa": "India", "svk": "India", "fin": "Intia", "zho": "印度", "isr": "הודו"}}, "ID": {"currency": "IDR", "callingCode": "62", "flag": "", "name": {"common": "Indonesia", "deu": "Indonesien", "fra": "Indonésie", "hrv": "Indonezija", "ita": "Indonesia", "jpn": "インドネシア", "nld": "Indonesië", "por": "Indonésia", "rus": "Индонезия", "spa": "Indonesia", "svk": "Indonézia", "fin": "Indonesia", "zho": "印度尼西亚", "isr": "אינדונזיה"}}, "IR": {"currency": "IRR", "callingCode": "98", "flag": "", "name": {"common": "Iran", "deu": "Iran", "fra": "Iran", "hrv": "Iran", "ita": "Iran", "jpn": "イラン・イスラム共和国", "nld": "Iran", "por": "Irão", "rus": "Иран", "spa": "Iran", "svk": "Irán", "fin": "Iran", "zho": "伊朗", "isr": "איראן"}}, "IQ": {"currency": "IQD", "callingCode": "964", "flag": "", "name": {"common": "Iraq", "deu": "Irak", "fra": "Irak", "hrv": "Irak", "ita": "Iraq", "jpn": "イラク", "nld": "Irak", "por": "Iraque", "rus": "Ирак", "spa": "Irak", "svk": "Irak", "fin": "Irak", "zho": "伊拉克", "isr": "עיראק"}}, "IE": {"currency": "EUR", "callingCode": "353", "flag": "", "name": {"common": "Ireland", "deu": "Irland", "fra": "Irlande", "hrv": "Irska", "ita": "Irlanda", "jpn": "アイルランド", "nld": "Ierland", "por": "Irlanda", "rus": "Ирландия", "spa": "Irlanda", "svk": "Írsko", "fin": "Irlanti", "zho": "爱尔兰", "isr": "אירלנד"}}, "IM": {"currency": "GBP", "callingCode": "44", "flag": "", "name": {"common": "Isle of Man", "deu": "Insel Man", "fra": "Île de Man", "hrv": "Otok Man", "ita": "Isola di Man", "jpn": "マン島", "nld": "Isle of Man", "por": "Ilha de Man", "rus": "Остров Мэн", "spa": "Isla de Man", "svk": "Man", "fin": "Mansaari", "zho": "马恩岛", "isr": "האי מאן"}}, "IL": {"currency": "ILS", "callingCode": "972", "flag": "", "name": {"common": "Israel", "deu": "Israel", "fra": "Israël", "hrv": "Izrael", "ita": "Israele", "jpn": "イスラエル", "nld": "Israël", "por": "Israel", "rus": "Израиль", "spa": "Israel", "svk": "Izrael", "fin": "Israel", "zho": "以色列", "isr": "ישראל"}}, "IT": {"currency": "EUR", "callingCode": "39", "flag": "", "name": {"common": "Italy", "deu": "Italien", "fra": "Italie", "hrv": "Italija", "ita": "Italia", "jpn": "イタリア", "nld": "Italië", "por": "Itália", "rus": "Италия", "spa": "Italia", "svk": "Taliansko", "fin": "Italia", "zho": "意大利", "isr": "איטליה"}}, "CI": {"currency": "XOF", "callingCode": "225", "flag": "", "name": {"common": "Ivory Coast", "deu": "Elfenbeinküste", "fra": "Côte d'Ivoire", "hrv": "Obala Bjelokosti", "ita": "Costa d'Avorio", "jpn": "コートジボワール", "nld": "Ivoorkust", "por": "Costa do Marfim", "rus": "Кот-д’Ивуар", "spa": "Costa de Marfil", "svk": "Pobržie Slonoviny", "fin": "Norsunluurannikko", "zho": "科特迪瓦", "isr": "חוף השנהב"}}, "JM": {"currency": "JMD", "callingCode": "1876", "flag": "", "name": {"common": "Jamaica", "deu": "Jamaika", "fra": "Jamaïque", "hrv": "Jamajka", "ita": "Giamaica", "jpn": "ジャマイカ", "nld": "Jamaica", "por": "Jamaica", "rus": "Ямайка", "spa": "Jamaica", "svk": "Jamajka", "fin": "Jamaika", "zho": "牙买加", "isr": "ג׳מייקה"}}, "JP": {"currency": "JPY", "callingCode": "81", "flag": "", "name": {"common": "Japan", "deu": "Japan", "fra": "Japon", "hrv": "Japan", "ita": "Giappone", "jpn": "日本", "nld": "Japan", "por": "Japão", "rus": "Япония", "spa": "Japón", "svk": "Japonsko", "fin": "Japani", "zho": "日本", "isr": "יפן"}}, "JE": {"currency": "GBP", "callingCode": "44", "flag": "", "name": {"common": "Jersey", "deu": "Jersey", "fra": "Jersey", "hrv": "Jersey", "ita": "Isola di Jersey", "jpn": "ジャージー", "nld": "Jersey", "por": "Jersey", "rus": "Джерси", "spa": "Jersey", "svk": "Jersey", "fin": "Jersey", "zho": "泽西岛", "isr": "ג׳רסי"}}, "JO": {"currency": "JOD", "callingCode": "962", "flag": "", "name": {"common": "Jordan", "deu": "Jordanien", "fra": "Jordanie", "hrv": "Jordan", "ita": "Giordania", "jpn": "ヨルダン", "nld": "Jordanië", "por": "Jordânia", "rus": "Иордания", "spa": "Jordania", "svk": "Jordánsko", "fin": "Jordania", "zho": "约旦", "isr": "ירדן"}}, "KZ": {"currency": "KZT", "callingCode": "76", "flag": "", "name": {"common": "Kazakhstan", "deu": "Kasachstan", "fra": "Kazakhstan", "hrv": "Kazahstan", "ita": "Kazakistan", "jpn": "カザフスタン", "nld": "Kazachstan", "por": "Cazaquistão", "rus": "Казахстан", "spa": "Kazajistán", "svk": "Kazachstan", "fin": "Kazakstan", "zho": "哈萨克斯坦", "isr": "קזחסטן"}}, "KE": {"currency": "KES", "callingCode": "254", "flag": "", "name": {"common": "Kenya", "deu": "Kenia", "fra": "Kenya", "hrv": "Kenija", "ita": "Kenya", "jpn": "ケニア", "nld": "Kenia", "por": "Quénia", "rus": "Кения", "spa": "Kenia", "svk": "Keňa", "fin": "Kenia", "zho": "肯尼亚", "isr": "קניה"}}, "KI": {"currency": "AUD", "callingCode": "686", "flag": "", "name": {"common": "Kiribati", "deu": "Kiribati", "fra": "Kiribati", "hrv": "Kiribati", "ita": "Kiribati", "jpn": "キリバス", "nld": "Kiribati", "por": "Kiribati", "rus": "Кирибати", "spa": "Kiribati", "svk": "Kiribati", "fin": "Kiribati", "zho": "基里巴斯", "isr": "קיריבאטי"}}, "XK": {"currency": "EUR", "callingCode": "383", "flag": "", "name": {"common": "Kosovo", "deu": "Kosovo", "fra": "Kosovo", "hrv": "Kosovo", "ita": "Kosovo", "nld": "Kosovo", "por": "Kosovo", "rus": "Республика Косово", "spa": "Kosovo", "svk": "Kosovo", "fin": "Kosovo", "zho": "科索沃", "isr": "קוסובו"}}, "KW": {"currency": "KWD", "callingCode": "965", "flag": "", "name": {"common": "Kuwait", "deu": "Kuwait", "fra": "Koweït", "hrv": "Kuvajt", "ita": "Kuwait", "jpn": "クウェート", "nld": "Koeweit", "por": "Kuwait", "rus": "Кувейт", "spa": "Kuwait", "svk": "Kuvajt", "fin": "Kuwait", "zho": "科威特", "isr": "כווית"}}, "KG": {"currency": "KGS", "callingCode": "996", "flag": "", "name": {"common": "Kyrgyzstan", "deu": "Kirgisistan", "fra": "Kirghizistan", "hrv": "Kirgistan", "ita": "Kirghizistan", "jpn": "キルギス", "nld": "Kirgizië", "por": "Quirguistão", "rus": "Киргизия", "spa": "Kirguizistán", "svk": "Kirgizsko", "fin": "Kirgisia", "zho": "吉尔吉斯斯坦", "isr": "קירגיזסטן"}}, "LA": {"currency": "LAK", "callingCode": "856", "flag": "", "name": {"common": "Laos", "deu": "Laos", "fra": "Laos", "hrv": "Laos", "ita": "Laos", "jpn": "ラオス人民民主共和国", "nld": "Laos", "por": "Laos", "rus": "Лаос", "spa": "Laos", "svk": "Laos", "fin": "Laos", "zho": "老挝", "isr": "לאוס"}}, "LV": {"currency": "EUR", "callingCode": "371", "flag": "", "name": {"common": "Latvia", "deu": "Lettland", "fra": "Lettonie", "hrv": "Latvija", "ita": "Lettonia", "jpn": "ラトビア", "nld": "Letland", "por": "Letónia", "rus": "Латвия", "spa": "Letonia", "svk": "Lotyšsko", "fin": "Latvia", "zho": "拉脱维亚", "isr": "לטביה"}}, "LB": {"currency": "LBP", "callingCode": "961", "flag": "", "name": {"common": "Lebanon", "deu": "Libanon", "fra": "Liban", "hrv": "Libanon", "ita": "Libano", "jpn": "レバノン", "nld": "Libanon", "por": "Líbano", "rus": "Ливан", "spa": "Líbano", "svk": "Libanon", "fin": "Libanon", "zho": "黎巴嫩", "isr": "לבנון"}}, "LS": {"currency": "LSL", "callingCode": "266", "flag": "", "name": {"common": "Lesotho", "deu": "Lesotho", "fra": "Lesotho", "hrv": "Lesoto", "ita": "Lesotho", "jpn": "レソト", "nld": "Lesotho", "por": "Lesoto", "rus": "Лесото", "spa": "Lesotho", "svk": "Lesotho", "fin": "Lesotho", "zho": "莱索托", "isr": "לסוטו"}}, "LR": {"currency": "LRD", "callingCode": "231", "flag": "", "name": {"common": "Liberia", "deu": "Liberia", "fra": "Liberia", "hrv": "Liberija", "ita": "Liberia", "jpn": "リベリア", "nld": "Liberia", "por": "Libéria", "rus": "Либерия", "spa": "Liberia", "svk": "Libéria", "fin": "Liberia", "zho": "利比里亚", "isr": "ליבריה"}}, "LY": {"currency": "LYD", "callingCode": "218", "flag": "", "name": {"common": "Libya", "deu": "Libyen", "fra": "Libye", "hrv": "Libija", "ita": "Libia", "jpn": "リビア", "nld": "Libië", "por": "Líbia", "rus": "Ливия", "spa": "Libia", "svk": "Líbya", "fin": "Libya", "zho": "利比亚", "isr": "לוב"}}, "LI": {"currency": "CHF", "callingCode": "423", "flag": "", "name": {"common": "Liechtenstein", "deu": "Liechtenstein", "fra": "Liechtenstein", "hrv": "Lihtenštajn", "ita": "Liechtenstein", "jpn": "リヒテンシュタイン", "nld": "Liechtenstein", "por": "Liechtenstein", "rus": "Лихтенштейн", "spa": "Liechtenstein", "svk": "Lichtenštajnsko", "fin": "Liechenstein", "zho": "列支敦士登", "isr": "ליכטנשטיין"}}, "LT": {"currency": "EUR", "callingCode": "370", "flag": "", "name": {"common": "Lithuania", "deu": "Litauen", "fra": "Lituanie", "hrv": "Litva", "ita": "Lituania", "jpn": "リトアニア", "nld": "Litouwen", "por": "Lituânia", "rus": "Литва", "spa": "Lituania", "svk": "Litva", "fin": "Liettua", "zho": "立陶宛", "isr": "ליטא"}}, "LU": {"currency": "EUR", "callingCode": "352", "flag": "", "name": {"common": "Luxembourg", "deu": "Luxemburg", "fra": "Luxembourg", "hrv": "Luksemburg", "ita": "Lussemburgo", "jpn": "ルクセンブルク", "nld": "Luxemburg", "por": "Luxemburgo", "rus": "Люксембург", "spa": "Luxemburgo", "svk": "Luxembursko", "fin": "Luxemburg", "zho": "卢森堡", "isr": "לוקסמבורג"}}, "MO": {"currency": "MOP", "callingCode": "853", "flag": "", "name": {"common": "Macau", "deu": "Macao", "fra": "Macao", "hrv": "Makao", "ita": "Macao", "jpn": "マカオ", "nld": "Macao", "por": "Macau", "rus": "Макао", "spa": "Macao", "fin": "Macao", "isr": "מקאו (מחוז מנהלי מיוחד של סין)"}}, "MK": {"currency": "MKD", "callingCode": "389", "flag": "", "name": {"common": "Macedonia", "deu": "Mazedonien", "fra": "Macédoine", "hrv": "Makedonija", "ita": "Macedonia", "jpn": "マケドニア旧ユーゴスラビア共和国", "nld": "Macedonië", "por": "Macedónia", "rus": "Республика Македония", "spa": "Macedonia", "svk": "Macedónsko", "fin": "Makedonia", "zho": "马其顿", "isr": "מקדוניה"}}, "MG": {"currency": "MGA", "callingCode": "261", "flag": "", "name": {"common": "Madagascar", "deu": "Madagaskar", "fra": "Madagascar", "hrv": "Madagaskar", "ita": "Madagascar", "jpn": "マダガスカル", "nld": "Madagaskar", "por": "Madagáscar", "rus": "Мадагаскар", "spa": "Madagascar", "svk": "Madagaskar", "fin": "Madagaskar", "zho": "马达加斯加", "isr": "מדגסקר"}}, "MW": {"currency": "MWK", "callingCode": "265", "flag": "", "name": {"common": "Malawi", "deu": "Malawi", "fra": "Malawi", "hrv": "Malavi", "ita": "Malawi", "jpn": "マラウイ", "nld": "Malawi", "por": "Malawi", "rus": "Малави", "spa": "Malawi", "svk": "Malawi", "fin": "Malawi", "zho": "马拉维", "isr": "מלאווי"}}, "MY": {"currency": "MYR", "callingCode": "60", "flag": "", "name": {"common": "Malaysia", "deu": "Malaysia", "fra": "Malaisie", "hrv": "Malezija", "ita": "Malesia", "jpn": "マレーシア", "nld": "Maleisië", "por": "Malásia", "rus": "Малайзия", "spa": "Malasia", "svk": "Malajzia", "fin": "Malesia", "zho": "马来西亚", "isr": "מלזיה"}}, "MV": {"currency": "MVR", "callingCode": "960", "flag": "", "name": {"common": "Maldives", "deu": "Malediven", "fra": "Maldives", "hrv": "Maldivi", "ita": "Maldive", "jpn": "モルディブ", "nld": "Maldiven", "por": "Maldivas", "rus": "Мальдивы", "spa": "Maldivas", "svk": "Maldivy", "fin": "Malediivit", "zho": "马尔代夫", "isr": "האיים המלדיביים"}}, "ML": {"currency": "XOF", "callingCode": "223", "flag": "", "name": {"common": "Mali", "deu": "Mali", "fra": "Mali", "hrv": "Mali", "ita": "Mali", "jpn": "マリ", "nld": "Mali", "por": "Mali", "rus": "Мали", "spa": "Mali", "svk": "Mali", "fin": "Mali", "zho": "马里", "isr": "מאלי"}}, "MT": {"currency": "EUR", "callingCode": "356", "flag": "", "name": {"common": "Malta", "deu": "Malta", "fra": "Malte", "hrv": "Malta", "ita": "Malta", "jpn": "マルタ", "nld": "Malta", "por": "Malta", "rus": "Мальта", "spa": "Malta", "svk": "Malta", "fin": "Malta", "zho": "马耳他", "isr": "מלטה"}}, "MH": {"currency": "USD", "callingCode": "692", "flag": "", "name": {"common": "Marshall Islands", "deu": "Marshallinseln", "fra": "Îles Marshall", "hrv": "Maršalovi Otoci", "ita": "Isole Marshall", "jpn": "マーシャル諸島", "nld": "Marshalleilanden", "por": "Ilhas Marshall", "rus": "Маршалловы Острова", "spa": "Islas Marshall", "svk": "Marshallove ostrovy", "fin": "Marshallinsaaret", "zho": "马绍尔群岛", "isr": "איי מרשל"}}, "MQ": {"currency": "EUR", "callingCode": "596", "flag": "", "name": {"common": "Martinique", "deu": "Martinique", "fra": "Martinique", "hrv": "Martinique", "ita": "Martinica", "jpn": "マルティニーク", "nld": "Martinique", "por": "Martinica", "rus": "Мартиника", "spa": "Martinica", "svk": "Martinik", "fin": "Martinique", "zho": "马提尼克", "isr": "מרטיניק"}}, "MR": {"currency": "MRO", "callingCode": "222", "flag": "", "name": {"common": "Mauritania", "deu": "Mauretanien", "fra": "Mauritanie", "hrv": "Mauritanija", "ita": "Mauritania", "jpn": "モーリタニア", "nld": "Mauritanië", "por": "Mauritânia", "rus": "Мавритания", "spa": "Mauritania", "svk": "Mauritánia", "fin": "Mauritania", "zho": "毛里塔尼亚", "isr": "מאוריטניה"}}, "MU": {"currency": "MUR", "callingCode": "230", "flag": "", "name": {"common": "Mauritius", "deu": "Mauritius", "fra": "Île Maurice", "hrv": "Mauricijus", "ita": "Mauritius", "jpn": "モーリシャス", "nld": "Mauritius", "por": "Maurício", "rus": "Маврикий", "spa": "Mauricio", "svk": "Maurícius", "fin": "Mauritius", "zho": "毛里求斯", "isr": "מאוריציוס"}}, "YT": {"currency": "EUR", "callingCode": "262", "flag": "", "name": {"common": "Mayotte", "deu": "Mayotte", "fra": "Mayotte", "hrv": "Mayotte", "ita": "Mayotte", "jpn": "マヨット", "nld": "Mayotte", "por": "Mayotte", "rus": "Майотта", "spa": "Mayotte", "svk": "Mayotte", "fin": "Mayotte", "zho": "马约特", "isr": "מאיוט"}}, "MX": {"currency": "MXN", "callingCode": "52", "flag": "", "name": {"common": "Mexico", "deu": "Mexiko", "fra": "Mexique", "hrv": "Meksiko", "ita": "Messico", "jpn": "メキシコ", "nld": "Mexico", "por": "México", "rus": "Мексика", "spa": "México", "svk": "Mexiko", "fin": "Meksiko", "zho": "墨西哥", "isr": "מקסיקו"}}, "FM": {"currency": "USD", "callingCode": "691", "flag": "", "name": {"common": "Micronesia", "deu": "Mikronesien", "fra": "Micronésie", "hrv": "Mikronezija", "ita": "Micronesia", "jpn": "ミクロネシア連邦", "nld": "Micronesië", "por": "Micronésia", "rus": "Федеративные Штаты Микронезии", "spa": "Micronesia", "svk": "Mikronézia", "fin": "Mikronesia", "zho": "密克罗尼西亚", "isr": "מיקרונזיה"}}, "MD": {"currency": "MDL", "callingCode": "373", "flag": "", "name": {"common": "Moldova", "deu": "Moldawie", "fra": "Moldavie", "hrv": "Moldova", "ita": "Moldavia", "jpn": "モルドバ共和国", "nld": "Moldavië", "por": "Moldávia", "rus": "Молдавия", "spa": "Moldavia", "svk": "Moldavsko", "fin": "Moldova", "zho": "摩尔多瓦", "isr": "מולדובה"}}, "MC": {"currency": "EUR", "callingCode": "377", "flag": "", "name": {"common": "Monaco", "deu": "Monaco", "fra": "Monaco", "hrv": "Monako", "ita": "Principato di Monaco", "jpn": "モナコ", "nld": "Monaco", "por": "Mónaco", "rus": "Монако", "spa": "Mónaco", "svk": "Monako", "fin": "Monaco", "zho": "摩纳哥", "isr": "מונקו"}}, "MN": {"currency": "MNT", "callingCode": "976", "flag": "", "name": {"common": "Mongolia", "deu": "Mongolei", "fra": "Mongolie", "hrv": "Mongolija", "ita": "Mongolia", "jpn": "モンゴル", "nld": "Mongolië", "por": "Mongólia", "rus": "Монголия", "spa": "Mongolia", "svk": "Mongolsko", "fin": "Mongolia", "zho": "蒙古", "isr": "מונגוליה"}}, "ME": {"currency": "EUR", "callingCode": "382", "flag": "", "name": {"common": "Montenegro", "deu": "Montenegro", "fra": "Monténégro", "hrv": "Crna Gora", "ita": "Montenegro", "jpn": "モンテネグロ", "nld": "Montenegro", "por": "Montenegro", "rus": "Черногория", "spa": "Montenegro", "svk": "Čierna Hora", "fin": "Montenegro", "zho": "黑山", "isr": "מונטנגרו"}}, "MS": {"currency": "XCD", "callingCode": "1664", "flag": "", "name": {"common": "Montserrat", "deu": "Montserrat", "fra": "Montserrat", "hrv": "Montserrat", "ita": "Montserrat", "jpn": "モントセラト", "nld": "Montserrat", "por": "Montserrat", "rus": "Монтсеррат", "spa": "Montserrat", "svk": "Montserrat", "fin": "Montserrat", "zho": "蒙特塞拉特", "isr": "מונסראט"}}, "MA": {"currency": "MAD", "callingCode": "212", "flag": "", "name": {"common": "Morocco", "deu": "Marokko", "fra": "Maroc", "hrv": "Maroko", "ita": "Marocco", "jpn": "モロッコ", "nld": "Marokko", "por": "Marrocos", "rus": "Марокко", "spa": "Marruecos", "svk": "Maroko", "fin": "Marokko", "zho": "摩洛哥", "isr": "מרוקו"}}, "MZ": {"currency": "MZN", "callingCode": "258", "flag": "", "name": {"common": "Mozambique", "deu": "Mosambik", "fra": "Mozambique", "hrv": "Mozambik", "ita": "Mozambico", "jpn": "モザンビーク", "nld": "Mozambique", "por": "Moçambique", "rus": "Мозамбик", "spa": "Mozambique", "svk": "Mozambik", "fin": "Mosambik", "zho": "莫桑比克", "isr": "מוזמביק"}}, "MM": {"currency": "MMK", "callingCode": "95", "flag": "", "name": {"common": "Myanmar", "deu": "Myanmar", "fra": "Birmanie", "hrv": "Mijanmar", "ita": "Birmania", "jpn": "ミャンマー", "nld": "Myanmar", "por": "Myanmar", "rus": "Мьянма", "spa": "Myanmar", "svk": "Mjanmarsko", "fin": "Myanmar", "zho": "缅甸", "isr": "מיאנמר (בורמה)"}}, "NA": {"currency": "NAD", "callingCode": "264", "flag": "", "name": {"common": "Namibia", "deu": "Namibia", "fra": "Namibie", "hrv": "Namibija", "ita": "Namibia", "jpn": "ナミビア", "nld": "Namibië", "por": "Namíbia", "rus": "Намибия", "spa": "Namibia", "svk": "Namíbia", "fin": "Namibia", "zho": "纳米比亚", "isr": "נמיביה"}}, "NR": {"currency": "AUD", "callingCode": "674", "flag": "", "name": {"common": "Nauru", "deu": "Nauru", "fra": "Nauru", "hrv": "Nauru", "ita": "Nauru", "jpn": "ナウル", "nld": "Nauru", "por": "Nauru", "rus": "Науру", "spa": "Nauru", "svk": "Nauru", "fin": "Nauru", "zho": "瑙鲁", "isr": "נאורו"}}, "NP": {"currency": "NPR", "callingCode": "977", "flag": "", "name": {"common": "Nepal", "deu": "Népal", "fra": "Népal", "hrv": "Nepal", "ita": "Nepal", "jpn": "ネパール", "nld": "Nepal", "por": "Nepal", "rus": "Непал", "spa": "Nepal", "svk": "Nepál", "fin": "Nepal", "zho": "尼泊尔", "isr": "נפאל"}}, "NL": {"currency": "EUR", "callingCode": "31", "flag": "", "name": {"common": "Netherlands", "deu": "Niederlande", "fra": "Pays-Bas", "hrv": "Nizozemska", "ita": "Paesi Bassi", "jpn": "オランダ", "nld": "Nederland", "por": "Holanda", "rus": "Нидерланды", "spa": "Países Bajos", "svk": "Holansko", "fin": "Alankomaat", "zho": "荷兰", "isr": "הולנד"}}, "NC": {"currency": "XPF", "callingCode": "687", "flag": "", "name": {"common": "New Caledonia", "deu": "Neukaledonien", "fra": "Nouvelle-Calédonie", "hrv": "Nova Kaledonija", "ita": "Nuova Caledonia", "jpn": "ニューカレドニア", "nld": "Nieuw-Caledonië", "por": "Nova Caledónia", "rus": "Новая Каледония", "spa": "Nueva Caledonia", "svk": "Nová Kaledónia", "fin": "Uusi-Kaledonia", "zho": "新喀里多尼亚", "isr": "קלדוניה החדשה"}}, "NZ": {"currency": "NZD", "callingCode": "64", "flag": "", "name": {"common": "New Zealand", "deu": "Neuseeland", "fra": "Nouvelle-Zélande", "hrv": "Novi Zeland", "ita": "Nuova Zelanda", "jpn": "ニュージーランド", "nld": "Nieuw-Zeeland", "por": "Nova Zelândia", "rus": "Новая Зеландия", "spa": "Nueva Zelanda", "svk": "Nový Zéland", "fin": "Uusi-Seelanti", "zho": "新西兰", "isr": "ניו זילנד"}}, "NI": {"currency": "NIO", "callingCode": "505", "flag": "", "name": {"common": "Nicaragua", "deu": "Nicaragua", "fra": "Nicaragua", "hrv": "Nikaragva", "ita": "Nicaragua", "jpn": "ニカラグア", "nld": "Nicaragua", "por": "Nicarágua", "rus": "Никарагуа", "spa": "Nicaragua", "svk": "Nikaragua", "fin": "Nicaragua", "zho": "尼加拉瓜", "isr": "ניקרגואה"}}, "NE": {"currency": "XOF", "callingCode": "227", "flag": "", "name": {"common": "Niger", "deu": "Niger", "fra": "Niger", "hrv": "Niger", "ita": "Niger", "jpn": "ニジェール", "nld": "Niger", "por": "Níger", "rus": "Нигер", "spa": "Níger", "svk": "Niger", "fin": "Niger", "zho": "尼日尔", "isr": "ניז׳ר"}}, "NG": {"currency": "NGN", "callingCode": "234", "flag": "", "name": {"common": "Nigeria", "deu": "Nigeria", "fra": "Nigéria", "hrv": "Nigerija", "ita": "Nigeria", "jpn": "ナイジェリア", "nld": "Nigeria", "por": "Nigéria", "rus": "Нигерия", "spa": "Nigeria", "svk": "Nigéria", "fin": "Nigeria", "zho": "尼日利亚", "isr": "ניגריה"}}, "NU": {"currency": "NZD", "callingCode": "683", "flag": "", "name": {"common": "Niue", "deu": "Niue", "fra": "Niue", "hrv": "Niue", "ita": "Niue", "jpn": "ニウエ", "nld": "Niue", "por": "Niue", "rus": "Ниуэ", "spa": "Niue", "svk": "Niue", "fin": "Niue", "zho": "纽埃", "isr": "ניווה"}}, "NF": {"currency": "AUD", "callingCode": "672", "flag": "", "name": {"common": "Norfolk Island", "deu": "Norfolkinsel", "fra": "Île Norfolk", "hrv": "Otok Norfolk", "ita": "Isola Norfolk", "jpn": "ノーフォーク島", "nld": "Norfolkeiland", "por": "Ilha Norfolk", "rus": "Норфолк", "spa": "Isla de Norfolk", "svk": "Norfolk", "fin": "Norfolkinsaari", "zho": "诺福克岛", "isr": "איי נורפוק"}}, "KP": {"currency": "KPW", "callingCode": "850", "flag": "", "name": {"common": "North Korea", "deu": "Nordkorea", "fra": "Corée du Nord", "hrv": "Sjeverna Koreja", "ita": "Corea del Nord", "jpn": "朝鮮民主主義人民共和国", "nld": "Noord-Korea", "por": "Coreia do Norte", "rus": "Северная Корея", "spa": "Corea del Norte", "svk": "Kórejská ľudovodemokratická republika (KĽR, Severná Kó)", "fin": "Pohjois-Korea", "zho": "朝鲜", "isr": "קוריאה הצפונית"}}, "MP": {"currency": "USD", "callingCode": "1670", "flag": "", "name": {"common": "Northern Mariana Islands", "deu": "Nördliche Marianen", "fra": "Îles Mariannes du Nord", "hrv": "Sjevernomarijanski otoci", "ita": "Isole Marianne Settentrionali", "jpn": "北マリアナ諸島", "nld": "Noordelijke Marianeneilanden", "por": "Marianas Setentrionais", "rus": "Северные Марианские острова", "spa": "Islas Marianas del Norte", "svk": "Severné Mariány", "fin": "Pohjois-Mariaanit", "zho": "北马里亚纳群岛", "isr": "איי מריאנה הצפוניים"}}, "NO": {"currency": "NOK", "callingCode": "47", "flag": "", "name": {"common": "Norway", "deu": "Norwegen", "fra": "Norvège", "hrv": "Norveška", "ita": "Norvegia", "jpn": "ノルウェー", "nld": "Noorwegen", "por": "Noruega", "rus": "Норвегия", "spa": "Noruega", "svk": "Nórsko", "fin": "Norja", "zho": "挪威", "isr": "נורווגיה"}}, "OM": {"currency": "OMR", "callingCode": "968", "flag": "", "name": {"common": "Oman", "deu": "Oman", "fra": "Oman", "hrv": "Oman", "ita": "oman", "jpn": "オマーン", "nld": "Oman", "por": "Omã", "rus": "Оман", "spa": "Omán", "svk": "Omán", "fin": "Oman", "zho": "阿曼", "isr": "עומאן"}}, "PK": {"currency": "PKR", "callingCode": "92", "flag": "", "name": {"common": "Pakistan", "deu": "Pakistan", "fra": "Pakistan", "hrv": "Pakistan", "ita": "Pakistan", "jpn": "パキスタン", "nld": "Pakistan", "por": "Paquistão", "rus": "Пакистан", "spa": "Pakistán", "svk": "Pakistan", "fin": "Pakistan", "zho": "巴基斯坦", "isr": "פקיסטן"}}, "PW": {"currency": "USD", "callingCode": "680", "flag": "", "name": {"common": "Palau", "deu": "Palau", "fra": "Palaos (Palau)", "hrv": "Palau", "ita": "Palau", "jpn": "パラオ", "nld": "Palau", "por": "Palau", "rus": "Палау", "spa": "Palau", "svk": "Palau", "fin": "Palau", "zho": "帕劳", "isr": "פלאו"}}, "PS": {"currency": "ILS", "callingCode": "970", "flag": "", "name": {"common": "Palestine", "deu": "Palästina", "fra": "Palestine", "hrv": "Palestina", "ita": "Palestina", "jpn": "パレスチナ", "nld": "Palestijnse gebieden", "por": "Palestina", "rus": "Палестина", "spa": "Palestina", "svk": "Palestína", "fin": "Palestiina", "zho": "巴勒斯坦", "isr": "השטחים הפלסטיניים"}}, "PA": {"currency": "PAB", "callingCode": "507", "flag": "", "name": {"common": "Panama", "deu": "Panama", "fra": "Panama", "hrv": "Panama", "ita": "Panama", "jpn": "パナマ", "nld": "Panama", "por": "Panamá", "rus": "Панама", "spa": "Panamá", "svk": "Panama", "fin": "Panama", "zho": "巴拿马", "isr": "פנמה"}}, "PG": {"currency": "PGK", "callingCode": "675", "flag": "", "name": {"common": "Papua New Guinea", "deu": "Papua-Neuguinea", "fra": "Papouasie-Nouvelle-Guinée", "hrv": "Papua Nova Gvineja", "ita": "Papua Nuova Guinea", "jpn": "パプアニューギニア", "nld": "Papoea-Nieuw-Guinea", "por": "Papua Nova Guiné", "rus": "Папуа — Новая Гвинея", "spa": "Papúa Nueva Guinea", "svk": "Papua-Nová Guinea", "fin": "Papua-Uusi-Guinea", "zho": "巴布亚新几内亚", "isr": "פפואה גינאה החדשה"}}, "PY": {"currency": "PYG", "callingCode": "595", "flag": "", "name": {"common": "Paraguay", "deu": "Paraguay", "fra": "Paraguay", "hrv": "Paragvaj", "ita": "Paraguay", "jpn": "パラグアイ", "nld": "Paraguay", "por": "Paraguai", "rus": "Парагвай", "spa": "Paraguay", "svk": "Paraguaj", "fin": "Paraguay", "zho": "巴拉圭", "isr": "פרגוואי"}}, "PE": {"currency": "PEN", "callingCode": "51", "flag": "", "name": {"common": "Peru", "deu": "Peru", "fra": "Pérou", "hrv": "Peru", "ita": "Perù", "jpn": "ペルー", "nld": "Peru", "por": "Perú", "rus": "Перу", "spa": "Perú", "svk": "Peru", "fin": "Peru", "zho": "秘鲁", "isr": "פרו"}}, "PH": {"currency": "PHP", "callingCode": "63", "flag": "", "name": {"common": "Philippines", "deu": "Philippinen", "fra": "Philippines", "hrv": "Filipini", "ita": "Filippine", "jpn": "フィリピン", "nld": "Filipijnen", "por": "Filipinas", "rus": "Филиппины", "spa": "Filipinas", "svk": "Filipíny", "fin": "Filippiinit", "zho": "菲律宾", "isr": "הפיליפינים"}}, "PN": {"currency": "NZD", "callingCode": "64", "flag": "", "name": {"common": "Pitcairn Islands", "deu": "Pitcairn", "fra": "Îles Pitcairn", "hrv": "Pitcairnovo otočje", "ita": "Isole Pitcairn", "jpn": "ピトケアン", "nld": "Pitcairneilanden", "por": "Ilhas Pitcairn", "rus": "Острова Питкэрн", "spa": "Islas Pitcairn", "svk": "Pitcairnove ostrovy", "fin": "Pitcairn", "zho": "皮特凯恩群岛", "isr": "איי פיטקרן"}}, "PL": {"currency": "PLN", "callingCode": "48", "flag": "", "name": {"common": "Poland", "deu": "Polen", "fra": "Pologne", "hrv": "Poljska", "ita": "Polonia", "jpn": "ポーランド", "nld": "Polen", "por": "Polónia", "rus": "Польша", "spa": "Polonia", "svk": "Poľsko", "fin": "Puola", "zho": "波兰", "isr": "פולין"}}, "PT": {"currency": "EUR", "callingCode": "351", "flag": "", "name": {"common": "Portugal", "deu": "Portugal", "fra": "Portugal", "hrv": "Portugal", "ita": "Portogallo", "jpn": "ポルトガル", "nld": "Portugal", "por": "Portugal", "rus": "Португалия", "spa": "Portugal", "svk": "Portugalsko", "fin": "Portugali", "zho": "葡萄牙", "isr": "פורטוגל"}}, "PR": {"currency": "USD", "callingCode": "1787", "flag": "", "name": {"common": "Puerto Rico", "deu": "Puerto Rico", "fra": "Porto Rico", "hrv": "Portoriko", "ita": "Porto Rico", "jpn": "プエルトリコ", "nld": "Puerto Rico", "por": "Porto Rico", "rus": "Пуэрто-Рико", "spa": "Puerto Rico", "svk": "Portoriko", "fin": "Puerto Rico", "zho": "波多黎各", "isr": "פוארטו ריקו"}}, "QA": {"currency": "QAR", "callingCode": "974", "flag": "", "name": {"common": "Qatar", "deu": "Katar", "fra": "Qatar", "hrv": "Katar", "ita": "Qatar", "jpn": "カタール", "nld": "Qatar", "por": "Catar", "rus": "Катар", "spa": "Catar", "svk": "Katar", "fin": "Qatar", "zho": "卡塔尔", "isr": "קטאר"}}, "CG": {"currency": "XAF", "callingCode": "242", "flag": "", "name": {"common": "Republic of the Congo", "cym": "Gweriniaeth y Congo", "deu": "Kongo", "fra": "Congo", "hrv": "Kongo", "ita": "Congo", "jpn": "コンゴ共和国", "nld": "Congo", "por": "Congo", "rus": "Республика Конго", "spa": "Congo", "svk": "Kongo", "fin": "Kongo-Brazzaville", "zho": "刚果", "isr": "קונגו - ברזאויל"}}, "RO": {"currency": "RON", "callingCode": "40", "flag": "", "name": {"common": "Romania", "deu": "Rumänien", "fra": "Roumanie", "hrv": "Rumunjska", "ita": "Romania", "jpn": "ルーマニア", "nld": "Roemenië", "por": "Roménia", "rus": "Румыния", "spa": "Rumania", "svk": "Rumunsko", "fin": "Romania", "zho": "罗马尼亚", "isr": "רומניה"}}, "RU": {"currency": "RUB", "callingCode": "7", "flag": "", "name": {"common": "Russia", "deu": "Russland", "fra": "Russie", "hrv": "Rusija", "ita": "Russia", "jpn": "ロシア連邦", "nld": "Rusland", "por": "Rússia", "rus": "Россия", "spa": "Rusia", "svk": "Rusko", "fin": "Venäjä", "zho": "俄罗斯", "isr": "רוסיה"}}, "RW": {"currency": "RWF", "callingCode": "250", "flag": "", "name": {"common": "Rwanda", "deu": "Ruanda", "fra": "Rwanda", "hrv": "Ruanda", "ita": "Ruanda", "jpn": "ルワンダ", "nld": "Rwanda", "por": "Ruanda", "rus": "Руанда", "spa": "Ruanda", "svk": "Rwanda", "fin": "Ruanda", "zho": "卢旺达", "isr": "רואנדה"}}, "RE": {"currency": "EUR", "callingCode": "262", "flag": "", "name": {"common": "Réunion", "deu": "Réunion", "fra": "Réunion", "hrv": "Réunion", "ita": "Riunione", "jpn": "レユニオン", "nld": "Réunion", "por": "Reunião", "rus": "Реюньон", "spa": "Reunión", "svk": "Réunion", "fin": "Réunion", "zho": "留尼旺岛", "isr": "ראוניון"}}, "BL": {"currency": "EUR", "callingCode": "590", "flag": "", "name": {"common": "Saint Barthélemy", "deu": "Saint-Barthélemy", "fra": "Saint-Barthélemy", "hrv": "Saint Barthélemy", "ita": "Antille Francesi", "jpn": "サン・バルテルミー", "nld": "Saint Barthélemy", "por": "São Bartolomeu", "rus": "Сен-Бартелеми", "spa": "San Bartolomé", "svk": "Svätý Bartolomej", "fin": "Saint-Barthélemy", "zho": "圣巴泰勒米", "isr": "סנט ברתולומיאו"}}, "KN": {"currency": "XCD", "callingCode": "1869", "flag": "", "name": {"common": "Saint Kitts and Nevis", "deu": "Saint Christopher und Nevis", "fra": "Saint-Christophe-et-Niévès", "hrv": "Sveti Kristof i Nevis", "ita": "Saint Kitts e Nevis", "jpn": "セントクリストファー・ネイビス", "nld": "Saint Kitts en Nevis", "por": "São Cristóvão e Nevis", "rus": "Сент-Китс и Невис", "spa": "San Cristóbal y Nieves", "svk": "Svätý Krištof a Nevis", "fin": "Saint Kitts ja Nevis", "zho": "圣基茨和尼维斯", "isr": "סנט קיטס ונוויס"}}, "LC": {"currency": "XCD", "callingCode": "1758", "flag": "", "name": {"common": "Saint Lucia", "deu": "Saint Lucia", "fra": "Sainte-Lucie", "hrv": "Sveta Lucija", "ita": "Santa Lucia", "jpn": "セントルシア", "nld": "Saint Lucia", "por": "Santa Lúcia", "rus": "Сент-Люсия", "spa": "Santa Lucía", "svk": "Svätá Lucia", "fin": "Saint Lucia", "zho": "圣卢西亚", "isr": "סנט לוסיה"}}, "MF": {"currency": "EUR", "callingCode": "590", "flag": "", "name": {"common": "Saint Martin", "deu": "Saint Martin", "fra": "Saint-Martin", "hrv": "Sveti Martin", "ita": "Saint Martin", "jpn": "サン・マルタン(フランス領)", "nld": "Saint-Martin", "por": "São Martinho", "rus": "Сен-Мартен", "spa": "Saint Martin", "svk": "Svätý Martin", "fin": "Saint-Martin", "zho": "圣马丁", "isr": "סן מרטן"}}, "PM": {"currency": "EUR", "callingCode": "508", "flag": "", "name": {"common": "Saint Pierre and Miquelon", "deu": "Saint-Pierre und Miquelon", "fra": "Saint-Pierre-et-Miquelon", "hrv": "Sveti Petar i Mikelon", "ita": "Saint-Pierre e Miquelon", "jpn": "サンピエール島・ミクロン島", "nld": "Saint Pierre en Miquelon", "por": "Saint-Pierre e Miquelon", "rus": "Сен-Пьер и Микелон", "spa": "San Pedro y Miquelón", "svk": "Saint Pierre a Miquelon", "fin": "Saint-Pierre ja Miquelon", "zho": "圣皮埃尔和密克隆", "isr": "סנט פייר ומיקלון"}}, "VC": {"currency": "XCD", "callingCode": "1784", "flag": "", "name": {"common": "Saint Vincent and the Grenadines", "deu": "Saint Vincent und die Grenadinen", "fra": "Saint-Vincent-et-les-Grenadines", "hrv": "Sveti Vincent i Grenadini", "ita": "Saint Vincent e Grenadine", "jpn": "セントビンセントおよびグレナディーン諸島", "nld": "Saint Vincent en de Grenadines", "por": "São Vincente e Granadinas", "rus": "Сент-Винсент и Гренадины", "spa": "San Vicente y Granadinas", "svk": "Svätý Vincent a Grenadíny", "fin": "Saint Vincent ja Grenadiinit", "zho": "圣文森特和格林纳丁斯", "isr": "סנט וינסנט והגרנדינים"}}, "WS": {"currency": "WST", "callingCode": "685", "flag": "", "name": {"common": "Samoa", "deu": "Samoa", "fra": "Samoa", "hrv": "Samoa", "ita": "Samoa", "jpn": "サモア", "nld": "Samoa", "por": "Samoa", "rus": "Самоа", "spa": "Samoa", "fin": "Samoa", "zho": "萨摩亚", "isr": "סמואה"}}, "SM": {"currency": "EUR", "callingCode": "378", "flag": "", "name": {"common": "San Marino", "deu": "San Marino", "fra": "Saint-Marin", "hrv": "San Marino", "ita": "San Marino", "jpn": "サンマリノ", "nld": "San Marino", "por": "San Marino", "rus": "Сан-Марино", "spa": "San Marino", "svk": "San Maríno", "fin": "San Marino", "zho": "圣马力诺", "isr": "סן מרינו"}}, "SA": {"currency": "SAR", "callingCode": "966", "flag": "", "name": {"common": "Saudi Arabia", "deu": "Saudi-Arabien", "fra": "Arabie Saoudite", "hrv": "Saudijska Arabija", "ita": "Arabia Saudita", "jpn": "サウジアラビア", "nld": "Saoedi-Arabië", "por": "Arábia Saudita", "rus": "Саудовская Аравия", "spa": "Arabia Saudí", "svk": "Saudská Arábia", "fin": "Saudi-Arabia", "zho": "沙特阿拉伯", "isr": "ערב הסעודית"}}, "SN": {"currency": "XOF", "callingCode": "221", "flag": "", "name": {"common": "Senegal", "deu": "Senegal", "fra": "Sénégal", "hrv": "Senegal", "ita": "Senegal", "jpn": "セネガル", "nld": "Senegal", "por": "Senegal", "rus": "Сенегал", "spa": "Senegal", "svk": "Senegal", "fin": "Senegal", "zho": "塞内加尔", "isr": "סנגל"}}, "RS": {"currency": "RSD", "callingCode": "381", "flag": "", "name": {"common": "Serbia", "deu": "Serbien", "fra": "Serbie", "hrv": "Srbija", "ita": "Serbia", "jpn": "セルビア", "nld": "Servië", "por": "Sérvia", "rus": "Сербия", "spa": "Serbia", "svk": "Srbsko", "fin": "Serbia", "zho": "塞尔维亚", "isr": "סרביה"}}, "SC": {"currency": "SCR", "callingCode": "248", "flag": "", "name": {"common": "Seychelles", "deu": "Seychellen", "fra": "Seychelles", "hrv": "Sejšeli", "ita": "Seychelles", "jpn": "セーシェル", "nld": "Seychellen", "por": "Seicheles", "rus": "Сейшельские Острова", "spa": "Seychelles", "svk": "Seychely", "fin": "Seychellit", "zho": "塞舌尔", "isr": "איי סיישל"}}, "SL": {"currency": "SLL", "callingCode": "232", "flag": "", "name": {"common": "Sierra Leone", "deu": "Sierra Leone", "fra": "Sierra Leone", "hrv": "Sijera Leone", "ita": "Sierra Leone", "jpn": "シエラレオネ", "nld": "Sierra Leone", "por": "Serra Leoa", "rus": "Сьерра-Леоне", "spa": "Sierra Leone", "svk": "Sierra Leone", "fin": "Sierra Leone", "zho": "塞拉利昂", "isr": "סיירה לאונה"}}, "SG": {"currency": "SGD", "callingCode": "65", "flag": "", "name": {"common": "Singapore", "deu": "Singapur", "fra": "Singapour", "hrv": "Singapur", "ita": "Singapore", "jpn": "シンガポール", "nld": "Singapore", "por": "Singapura", "rus": "Сингапур", "spa": "Singapur", "svk": "Singapur", "fin": "Singapore", "isr": "סינגפור"}}, "SX": {"currency": "ANG", "callingCode": "1721", "flag": "", "name": {"common": "Sint Maarten", "deu": "Sint Maarten", "fra": "Saint-Martin", "ita": "Sint Maarten", "jpn": "シント・マールテン", "nld": "Sint Maarten", "por": "São Martinho", "rus": "Синт-Мартен", "spa": "Sint Maarten", "svk": "Svätý Martin", "fin": "Sint Maarten", "zho": "圣马丁岛", "isr": "סנט מארטן"}}, "SK": {"currency": "EUR", "callingCode": "421", "flag": "", "name": {"common": "Slovakia", "deu": "Slowakei", "fra": "Slovaquie", "hrv": "Slovačka", "ita": "Slovacchia", "jpn": "スロバキア", "nld": "Slowakije", "por": "Eslováquia", "rus": "Словакия", "spa": "República Eslovaca", "svk": "Slovensko", "fin": "Slovakia", "zho": "斯洛伐克", "isr": "סלובקיה"}}, "SI": {"currency": "EUR", "callingCode": "386", "flag": "", "name": {"common": "Slovenia", "deu": "Slowenien", "fra": "Slovénie", "hrv": "Slovenija", "ita": "Slovenia", "jpn": "スロベニア", "nld": "Slovenië", "por": "Eslovénia", "rus": "Словения", "spa": "Eslovenia", "svk": "Slovinsko", "fin": "Slovenia", "zho": "斯洛文尼亚", "isr": "סלובניה"}}, "SB": {"currency": "SBD", "callingCode": "677", "flag": "", "name": {"common": "Solomon Islands", "deu": "Salomonen", "fra": "Îles Salomon", "hrv": "Solomonski Otoci", "ita": "Isole Salomone", "jpn": "ソロモン諸島", "nld": "Salomonseilanden", "por": "Ilhas Salomão", "rus": "Соломоновы Острова", "spa": "Islas Salomón", "svk": "Salomonove ostrovy", "fin": "Salomonsaaret", "zho": "所罗门群岛", "isr": "איי שלמה"}}, "SO": {"currency": "SOS", "callingCode": "252", "flag": "", "name": {"common": "Somalia", "deu": "Somalia", "fra": "Somalie", "hrv": "Somalija", "ita": "Somalia", "jpn": "ソマリア", "nld": "Somalië", "por": "Somália", "rus": "Сомали", "spa": "Somalia", "svk": "Somálsko", "fin": "Somalia", "zho": "索马里", "isr": "סומליה"}}, "ZA": {"currency": "ZAR", "callingCode": "27", "flag": "", "name": {"common": "South Africa", "deu": "Republik Südafrika", "fra": "Afrique du Sud", "hrv": "Južnoafrička Republika", "ita": "Sud Africa", "jpn": "南アフリカ", "nld": "Zuid-Afrika", "por": "África do Sul", "rus": "Южно-Африканская Республика", "spa": "República de Sudáfrica", "svk": "Juhoafrická republika", "fin": "Etelä-Afrikka", "zho": "南非", "isr": "דרום אפריקה"}}, "GS": {"currency": "GBP", "callingCode": "500", "flag": "", "name": {"common": "South Georgia", "deu": "Südgeorgien und die Südlichen Sandwichinseln", "fra": "Géorgie du Sud-et-les Îles Sandwich du Sud", "hrv": "Južna Georgija i otočje Južni Sandwich", "ita": "Georgia del Sud e Isole Sandwich Meridionali", "jpn": "サウスジョージア・サウスサンドウィッチ諸島", "nld": "Zuid-Georgia en Zuidelijke Sandwicheilanden", "por": "Ilhas Geórgia do Sul e Sandwich do Sul", "rus": "Южная Георгия и Южные Сандвичевы острова", "spa": "Islas Georgias del Sur y Sandwich del Sur", "svk": "Južná Georgia a Južné Sandwichove ostrovy", "fin": "Etelä-Georgia ja Eteläiset Sandwichsaaret", "zho": "南乔治亚", "isr": "ג׳ורג׳יה הדרומית ואיי סנדוויץ׳ הדרומיים"}}, "KR": {"currency": "KRW", "callingCode": "82", "flag": "", "name": {"common": "South Korea", "deu": "Südkorea", "fra": "Corée du Sud", "hrv": "Južna Koreja", "ita": "Corea del Sud", "jpn": "大韓民国", "nld": "Zuid-Korea", "por": "Coreia do Sul", "rus": "Южная Корея", "spa": "Corea del Sur", "svk": "Južná Kórea", "fin": "Etelä-Korea", "zho": "韩国", "isr": "קוריאה הדרומית"}}, "SS": {"currency": "SSP", "callingCode": "211", "flag": "", "name": {"common": "South Sudan", "deu": "Südsudan", "fra": "Soudan du Sud", "hrv": "Južni Sudan", "ita": "Sudan del sud", "jpn": "南スーダン", "nld": "Zuid-Soedan", "por": "Sudão do Sul", "rus": "Южный Судан", "spa": "Sudán del Sur", "svk": "Južný Sudán", "fin": "Etelä-Sudan", "zho": "南苏丹", "isr": "דרום סודן"}}, "ES": {"currency": "EUR", "callingCode": "34", "flag": "", "name": {"common": "Spain", "deu": "Spanien", "fra": "Espagne", "hrv": "Španjolska", "ita": "Spagna", "jpn": "スペイン", "nld": "Spanje", "por": "Espanha", "rus": "Испания", "spa": "España", "svk": "Španielsko", "fin": "Espanja", "zho": "西班牙", "isr": "ספרד"}}, "LK": {"currency": "LKR", "callingCode": "94", "flag": "", "name": {"common": "Sri Lanka", "deu": "Sri Lanka", "fra": "Sri Lanka", "hrv": "Šri Lanka", "ita": "Sri Lanka", "jpn": "スリランカ", "nld": "Sri Lanka", "por": "Sri Lanka", "rus": "Шри-Ланка", "spa": "Sri Lanka", "svk": "Srí Lanka", "fin": "Sri Lanka", "zho": "斯里兰卡", "isr": "סרי לנקה"}}, "SD": {"currency": "SDG", "callingCode": "249", "flag": "", "name": {"common": "Sudan", "deu": "Sudan", "fra": "Soudan", "hrv": "Sudan", "ita": "Sudan", "jpn": "スーダン", "nld": "Soedan", "por": "Sudão", "rus": "Судан", "spa": "Sudán", "svk": "Sudán", "fin": "Sudan", "zho": "苏丹", "isr": "סודן"}}, "SR": {"currency": "SRD", "callingCode": "597", "flag": "", "name": {"common": "Suriname", "deu": "Suriname", "fra": "Surinam", "hrv": "Surinam", "ita": "Suriname", "jpn": "スリナム", "nld": "Suriname", "por": "Suriname", "rus": "Суринам", "spa": "Surinam", "svk": "Surinam", "fin": "Suriname", "zho": "苏里南", "isr": "סורינם"}}, "SJ": {"currency": "NOK", "callingCode": "4779", "flag": "", "name": {"common": "Svalbard and Jan Mayen", "deu": "Spitzbergen", "fra": "Svalbard et Jan Mayen", "hrv": "Svalbard i Jan Mayen", "ita": "Svalbard e Jan Mayen", "jpn": "スヴァールバル諸島およびヤンマイエン島", "nld": "Svalbard en Jan Mayen", "por": "Ilhas Svalbard e Jan Mayen", "rus": "Шпицберген и Ян-Майен", "spa": "Islas Svalbard y Jan Mayen", "svk": "Svalbard a Jan Mayen", "fin": "Huippuvuoret", "zho": "斯瓦尔巴特", "isr": "סוולבארד ויאן מאיין"}}, "SZ": {"currency": "SZL", "callingCode": "268", "flag": "", "name": {"common": "Swaziland", "deu": "Swasiland", "fra": "Swaziland", "hrv": "Svazi", "ita": "Swaziland", "jpn": "スワジランド", "nld": "Swaziland", "por": "Suazilândia", "rus": "Свазиленд", "spa": "Suazilandia", "svk": "Svazijsko", "fin": "Swazimaa", "zho": "斯威士兰", "isr": "סווזילנד"}}, "SE": {"currency": "SEK", "callingCode": "46", "flag": "", "name": {"common": "Sweden", "deu": "Schweden", "fra": "Suède", "hrv": "Švedska", "ita": "Svezia", "jpn": "スウェーデン", "nld": "Zweden", "por": "Suécia", "rus": "Швеция", "spa": "Suecia", "svk": "šveédsko", "fin": "Ruotsi", "zho": "瑞典", "isr": "שוודיה"}}, "CH": {"currency": "CHE", "callingCode": "41", "flag": "", "name": {"common": "Switzerland", "deu": "Schweiz", "fra": "Suisse", "hrv": "Švicarska", "ita": "Svizzera", "jpn": "スイス", "nld": "Zwitserland", "por": "Suíça", "rus": "Швейцария", "spa": "Suiza", "svk": "Švajčiarsko", "fin": "Sveitsi", "zho": "瑞士", "isr": "שווייץ"}}, "SY": {"currency": "SYP", "callingCode": "963", "flag": "", "name": {"common": "Syria", "deu": "Syrien", "fra": "Syrie", "hrv": "Sirija", "ita": "Siria", "jpn": "シリア・アラブ共和国", "nld": "Syrië", "por": "Síria", "rus": "Сирия", "spa": "Siria", "svk": "Sýria", "fin": "Syyria", "zho": "叙利亚", "isr": "סוריה"}}, "ST": {"currency": "STD", "callingCode": "239", "flag": "", "name": {"common": "São Tomé and Príncipe", "deu": "São Tomé und Príncipe", "fra": "São Tomé et Príncipe", "hrv": "Sveti Toma i Princip", "ita": "São Tomé e Príncipe", "jpn": "サントメ・プリンシペ", "nld": "Sao Tomé en Principe", "por": "São Tomé e Príncipe", "rus": "Сан-Томе и Принсипи", "spa": "Santo Tomé y Príncipe", "svk": "Svätý Tomáš a Princov ostrov", "fin": "São Téme ja Príncipe", "zho": "圣多美和普林西比", "isr": "סאו טומה ופרינסיפה"}}, "TW": {"currency": "TWD", "callingCode": "886", "flag": "", "name": {"common": "Taiwan", "deu": "Taiwan", "fra": "Taïwan", "hrv": "Tajvan", "ita": "Taiwan", "jpn": "台湾(台湾省/中華民国)", "nld": "Taiwan", "por": "Ilha Formosa", "rus": "Тайвань", "spa": "Taiwán", "svk": "Taiwan", "fin": "Taiwan", "isr": "טייוואן"}}, "TJ": {"currency": "TJS", "callingCode": "992", "flag": "", "name": {"common": "Tajikistan", "deu": "Tadschikistan", "fra": "Tadjikistan", "hrv": "Tađikistan", "ita": "Tagikistan", "jpn": "タジキスタン", "nld": "Tadzjikistan", "por": "Tajiquistão", "rus": "Таджикистан", "spa": "Tayikistán", "svk": "Tadžikistan", "fin": "Tadžikistan", "zho": "塔吉克斯坦", "isr": "טג׳יקיסטן"}}, "TZ": {"currency": "TZS", "callingCode": "255", "flag": "", "name": {"common": "Tanzania", "deu": "Tansania", "fra": "Tanzanie", "hrv": "Tanzanija", "ita": "Tanzania", "jpn": "タンザニア", "nld": "Tanzania", "por": "Tanzânia", "rus": "Танзания", "spa": "Tanzania", "svk": "Tanzánia", "fin": "Tansania", "zho": "坦桑尼亚", "isr": "טנזניה"}}, "TH": {"currency": "THB", "callingCode": "66", "flag": "", "name": {"common": "Thailand", "deu": "Thailand", "fra": "Thaïlande", "hrv": "Tajland", "ita": "Tailandia", "jpn": "タイ", "nld": "Thailand", "por": "Tailândia", "rus": "Таиланд", "spa": "Tailandia", "svk": "Thajsko", "fin": "Thaimaa", "zho": "泰国", "isr": "תאילנד"}}, "TL": {"currency": "USD", "callingCode": "670", "flag": "", "name": {"common": "Timor-Leste", "deu": "Timor-Leste", "fra": "Timor oriental", "hrv": "Istočni Timor", "ita": "Timor Est", "jpn": "東ティモール", "nld": "Oost-Timor", "por": "Timor-Leste", "rus": "Восточный Тимор", "spa": "Timor Oriental", "svk": "Východný Timor", "fin": "Itä-Timor", "zho": "东帝汶", "isr": "טימור לסטה"}}, "TG": {"currency": "XOF", "callingCode": "228", "flag": "", "name": {"common": "Togo", "deu": "Togo", "fra": "Togo", "hrv": "Togo", "ita": "Togo", "jpn": "トーゴ", "nld": "Togo", "por": "Togo", "rus": "Того", "spa": "Togo", "svk": "Togo", "fin": "Togo", "zho": "多哥", "isr": "טוגו"}}, "TK": {"currency": "NZD", "callingCode": "690", "flag": "", "name": {"common": "Tokelau", "deu": "Tokelau", "fra": "Tokelau", "hrv": "Tokelau", "ita": "Isole Tokelau", "jpn": "トケラウ", "nld": "Tokelau", "por": "Tokelau", "rus": "Токелау", "spa": "Islas Tokelau", "svk": "Tokelau", "fin": "Tokelau", "zho": "托克劳", "isr": "טוקלאו"}}, "TO": {"currency": "TOP", "callingCode": "676", "flag": "", "name": {"common": "Tonga", "deu": "Tonga", "fra": "Tonga", "hrv": "Tonga", "ita": "Tonga", "jpn": "トンガ", "nld": "Tonga", "por": "Tonga", "rus": "Тонга", "spa": "Tonga", "svk": "Tonga", "fin": "Tonga", "zho": "汤加", "isr": "טונגה"}}, "TT": {"currency": "TTD", "callingCode": "1868", "flag": "", "name": {"common": "Trinidad and Tobago", "deu": "Trinidad und Tobago", "fra": "Trinité-et-Tobago", "hrv": "Trinidad i Tobago", "ita": "Trinidad e Tobago", "jpn": "トリニダード・トバゴ", "nld": "Trinidad en Tobago", "por": "Trinidade e Tobago", "rus": "Тринидад и Тобаго", "spa": "Trinidad y Tobago", "svk": "Trinidad a Tobago", "fin": "Trinidad ja Tobago", "zho": "特立尼达和多巴哥", "isr": "טרינידד וטובגו"}}, "TN": {"currency": "TND", "callingCode": "216", "flag": "", "name": {"common": "Tunisia", "deu": "Tunesien", "fra": "Tunisie", "hrv": "Tunis", "ita": "Tunisia", "jpn": "チュニジア", "nld": "Tunesië", "por": "Tunísia", "rus": "Тунис", "spa": "Túnez", "svk": "Tunisko", "fin": "Tunisia", "zho": "突尼斯", "isr": "טוניסיה"}}, "TR": {"currency": "TRY", "callingCode": "90", "flag": "", "name": {"common": "Turkey", "deu": "Türkei", "fra": "Turquie", "hrv": "Turska", "ita": "Turchia", "jpn": "トルコ", "nld": "Turkije", "por": "Turquia", "rus": "Турция", "spa": "Turquía", "svk": "Turecko", "fin": "Turkki", "zho": "土耳其", "isr": "טורקיה"}}, "TM": {"currency": "TMT", "callingCode": "993", "flag": "", "name": {"common": "Turkmenistan", "deu": "Turkmenistan", "fra": "Turkménistan", "hrv": "Turkmenistan", "ita": "Turkmenistan", "jpn": "トルクメニスタン", "nld": "Turkmenistan", "por": "Turquemenistão", "rus": "Туркмения", "spa": "Turkmenistán", "svk": "Turkménsko", "fin": "Turkmenistan", "zho": "土库曼斯坦", "isr": "טורקמניסטן"}}, "TC": {"currency": "USD", "callingCode": "1649", "flag": "", "name": {"common": "Turks and Caicos Islands", "deu": "Turks-und Caicosinseln", "fra": "Îles Turques-et-Caïques", "hrv": "Otoci Turks i Caicos", "ita": "Isole Turks e Caicos", "jpn": "タークス・カイコス諸島", "nld": "Turks-en Caicoseilanden", "por": "Ilhas Turks e Caicos", "rus": "Теркс и Кайкос", "spa": "Islas Turks y Caicos", "svk": "Turks a Caicos", "fin": "Turks-ja Caicossaaret", "zho": "特克斯和凯科斯群岛", "isr": "איי טורקס וקאיקוס"}}, "TV": {"currency": "AUD", "callingCode": "688", "flag": "", "name": {"common": "Tuvalu", "deu": "Tuvalu", "fra": "Tuvalu", "hrv": "Tuvalu", "ita": "Tuvalu", "jpn": "ツバル", "nld": "Tuvalu", "por": "Tuvalu", "rus": "Тувалу", "spa": "Tuvalu", "svk": "Tuvalu", "fin": "Tuvalu", "zho": "图瓦卢", "isr": "טובאלו"}}, "UG": {"currency": "UGX", "callingCode": "256", "flag": "", "name": {"common": "Uganda", "deu": "Uganda", "fra": "Ouganda", "hrv": "Uganda", "ita": "Uganda", "jpn": "ウガンダ", "nld": "Oeganda", "por": "Uganda", "rus": "Уганда", "spa": "Uganda", "svk": "Uganda", "fin": "Uganda", "zho": "乌干达", "isr": "אוגנדה"}}, "UA": {"currency": "UAH", "callingCode": "380", "flag": "", "name": {"common": "Ukraine", "deu": "Ukraine", "fra": "Ukraine", "hrv": "Ukrajina", "ita": "Ucraina", "jpn": "ウクライナ", "nld": "Oekraïne", "por": "Ucrânia", "rus": "Украина", "spa": "Ucrania", "svk": "Ukrajina", "fin": "Ukraina", "zho": "乌克兰", "isr": "אוקראינה"}}, "AE": {"currency": "AED", "callingCode": "971", "flag": "", "name": {"common": "United Arab Emirates", "deu": "Vereinigte Arabische Emirate", "fra": "Émirats arabes unis", "hrv": "Ujedinjeni Arapski Emirati", "ita": "Emirati Arabi Uniti", "jpn": "アラブ首長国連邦", "nld": "Verenigde Arabische Emiraten", "por": "Emirados Árabes Unidos", "rus": "Объединённые Арабские Эмираты", "spa": "Emiratos Árabes Unidos", "svk": "Spojené arabské emiráty", "fin": "Arabiemiraatit", "zho": "阿拉伯联合酋长国", "isr": "איחוד האמירויות הערביות"}}, "GB": {"currency": "GBP", "callingCode": "44", "flag": "", "name": {"common": "United Kingdom", "deu": "Vereinigtes Königreich", "fra": "Royaume-Uni", "hrv": "Ujedinjeno Kraljevstvo", "ita": "Regno Unito", "jpn": "イギリス", "nld": "Verenigd Koninkrijk", "por": "Reino Unido", "rus": "Великобритания", "spa": "Reino Unido", "svk": "Veľká Británia (Spojené kráľovstvo)", "fin": "Yhdistynyt kuningaskunta", "zho": "英国", "isr": "הממלכה המאוחדת"}}, "US": {"currency": "USD", "callingCode": "1", "flag": "", "name": {"common": "United States", "deu": "Vereinigte Staaten von Amerika", "fra": "États-Unis", "hrv": "Sjedinjene Američke Države", "ita": "Stati Uniti d'America", "jpn": "アメリカ合衆国", "nld": "Verenigde Staten", "por": "Estados Unidos", "rus": "Соединённые Штаты Америки", "spa": "Estados Unidos", "svk": "Spojené štáty", "fin": "Yhdysvallat", "zho": "美国", "isr": "ארצות הברית"}}, "UM": {"currency": "USD", "flag": "", "name": {"common": "United States Minor Outlying Islands", "deu": "Kleinere Inselbesitzungen der Vereinigten Staaten", "fra": "Îles mineures éloignées des États-Unis", "hrv": "Mali udaljeni otoci SAD-a", "ita": "Isole minori esterne degli Stati Uniti d'America", "jpn": "合衆国領有小離島", "nld": "Kleine afgelegen eilanden van de Verenigde Staten", "por": "Ilhas Menores Distantes dos Estados Unidos", "rus": "Внешние малые острова США", "spa": "Islas Ultramarinas Menores de Estados Unidos", "svk": "Menšie odľahlé ostrovy USA", "fin": "Yhdysvaltain asumattomat saaret", "zho": "美国本土外小岛屿", "isr": "האיים המרוחקים הקטנים של ארה״ב"}}, "VI": {"currency": "USD", "callingCode": "1340", "flag": "", "name": {"common": "United States Virgin Islands", "deu": "Amerikanische Jungferninseln", "fra": "Îles Vierges des États-Unis", "hrv": "Američki Djevičanski Otoci", "ita": "Isole Vergini americane", "jpn": "アメリカ領ヴァージン諸島", "nld": "Amerikaanse Maagdeneilanden", "por": "Ilhas Virgens dos Estados Unidos", "rus": "Виргинские Острова", "spa": "Islas Vírgenes de los Estados Unidos", "svk": "Americké Panenské ostrovy", "fin": "Neitsytsaaret", "zho": "美属维尔京群岛", "isr": "איי הבתולה של ארצות הברית"}}, "UY": {"currency": "UYI", "callingCode": "598", "flag": "", "name": {"common": "Uruguay", "deu": "Uruguay", "fra": "Uruguay", "hrv": "Urugvaj", "ita": "Uruguay", "jpn": "ウルグアイ", "nld": "Uruguay", "por": "Uruguai", "rus": "Уругвай", "spa": "Uruguay", "svk": "Uruguaj", "fin": "Uruguay", "zho": "乌拉圭", "isr": "אורוגוואי"}}, "UZ": {"currency": "UZS", "callingCode": "998", "flag": "", "name": {"common": "Uzbekistan", "deu": "Usbekistan", "fra": "Ouzbékistan", "hrv": "Uzbekistan", "ita": "Uzbekistan", "jpn": "ウズベキスタン", "nld": "Oezbekistan", "por": "Uzbequistão", "rus": "Узбекистан", "spa": "Uzbekistán", "svk": "Uzbekistan", "fin": "Uzbekistan", "zho": "乌兹别克斯坦", "isr": "אוזבקיסטן"}}, "VU": {"currency": "VUV", "callingCode": "678", "flag": "", "name": {"common": "Vanuatu", "deu": "Vanuatu", "fra": "Vanuatu", "hrv": "Vanuatu", "ita": "Vanuatu", "jpn": "バヌアツ", "nld": "Vanuatu", "por": "Vanuatu", "rus": "Вануату", "spa": "Vanuatu", "svk": "Vanuatu", "fin": "Vanuatu", "zho": "瓦努阿图", "isr": "ונואטו"}}, "VA": {"currency": "EUR", "callingCode": "3906698", "flag": "", "name": {"common": "Vatican City", "deu": "Vatikanstadt", "fra": "Cité du Vatican", "hrv": "Vatikan", "ita": "Città del Vaticano", "jpn": "バチカン市国", "nld": "Vaticaanstad", "por": "Cidade do Vaticano", "rus": "Ватикан", "spa": "Ciudad del Vaticano", "svk": "Vatikán", "fin": "Vatikaani", "zho": "梵蒂冈", "isr": "הוותיקן"}}, "VE": {"currency": "VEF", "callingCode": "58", "flag": "", "name": {"common": "Venezuela", "deu": "Venezuela", "fra": "Venezuela", "hrv": "Venezuela", "ita": "Venezuela", "jpn": "ベネズエラ・ボリバル共和国", "nld": "Venezuela", "por": "Venezuela", "rus": "Венесуэла", "spa": "Venezuela", "svk": "Venezuela", "fin": "Venezuela", "zho": "委内瑞拉", "isr": "ונצואלה"}}, "VN": {"currency": "VND", "callingCode": "84", "flag": "", "name": {"common": "Vietnam", "deu": "Vietnam", "fra": "Viêt Nam", "hrv": "Vijetnam", "ita": "Vietnam", "jpn": "ベトナム", "nld": "Vietnam", "por": "Vietname", "rus": "Вьетнам", "spa": "Vietnam", "svk": "Vietnam", "fin": "Vietnam", "zho": "越南", "isr": "וייטנאם"}}, "WF": {"currency": "XPF", "callingCode": "681", "flag": "", "name": {"common": "Wallis and Futuna", "deu": "Wallis und Futuna", "fra": "Wallis-et-Futuna", "hrv": "Wallis i Fortuna", "ita": "Wallis e Futuna", "jpn": "ウォリス・フツナ", "nld": "Wallis en Futuna", "por": "Wallis e Futuna", "rus": "Уоллис и Футуна", "spa": "Wallis y Futuna", "svk": "Wallis a Futuna", "fin": "Wallis ja Futuna", "zho": "瓦利斯和富图纳群岛", "isr": "איי ווליס ופוטונה"}}, "EH": {"currency": "MAD", "callingCode": "212", "flag": "", "name": {"common": "Western Sahara", "deu": "Westsahara", "fra": "Sahara Occidental", "hrv": "Zapadna Sahara", "ita": "Sahara Occidentale", "jpn": "西サハラ", "nld": "Westelijke Sahara", "por": "Saara Ocidental", "rus": "Западная Сахара", "spa": "Sahara Occidental", "svk": "Západná Sahara", "fin": "Länsi-Sahara", "zho": "西撒哈拉", "isr": "סהרה המערבית"}}, "YE": {"currency": "YER", "callingCode": "967", "flag": "", "name": {"common": "Yemen", "deu": "Jemen", "fra": "Yémen", "hrv": "Jemen", "ita": "Yemen", "jpn": "イエメン", "nld": "Jemen", "por": "Iémen", "rus": "Йемен", "spa": "Yemen", "svk": "Jemen", "fin": "Jemen", "zho": "也门", "isr": "תימן"}}, "ZM": {"currency": "ZMW", "callingCode": "260", "flag": "", "name": {"common": "Zambia", "deu": "Sambia", "fra": "Zambie", "hrv": "Zambija", "ita": "Zambia", "jpn": "ザンビア", "nld": "Zambia", "por": "Zâmbia", "rus": "Замбия", "spa": "Zambia", "svk": "Zambia", "fin": "Sambia", "zho": "赞比亚", "isr": "זמביה"}}, "ZW": {"currency": "ZWL", "callingCode": "263", "flag": "", "name": {"common": "Zimbabwe", "deu": "Simbabwe", "fra": "Zimbabwe", "hrv": "Zimbabve", "ita": "Zimbabwe", "jpn": "ジンバブエ", "nld": "Zimbabwe", "por": "Zimbabwe", "rus": "Зимбабве", "spa": "Zimbabue", "svk": "Zimbabwe", "fin": "Zimbabwe", "zho": "津巴布韦", "isr": "זימבבואה"}}, "AX": {"currency": "EUR", "callingCode": "358", "flag": "", "name": {"common": "Åland Islands", "deu": "Åland", "fra": "Ahvenanmaa", "hrv": "Ålandski otoci", "ita": "Isole Aland", "jpn": "オーランド諸島", "nld": "Ålandeilanden", "por": "Alândia", "rus": "Аландские острова", "spa": "Alandia", "svk": "Alandy", "fin": "Ahvenanmaa", "zho": "奥兰群岛", "isr": "איי אולנד"}}} \ No newline at end of file diff --git a/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_data_countriesemoji.json b/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_data_countriesemoji.json deleted file mode 100644 index ec7aadc6..00000000 --- a/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_data_countriesemoji.json +++ /dev/null @@ -1 +0,0 @@ -{"AF": {"currency": "AFN", "callingCode": "93", "flag": "flag-af", "name": {"common": "Afghanistan", "cym": "Affganistan", "deu": "Afghanistan", "fra": "Afghanistan", "hrv": "Afganistan", "ita": "Afghanistan", "jpn": "アフガニスタン", "nld": "Afghanistan", "por": "Afeganistão", "rus": "Афганистан", "spa": "Afganistán", "svk": "Afganistan", "fin": "Afganistan", "zho": "阿富汗", "isr": "אפגניסטן"}}, "AL": {"currency": "ALL", "callingCode": "355", "flag": "flag-al", "name": {"common": "Albania", "cym": "Albania", "deu": "Albanien", "fra": "Albanie", "hrv": "Albanija", "ita": "Albania", "jpn": "アルバニア", "nld": "Albanië", "por": "Albânia", "rus": "Албания", "spa": "Albania", "svk": "Albánsko", "fin": "Albania", "zho": "阿尔巴尼亚", "isr": "אלבניה"}}, "DZ": {"currency": "DZD", "callingCode": "213", "flag": "flag-dz", "name": {"common": "Algeria", "cym": "Algeria", "deu": "Algerien", "fra": "Algérie", "hrv": "Alžir", "ita": "Algeria", "jpn": "アルジェリア", "nld": "Algerije", "por": "Argélia", "rus": "Алжир", "spa": "Argelia", "svk": "Alžírsko", "fin": "Algeria", "zho": "阿尔及利亚", "isr": "אלג׳יריה"}}, "AS": {"currency": "USD", "callingCode": "1684", "flag": "flag-as", "name": {"common": "American Samoa", "deu": "Amerikanisch-Samoa", "fra": "Samoa américaines", "hrv": "Američka Samoa", "ita": "Samoa Americane", "jpn": "アメリカ領サモア", "nld": "Amerikaans Samoa", "por": "Samoa Americana", "rus": "Американское Самоа", "spa": "Samoa Americana", "svk": "Americká Samoa", "fin": "Amerikan Samoa", "zho": "美属萨摩亚", "isr": "סמואה האמריקנית"}}, "AD": {"currency": "EUR", "callingCode": "376", "flag": "flag-ad", "name": {"common": "Andorra", "cym": "Andorra", "deu": "Andorra", "fra": "Andorre", "hrv": "Andora", "ita": "Andorra", "jpn": "アンドラ", "nld": "Andorra", "por": "Andorra", "rus": "Андорра", "spa": "Andorra", "svk": "Andorra", "fin": "Andorra", "zho": "安道尔", "isr": "אנדורה"}}, "AO": {"currency": "AOA", "callingCode": "244", "flag": "flag-ao", "name": {"common": "Angola", "cym": "Angola", "deu": "Angola", "fra": "Angola", "hrv": "Angola", "ita": "Angola", "jpn": "アンゴラ", "nld": "Angola", "por": "Angola", "rus": "Ангола", "spa": "Angola", "svk": "Angola", "fin": "Angola", "zho": "安哥拉", "isr": "אנגולה"}}, "AI": {"currency": "XCD", "callingCode": "1264", "flag": "flag-ai", "name": {"common": "Anguilla", "deu": "Anguilla", "fra": "Anguilla", "hrv": "Angvila", "ita": "Anguilla", "jpn": "アンギラ", "nld": "Anguilla", "por": "Anguilla", "rus": "Ангилья", "spa": "Anguilla", "svk": "Anguilla", "fin": "Anguilla", "zho": "安圭拉", "isr": "אנגילה"}}, "AQ": {"flag": "flag-aq", "callingCode": "672", "name": {"common": "Antarctica", "cym": "Antarctica", "deu": "Antarktis", "fra": "Antarctique", "hrv": "Antarktika", "ita": "Antartide", "jpn": "南極", "nld": "Antarctica", "por": "Antártida", "rus": "Антарктида", "spa": "Antártida", "svk": "Antarktída", "fin": "Etelämanner", "zho": "南极洲", "isr": "אנטארקטיקה"}}, "AG": {"currency": "XCD", "callingCode": "1268", "flag": "flag-ag", "name": {"common": "Antigua and Barbuda", "cym": "Antigwa a Barbiwda", "deu": "Antigua und Barbuda", "fra": "Antigua-et-Barbuda", "hrv": "Antigva i Barbuda", "ita": "Antigua e Barbuda", "jpn": "アンティグア・バーブーダ", "nld": "Antigua en Barbuda", "por": "Antígua e Barbuda", "rus": "Антигуа и Барбуда", "spa": "Antigua y Barbuda", "svk": "Antigua a Barbuda", "fin": "Antigua ja Barbuda", "zho": "安提瓜和巴布达", "isr": "אנטיגואה וברבודה"}}, "AR": {"currency": "ARS", "callingCode": "54", "flag": "flag-ar", "name": {"common": "Argentina", "cym": "Ariannin", "deu": "Argentinien", "fra": "Argentine", "hrv": "Argentina", "ita": "Argentina", "jpn": "アルゼンチン", "nld": "Argentinië", "por": "Argentina", "rus": "Аргентина", "spa": "Argentina", "svk": "Argentína", "fin": "Argentiina", "zho": "阿根廷", "isr": "ארגנטינה"}}, "AM": {"currency": "AMD", "callingCode": "374", "flag": "flag-am", "name": {"common": "Armenia", "cym": "Armenia", "deu": "Armenien", "fra": "Arménie", "hrv": "Armenija", "ita": "Armenia", "jpn": "アルメニア", "nld": "Armenië", "por": "Arménia", "rus": "Армения", "spa": "Armenia", "svk": "Arménsko", "fin": "Armenia", "zho": "亚美尼亚", "isr": "ארמניה"}}, "AW": {"currency": "AWG", "callingCode": "297", "flag": "flag-aw", "name": {"common": "Aruba", "deu": "Aruba", "fra": "Aruba", "hrv": "Aruba", "ita": "Aruba", "jpn": "アルバ", "nld": "Aruba", "por": "Aruba", "rus": "Аруба", "spa": "Aruba", "svk": "Aruba", "fin": "Aruba", "zho": "阿鲁巴", "isr": "ארובה"}}, "AU": {"currency": "AUD", "callingCode": "61", "flag": "flag-au", "name": {"common": "Australia", "cym": "Awstralia", "deu": "Australien", "fra": "Australie", "hrv": "Australija", "ita": "Australia", "jpn": "オーストラリア", "nld": "Australië", "por": "Austrália", "rus": "Австралия", "spa": "Australia", "svk": "Austrália", "fin": "Australia", "zho": "澳大利亚", "isr": "אוסטרליה"}}, "AT": {"currency": "EUR", "callingCode": "43", "flag": "flag-at", "name": {"common": "Austria", "cym": "Awstria", "deu": "Österreich", "fra": "Autriche", "hrv": "Austrija", "ita": "Austria", "jpn": "オーストリア", "nld": "Oostenrijk", "por": "Áustria", "rus": "Австрия", "spa": "Austria", "svk": "Rakúsko", "fin": "Itävalta", "zho": "奥地利", "isr": "אוסטריה"}}, "AZ": {"currency": "AZN", "callingCode": "994", "flag": "flag-az", "name": {"common": "Azerbaijan", "cym": "Aserbaijan", "deu": "Aserbaidschan", "fra": "Azerbaïdjan", "hrv": "Azerbajdžan", "ita": "Azerbaijan", "jpn": "アゼルバイジャン", "nld": "Azerbeidzjan", "por": "Azerbeijão", "rus": "Азербайджан", "spa": "Azerbaiyán", "svk": "Azerbajdžan", "fin": "Azerbaidzan", "zho": "阿塞拜疆", "isr": "אזרבייג׳ן"}}, "BS": {"currency": "BSD", "callingCode": "1242", "flag": "flag-bs", "name": {"common": "Bahamas", "cym": "Bahamas", "deu": "Bahamas", "fra": "Bahamas", "hrv": "Bahami", "ita": "Bahamas", "jpn": "バハマ", "nld": "Bahama’s", "por": "Bahamas", "rus": "Багамские Острова", "spa": "Bahamas", "svk": "Bahamy", "fin": "Bahamasaaret", "zho": "巴哈马", "isr": "איי בהאמה"}}, "BH": {"currency": "BHD", "callingCode": "973", "flag": "flag-bh", "name": {"common": "Bahrain", "cym": "Bahrain", "deu": "Bahrain", "fra": "Bahreïn", "hrv": "Bahrein", "ita": "Bahrein", "jpn": "バーレーン", "nld": "Bahrein", "por": "Bahrein", "rus": "Бахрейн", "spa": "Bahrein", "svk": "Bahrajn", "fin": "Bahrain", "zho": "巴林", "isr": "בחריין"}}, "BD": {"currency": "BDT", "callingCode": "880", "flag": "flag-bd", "name": {"common": "Bangladesh", "cym": "Bangladesh", "deu": "Bangladesch", "fra": "Bangladesh", "hrv": "Bangladeš", "ita": "Bangladesh", "jpn": "バングラデシュ", "nld": "Bangladesh", "por": "Bangladesh", "rus": "Бангладеш", "spa": "Bangladesh", "svk": "Bangladéš", "fin": "Bangladesh", "zho": "孟加拉国", "isr": "בנגלדש"}}, "BB": {"currency": "BBD", "callingCode": "1246", "flag": "flag-bb", "name": {"common": "Barbados", "cym": "Barbados", "deu": "Barbados", "fra": "Barbade", "hrv": "Barbados", "ita": "Barbados", "jpn": "バルバドス", "nld": "Barbados", "por": "Barbados", "rus": "Барбадос", "spa": "Barbados", "svk": "Barbados", "fin": "Barbados", "zho": "巴巴多斯", "isr": "ברבדוס"}}, "BY": {"currency": "BYR", "callingCode": "375", "flag": "flag-by", "name": {"common": "Belarus", "cym": "Belarws", "deu": "Weißrussland", "fra": "Biélorussie", "hrv": "Bjelorusija", "ita": "Bielorussia", "jpn": "ベラルーシ", "nld": "Wit-Rusland", "por": "Bielorússia", "rus": "Белоруссия", "spa": "Bielorrusia", "svk": "Bielorusko", "fin": "Valko-Venäjä", "zho": "白俄罗斯", "isr": "בלארוס"}}, "BE": {"currency": "EUR", "callingCode": "32", "flag": "flag-be", "name": {"common": "Belgium", "cym": "Gwlad Belg", "deu": "Belgien", "fra": "Belgique", "hrv": "Belgija", "ita": "Belgio", "jpn": "ベルギー", "nld": "België", "por": "Bélgica", "rus": "Бельгия", "spa": "Bélgica", "svk": "Belgicko", "fin": "Belgia", "zho": "比利时", "isr": "בלגיה"}}, "BZ": {"currency": "BZD", "callingCode": "501", "flag": "flag-bz", "name": {"common": "Belize", "cym": "Belize", "deu": "Belize", "fra": "Belize", "hrv": "Belize", "ita": "Belize", "jpn": "ベリーズ", "nld": "Belize", "por": "Belize", "rus": "Белиз", "spa": "Belice", "svk": "Belize", "fin": "Belize", "zho": "伯利兹", "isr": "בליז"}}, "BJ": {"currency": "XOF", "callingCode": "229", "flag": "flag-bj", "name": {"common": "Benin", "cym": "Benin", "deu": "Benin", "fra": "Bénin", "hrv": "Benin", "ita": "Benin", "jpn": "ベナン", "nld": "Benin", "por": "Benin", "rus": "Бенин", "spa": "Benín", "svk": "Benin", "fin": "Benin", "zho": "贝宁", "isr": "בנין"}}, "BM": {"currency": "BMD", "callingCode": "1441", "flag": "flag-bm", "name": {"common": "Bermuda", "cym": "Bermiwda", "deu": "Bermuda", "fra": "Bermudes", "hrv": "Bermudi", "ita": "Bermuda", "jpn": "バミューダ", "nld": "Bermuda", "por": "Bermudas", "rus": "Бермудские Острова", "spa": "Bermudas", "svk": "Bermudy", "fin": "Bermuda", "zho": "百慕大", "isr": "ברמודה"}}, "BT": {"currency": "BTN", "callingCode": "975", "flag": "flag-bt", "name": {"common": "Bhutan", "cym": "Bhwtan", "deu": "Bhutan", "fra": "Bhoutan", "hrv": "Butan", "ita": "Bhutan", "jpn": "ブータン", "nld": "Bhutan", "por": "Butão", "rus": "Бутан", "spa": "Bután", "svk": "Bhután", "fin": "Bhutan", "zho": "不丹", "isr": "בהוטן"}}, "BO": {"currency": "BOB", "callingCode": "591", "flag": "flag-bo", "name": {"common": "Bolivia", "cym": "Bolifia", "deu": "Bolivien", "fra": "Bolivie", "hrv": "Bolivija", "ita": "Bolivia", "jpn": "ボリビア多民族国", "nld": "Bolivia", "por": "Bolívia", "rus": "Боливия", "spa": "Bolivia", "svk": "Bolívija", "fin": "Bolivia", "zho": "玻利维亚", "isr": "בוליביה"}}, "BA": {"currency": "BAM", "callingCode": "387", "flag": "flag-ba", "name": {"common": "Bosnia and Herzegovina", "cym": "Bosnia a Hercegovina", "deu": "Bosnien und Herzegowina", "fra": "Bosnie-Herzégovine", "hrv": "Bosna i Hercegovina", "ita": "Bosnia ed Erzegovina", "jpn": "ボスニア・ヘルツェゴビナ", "nld": "Bosnië en Herzegovina", "por": "Bósnia e Herzegovina", "rus": "Босния и Герцеговина", "spa": "Bosnia y Herzegovina", "svk": "Bosna a Hercegovina", "fin": "Bosnia ja Hertsegovina", "zho": "波斯尼亚和黑塞哥维那", "isr": "בוסניה והרצגובינה"}}, "BW": {"currency": "BWP", "callingCode": "267", "flag": "flag-bw", "name": {"common": "Botswana", "deu": "Botswana", "fra": "Botswana", "hrv": "Bocvana", "ita": "Botswana", "jpn": "ボツワナ", "nld": "Botswana", "por": "Botswana", "rus": "Ботсвана", "spa": "Botswana", "svk": "Botswana", "fin": "Botswana", "zho": "博茨瓦纳", "isr": "בוצוואנה"}}, "BV": {"currency": "NOK", "flag": "flag-bv", "name": {"common": "Bouvet Island", "deu": "Bouvetinsel", "fra": "Île Bouvet", "hrv": "Otok Bouvet", "ita": "Isola Bouvet", "jpn": "ブーベ島", "nld": "Bouveteiland", "por": "Ilha Bouvet", "rus": "Остров Буве", "spa": "Isla Bouvet", "svk": "Bouvetov ostrov", "fin": "Bouvet'nsaari", "zho": "布维岛", "isr": "איי בובה"}}, "BR": {"currency": "BRL", "callingCode": "55", "flag": "flag-br", "name": {"common": "Brazil", "cym": "Brasil", "deu": "Brasilien", "fra": "Brésil", "hrv": "Brazil", "ita": "Brasile", "jpn": "ブラジル", "nld": "Brazilië", "por": "Brasil", "rus": "Бразилия", "spa": "Brasil", "svk": "Brazília", "fin": "Brasilia", "zho": "巴西", "isr": "ברזיל"}}, "IO": {"currency": "USD", "callingCode": "246", "flag": "flag-io", "name": {"common": "British Indian Ocean Territory", "cym": "Tiriogaeth Brydeinig Cefnfor India", "deu": "Britisches Territorium im Indischen Ozean", "fra": "Territoire britannique de l'océan Indien", "hrv": "Britanski Indijskooceanski teritorij", "ita": "Territorio britannico dell'oceano indiano", "jpn": "イギリス領インド洋地域", "nld": "Britse Gebieden in de Indische Oceaan", "por": "Território Britânico do Oceano Índico", "rus": "Британская территория в Индийском океане", "spa": "Territorio Británico del Océano Índico", "svk": "Britské indickooceánske územie", "fin": "Brittiläinen Intian valtameren alue", "zho": "英属印度洋领地", "isr": "הטריטוריה הבריטית באוקיינוס ההודי"}}, "VG": {"currency": "USD", "callingCode": "1284", "flag": "flag-vg", "name": {"common": "British Virgin Islands", "deu": "Britische Jungferninseln", "fra": "Îles Vierges britanniques", "hrv": "Britanski Djevičanski Otoci", "ita": "Isole Vergini Britanniche", "jpn": "イギリス領ヴァージン諸島", "nld": "Britse Maagdeneilanden", "por": "Ilhas Virgens", "rus": "Британские Виргинские острова", "spa": "Islas Vírgenes del Reino Unido", "svk": "Panenské ostrovy", "fin": "Neitsytsaaret", "zho": "英属维尔京群岛", "isr": "איי הבתולה הבריטיים"}}, "BN": {"currency": "BND", "callingCode": "673", "flag": "flag-bn", "name": {"common": "Brunei", "cym": "Brunei", "deu": "Brunei", "fra": "Brunei", "hrv": "Brunej", "ita": "Brunei", "jpn": "ブルネイ・ダルサラーム", "nld": "Brunei", "por": "Brunei", "rus": "Бруней", "spa": "Brunei", "svk": "Brunej", "fin": "Brunei", "zho": "文莱", "isr": "ברוניי"}}, "BG": {"currency": "BGN", "callingCode": "359", "flag": "flag-bg", "name": {"common": "Bulgaria", "cym": "Bwlgaria", "deu": "Bulgarien", "fra": "Bulgarie", "hrv": "Bugarska", "ita": "Bulgaria", "jpn": "ブルガリア", "nld": "Bulgarije", "por": "Bulgária", "rus": "Болгария", "spa": "Bulgaria", "svk": "Bulharsko", "fin": "Bulgaria", "zho": "保加利亚", "isr": "בולגריה"}}, "BF": {"currency": "XOF", "callingCode": "226", "flag": "flag-bf", "name": {"common": "Burkina Faso", "cym": "Burkina Faso", "deu": "Burkina Faso", "fra": "Burkina Faso", "hrv": "Burkina Faso", "ita": "Burkina Faso", "jpn": "ブルキナファソ", "nld": "Burkina Faso", "por": "Burkina Faso", "rus": "Буркина-Фасо", "spa": "Burkina Faso", "svk": "Burkina Faso", "fin": "Burkina Faso", "zho": "布基纳法索", "isr": "בורקינה פאסו"}}, "BI": {"currency": "BIF", "callingCode": "257", "flag": "flag-bi", "name": {"common": "Burundi", "cym": "Bwrwndi", "deu": "Burundi", "fra": "Burundi", "hrv": "Burundi", "ita": "Burundi", "jpn": "ブルンジ", "nld": "Burundi", "por": "Burundi", "rus": "Бурунди", "spa": "Burundi", "svk": "Burundi", "fin": "Burundi", "zho": "布隆迪", "isr": "בורונדי"}}, "KH": {"currency": "KHR", "callingCode": "855", "flag": "flag-kh", "name": {"common": "Cambodia", "cym": "Cambodia", "deu": "Kambodscha", "fra": "Cambodge", "hrv": "Kambodža", "ita": "Cambogia", "jpn": "カンボジア", "nld": "Cambodja", "por": "Camboja", "rus": "Камбоджа", "spa": "Camboya", "svk": "Kambodža", "fin": "Kambodža", "zho": "柬埔寨", "isr": "קמבודיה"}}, "CM": {"currency": "XAF", "callingCode": "237", "flag": "flag-cm", "name": {"common": "Cameroon", "cym": "Camerŵn", "deu": "Kamerun", "fra": "Cameroun", "hrv": "Kamerun", "ita": "Camerun", "jpn": "カメルーン", "nld": "Kameroen", "por": "Camarões", "rus": "Камерун", "spa": "Camerún", "svk": "Kamerun", "fin": "Kamerun", "zho": "喀麦隆", "isr": "קמרון"}}, "CA": {"currency": "CAD", "callingCode": "1", "flag": "flag-ca", "name": {"common": "Canada", "cym": "Canada", "deu": "Kanada", "fra": "Canada", "hrv": "Kanada", "ita": "Canada", "jpn": "カナダ", "nld": "Canada", "por": "Canadá", "rus": "Канада", "spa": "Canadá", "svk": "Kanada", "fin": "Kanada", "zho": "加拿大", "isr": "קנדה"}}, "CV": {"currency": "CVE", "callingCode": "238", "flag": "flag-cv", "name": {"common": "Cape Verde", "cym": "Cape Verde", "deu": "Kap Verde", "fra": "Îles du Cap-Vert", "hrv": "Zelenortska Republika", "ita": "Capo Verde", "jpn": "カーボベルデ", "nld": "Kaapverdië", "por": "Cabo Verde", "rus": "Кабо-Верде", "spa": "Cabo Verde", "svk": "Kapverdy", "fin": "Kap Verde", "zho": "佛得角", "isr": "כף ורדה"}}, "KY": {"currency": "KYD", "callingCode": "1345", "flag": "flag-ky", "name": {"common": "Cayman Islands", "cym": "Ynysoedd_Cayman", "deu": "Kaimaninseln", "fra": "Îles Caïmans", "hrv": "Kajmanski otoci", "ita": "Isole Cayman", "jpn": "ケイマン諸島", "nld": "Caymaneilanden", "por": "Ilhas Caimão", "rus": "Каймановы острова", "spa": "Islas Caimán", "svk": "Kajmanie ostrovy", "fin": "Caymansaaret", "zho": "开曼群岛", "isr": "איי קיימן"}}, "CF": {"currency": "XAF", "callingCode": "236", "flag": "flag-cf", "name": {"common": "Central African Republic", "cym": "Gweriniaeth Canolbarth Affrica", "deu": "Zentralafrikanische Republik", "fra": "République centrafricaine", "hrv": "Srednjoafrička Republika", "ita": "Repubblica Centrafricana", "jpn": "中央アフリカ共和国", "nld": "Centraal-Afrikaanse Republiek", "por": "República Centro-Africana", "rus": "Центральноафриканская Республика", "spa": "República Centroafricana", "svk": "Stredoafrická republika", "fin": "Keski-Afrikan tasavalta", "zho": "中非共和国", "isr": "הרפובליקה של מרכז אפריקה"}}, "TD": {"currency": "XAF", "callingCode": "235", "flag": "flag-td", "name": {"common": "Chad", "cym": "Tsiad", "deu": "Tschad", "fra": "Tchad", "hrv": "Čad", "ita": "Ciad", "jpn": "チャド", "nld": "Tsjaad", "por": "Chade", "rus": "Чад", "spa": "Chad", "svk": "Čad", "fin": "Tšad", "zho": "乍得", "isr": "צ׳אד"}}, "CL": {"currency": "CLF", "callingCode": "56", "flag": "flag-cl", "name": {"common": "Chile", "cym": "Chile", "deu": "Chile", "fra": "Chili", "hrv": "Čile", "ita": "Cile", "jpn": "チリ", "nld": "Chili", "por": "Chile", "rus": "Чили", "spa": "Chile", "svk": "Čile", "fin": "Chile", "zho": "智利", "isr": "צ׳ילה"}}, "CN": {"currency": "CNY", "callingCode": "86", "flag": "flag-cn", "name": {"common": "China", "cym": "Tsieina", "deu": "China", "fra": "Chine", "hrv": "Kina", "ita": "Cina", "jpn": "中国", "nld": "China", "por": "China", "rus": "Китай", "spa": "China", "svk": "Čína", "fin": "Kiina", "isr": "סין"}}, "CX": {"currency": "AUD", "callingCode": "61", "flag": "flag-cx", "name": {"common": "Christmas Island", "cym": "Ynys y Nadolig", "deu": "Weihnachtsinsel", "fra": "Île Christmas", "hrv": "Božićni otok", "ita": "Isola di Natale", "jpn": "クリスマス島", "nld": "Christmaseiland", "por": "Ilha do Natal", "rus": "Остров Рождества", "spa": "Isla de Navidad", "svk": "Vianočnú ostrov", "fin": "Joulusaari", "zho": "圣诞岛", "isr": "האי כריסטמס"}}, "CC": {"currency": "AUD", "callingCode": "61", "flag": "flag-cc", "name": {"common": "Cocos (Keeling) Islands", "cym": "Ynysoedd Cocos", "deu": "Kokosinseln", "fra": "Îles Cocos", "hrv": "Kokosovi Otoci", "ita": "Isole Cocos e Keeling", "jpn": "ココス(キーリング)諸島", "nld": "Cocoseilanden", "por": "Ilhas Cocos (Keeling)", "rus": "Кокосовые острова", "spa": "Islas Cocos o Islas Keeling", "svk": "Kokosové ostrovy", "fin": "Kookossaaret", "zho": "科科斯", "isr": "איי קוקוס (קילינג)"}}, "CO": {"currency": "COP", "callingCode": "57", "flag": "flag-co", "name": {"common": "Colombia", "cym": "Colombia", "deu": "Kolumbien", "fra": "Colombie", "hrv": "Kolumbija", "ita": "Colombia", "jpn": "コロンビア", "nld": "Colombia", "por": "Colômbia", "rus": "Колумбия", "spa": "Colombia", "svk": "Kolumbia", "fin": "Kolumbia", "zho": "哥伦比亚", "isr": "קולומביה"}}, "KM": {"currency": "KMF", "callingCode": "269", "flag": "flag-km", "name": {"common": "Comoros", "cym": "Comoros", "deu": "Union der Komoren", "fra": "Comores", "hrv": "Komori", "ita": "Comore", "jpn": "コモロ", "nld": "Comoren", "por": "Comores", "rus": "Коморы", "spa": "Comoras", "svk": "Komory", "fin": "Komorit", "zho": "科摩罗", "isr": "קומורו"}}, "CK": {"currency": "NZD", "callingCode": "682", "flag": "flag-ck", "name": {"common": "Cook Islands", "cym": "Ynysoedd Cook", "deu": "Cookinseln", "fra": "Îles Cook", "hrv": "Cookovo Otočje", "ita": "Isole Cook", "jpn": "クック諸島", "nld": "Cookeilanden", "por": "Ilhas Cook", "rus": "Острова Кука", "spa": "Islas Cook", "svk": "Cookove ostrovy", "fin": "Cookinsaaret", "zho": "库克群岛", "isr": "איי קוק"}}, "CR": {"currency": "CRC", "callingCode": "506", "flag": "flag-cr", "name": {"common": "Costa Rica", "cym": "Costa Rica", "deu": "Costa Rica", "fra": "Costa Rica", "hrv": "Kostarika", "ita": "Costa Rica", "jpn": "コスタリカ", "nld": "Costa Rica", "por": "Costa Rica", "rus": "Коста-Рика", "spa": "Costa Rica", "svk": "Kostarika", "fin": "Costa Rica", "zho": "哥斯达黎加", "isr": "קוסטה ריקה"}}, "HR": {"currency": "HRK", "callingCode": "385", "flag": "flag-hr", "name": {"common": "Croatia", "cym": "Croatia", "deu": "Kroatien", "fra": "Croatie", "hrv": "Hrvatska", "ita": "Croazia", "jpn": "クロアチア", "nld": "Kroatië", "por": "Croácia", "rus": "Хорватия", "spa": "Croacia", "svk": "Chorvátsko", "fin": "Kroatia", "zho": "克罗地亚", "isr": "קרואטיה"}}, "CU": {"currency": "CUC", "callingCode": "53", "flag": "flag-cu", "name": {"common": "Cuba", "cym": "Ciwba", "deu": "Kuba", "fra": "Cuba", "hrv": "Kuba", "ita": "Cuba", "jpn": "キューバ", "nld": "Cuba", "por": "Cuba", "rus": "Куба", "spa": "Cuba", "svk": "Kuba", "fin": "Kuuba", "zho": "古巴", "isr": "קובה"}}, "CW": {"currency": "ANG", "callingCode": "5999", "flag": "flag-cw", "name": {"common": "Curaçao", "deu": "Curaçao", "fra": "Curaçao", "nld": "Curaçao", "por": "ilha da Curação", "rus": "Кюрасао", "spa": "Curazao", "svk": "CuraÇao", "fin": "Curaçao", "zho": "库拉索", "isr": "קוראסאו"}}, "CY": {"currency": "EUR", "callingCode": "357", "flag": "flag-cy", "name": {"common": "Cyprus", "cym": "Cyprus", "deu": "Zypern", "fra": "Chypre", "hrv": "Cipar", "ita": "Cipro", "jpn": "キプロス", "nld": "Cyprus", "por": "Chipre", "rus": "Кипр", "spa": "Chipre", "svk": "Cyprus", "fin": "Kypros", "zho": "塞浦路斯", "isr": "קפריסין"}}, "CZ": {"currency": "CZK", "callingCode": "420", "flag": "flag-cz", "name": {"common": "Czech Republic", "cym": "Y Weriniaeth Tsiec", "deu": "Tschechische Republik", "fra": "République tchèque", "hrv": "Češka", "ita": "Repubblica Ceca", "jpn": "チェコ", "nld": "Tsjechië", "por": "República Checa", "rus": "Чехия", "spa": "República Checa", "svk": "Česko", "fin": "Tšekki", "zho": "捷克", "isr": "צ׳כיה"}}, "CD": {"currency": "CDF", "callingCode": "243", "flag": "flag-cd", "name": {"common": "DR Congo", "cym": "Gweriniaeth Ddemocrataidd Congo", "deu": "Kongo (Dem. Rep.)", "fra": "Congo (Rép. dém.)", "hrv": "Kongo, Demokratska Republika", "ita": "Congo (Rep. Dem.)", "jpn": "コンゴ民主共和国", "nld": "Congo (DRC)", "por": "República Democrática do Congo", "rus": "Демократическая Республика Конго", "spa": "Congo (Rep. Dem.)", "svk": "Kongo", "fin": "Kongon demokraattinen tasavalta", "zho": "民主刚果", "isr": "קונגו - קינשאסה"}}, "DK": {"currency": "DKK", "callingCode": "45", "flag": "flag-dk", "name": {"common": "Denmark", "cym": "Denmarc", "deu": "Dänemark", "fra": "Danemark", "hrv": "Danska", "ita": "Danimarca", "jpn": "デンマーク", "nld": "Denemarken", "por": "Dinamarca", "rus": "Дания", "spa": "Dinamarca", "svk": "Dánsko", "fin": "Tanska", "zho": "丹麦", "isr": "דנמרק"}}, "DJ": {"currency": "DJF", "callingCode": "253", "flag": "flag-dj", "name": {"common": "Djibouti", "cym": "Djibouti", "deu": "Dschibuti", "fra": "Djibouti", "hrv": "Džibuti", "ita": "Gibuti", "jpn": "ジブチ", "nld": "Djibouti", "por": "Djibouti", "rus": "Джибути", "spa": "Djibouti", "svk": "Džibutsko", "fin": "Dijibouti", "zho": "吉布提", "isr": "ג׳יבוטי"}}, "DM": {"currency": "XCD", "callingCode": "1767", "flag": "flag-dm", "name": {"common": "Dominica", "cym": "Dominica", "deu": "Dominica", "fra": "Dominique", "hrv": "Dominika", "ita": "Dominica", "jpn": "ドミニカ国", "nld": "Dominica", "por": "Dominica", "rus": "Доминика", "spa": "Dominica", "svk": "Dominika", "fin": "Dominica", "zho": "多米尼加", "isr": "דומיניקה"}}, "DO": {"currency": "DOP", "callingCode": "1809", "flag": "flag-do", "name": {"common": "Dominican Republic", "cym": "Gweriniaeth_Dominica", "deu": "Dominikanische Republik", "fra": "République dominicaine", "hrv": "Dominikanska Republika", "ita": "Repubblica Dominicana", "jpn": "ドミニカ共和国", "nld": "Dominicaanse Republiek", "por": "República Dominicana", "rus": "Доминиканская Республика", "spa": "República Dominicana", "svk": "Dominikánska republika", "fin": "Dominikaaninen tasavalta", "zho": "多明尼加", "isr": "הרפובליקה הדומיניקנית"}}, "EC": {"currency": "USD", "callingCode": "593", "flag": "flag-ec", "name": {"common": "Ecuador", "cym": "Ecwador", "deu": "Ecuador", "fra": "Équateur", "hrv": "Ekvador", "ita": "Ecuador", "jpn": "エクアドル", "nld": "Ecuador", "por": "Equador", "rus": "Эквадор", "spa": "Ecuador", "svk": "Ekvádor", "fin": "Ecuador", "zho": "厄瓜多尔", "isr": "אקוודור"}}, "EG": {"currency": "EGP", "callingCode": "20", "flag": "flag-eg", "name": {"common": "Egypt", "cym": "Yr Aifft", "deu": "Ägypten", "fra": "Égypte", "hrv": "Egipat", "ita": "Egitto", "jpn": "エジプト", "nld": "Egypte", "por": "Egito", "rus": "Египет", "spa": "Egipto", "svk": "Egypt", "fin": "Egypti", "zho": "埃及", "isr": "מצרים"}}, "SV": {"currency": "SVC", "callingCode": "503", "flag": "flag-sv", "name": {"common": "El Salvador", "cym": "El Salvador", "deu": "El Salvador", "fra": "Salvador", "hrv": "Salvador", "ita": "El Salvador", "jpn": "エルサルバドル", "nld": "El Salvador", "por": "El Salvador", "rus": "Сальвадор", "spa": "El Salvador", "svk": "Salvádor", "fin": "El Salvador", "zho": "萨尔瓦多", "isr": "אל סלבדור"}}, "GQ": {"currency": "XAF", "callingCode": "240", "flag": "flag-gq", "name": {"common": "Equatorial Guinea", "cym": "Gini Gyhydeddol", "deu": "Äquatorialguinea", "fra": "Guinée équatoriale", "hrv": "Ekvatorijalna Gvineja", "ita": "Guinea Equatoriale", "jpn": "赤道ギニア", "nld": "Equatoriaal-Guinea", "por": "Guiné Equatorial", "rus": "Экваториальная Гвинея", "spa": "Guinea Ecuatorial", "svk": "Rovníková Guinea", "fin": "Päiväntasaajan Guinea", "zho": "赤道几内亚", "isr": "גינאה המשוונית"}}, "ER": {"currency": "ERN", "callingCode": "291", "flag": "flag-er", "name": {"common": "Eritrea", "cym": "Eritrea", "deu": "Eritrea", "fra": "Érythrée", "hrv": "Eritreja", "ita": "Eritrea", "jpn": "エリトリア", "nld": "Eritrea", "por": "Eritreia", "rus": "Эритрея", "spa": "Eritrea", "svk": "Eritrea", "fin": "Eritrea", "zho": "厄立特里亚", "isr": "אריתריאה"}}, "EE": {"currency": "EUR", "callingCode": "372", "flag": "flag-ee", "name": {"common": "Estonia", "cym": "Estonia", "deu": "Estland", "fra": "Estonie", "hrv": "Estonija", "ita": "Estonia", "jpn": "エストニア", "nld": "Estland", "por": "Estónia", "rus": "Эстония", "spa": "Estonia", "svk": "Estónsko", "fin": "Viro", "zho": "爱沙尼亚", "isr": "אסטוניה"}}, "ET": {"currency": "ETB", "callingCode": "251", "flag": "flag-et", "name": {"common": "Ethiopia", "cym": "Ethiopia", "deu": "Äthiopien", "fra": "Éthiopie", "hrv": "Etiopija", "ita": "Etiopia", "jpn": "エチオピア", "nld": "Ethiopië", "por": "Etiópia", "rus": "Эфиопия", "spa": "Etiopía", "svk": "Etiópia", "fin": "Etiopia", "zho": "埃塞俄比亚", "isr": "אתיופיה"}}, "FK": {"currency": "FKP", "callingCode": "500", "flag": "flag-fk", "name": {"common": "Falkland Islands", "deu": "Falklandinseln", "fra": "Îles Malouines", "hrv": "Falklandski Otoci", "ita": "Isole Falkland o Isole Malvine", "jpn": "フォークランド(マルビナス)諸島", "nld": "Falklandeilanden", "por": "Ilhas Malvinas", "rus": "Фолклендские острова", "spa": "Islas Malvinas", "svk": "Falklandy", "fin": "Falkandinsaaret", "zho": "福克兰群岛", "isr": "איי פוקלנד"}}, "FO": {"currency": "DKK", "callingCode": "298", "flag": "flag-fo", "name": {"common": "Faroe Islands", "deu": "Färöer-Inseln", "fra": "Îles Féroé", "hrv": "Farski Otoci", "ita": "Isole Far Oer", "jpn": "フェロー諸島", "nld": "Faeröer", "por": "Ilhas Faroé", "rus": "Фарерские острова", "spa": "Islas Faroe", "svk": "Faerské ostrovy", "fin": "Färsaaret", "zho": "法罗群岛", "isr": "איי פארו"}}, "FJ": {"currency": "FJD", "callingCode": "679", "flag": "flag-fj", "name": {"common": "Fiji", "deu": "Fidschi", "fra": "Fidji", "hrv": "Fiđi", "ita": "Figi", "jpn": "フィジー", "nld": "Fiji", "por": "Fiji", "rus": "Фиджи", "spa": "Fiyi", "svk": "Fidži", "fin": "Fidži", "zho": "斐济", "isr": "פיג׳י"}}, "FI": {"currency": "EUR", "callingCode": "358", "flag": "flag-fi", "name": {"common": "Finland", "deu": "Finnland", "fra": "Finlande", "hrv": "Finska", "ita": "Finlandia", "jpn": "フィンランド", "nld": "Finland", "por": "Finlândia", "rus": "Финляндия", "spa": "Finlandia", "svk": "Fínsko", "fin": "Suomi", "zho": "芬兰", "isr": "פינלנד"}}, "FR": {"currency": "EUR", "callingCode": "33", "flag": "flag-fr", "name": {"common": "France", "deu": "Frankreich", "fra": "France", "hrv": "Francuska", "ita": "Francia", "jpn": "フランス", "nld": "Frankrijk", "por": "França", "rus": "Франция", "spa": "Francia", "svk": "Francúzsko", "fin": "Ranska", "zho": "法国", "isr": "צרפת"}}, "GF": {"currency": "EUR", "callingCode": "594", "flag": "flag-gf", "name": {"common": "French Guiana", "deu": "Französisch Guyana", "fra": "Guyane", "hrv": "Francuska Gvajana", "ita": "Guyana francese", "jpn": "フランス領ギアナ", "nld": "Frans-Guyana", "por": "Guiana Francesa", "rus": "Французская Гвиана", "spa": "Guayana Francesa", "svk": "Guyana", "fin": "Ranskan Guayana", "zho": "法属圭亚那", "isr": "גיאנה הצרפתית"}}, "PF": {"currency": "XPF", "callingCode": "689", "flag": "flag-pf", "name": {"common": "French Polynesia", "deu": "Französisch-Polynesien", "fra": "Polynésie française", "hrv": "Francuska Polinezija", "ita": "Polinesia Francese", "jpn": "フランス領ポリネシア", "nld": "Frans-Polynesië", "por": "Polinésia Francesa", "rus": "Французская Полинезия", "spa": "Polinesia Francesa", "svk": "Francúzska Polynézia", "fin": "Ranskan Polynesia", "zho": "法属波利尼西亚", "isr": "פולינזיה הצרפתית"}}, "TF": {"currency": "EUR", "flag": "flag-tf", "name": {"common": "French Southern and Antarctic Lands", "deu": "Französische Süd-und Antarktisgebiete", "fra": "Terres australes et antarctiques françaises", "hrv": "Francuski južni i antarktički teritoriji", "ita": "Territori Francesi del Sud", "jpn": "フランス領南方・南極地域", "nld": "Franse Gebieden in de zuidelijke Indische Oceaan", "por": "Terras Austrais e Antárticas Francesas", "rus": "Французские Южные и Антарктические территории", "spa": "Tierras Australes y Antárticas Francesas", "svk": "Francúzske juŽné a antarktické územia", "fin": "Ranskan eteläiset ja antarktiset alueet", "zho": "法国南部和南极土地", "isr": "הטריטוריות הדרומיות של צרפת"}}, "GA": {"currency": "XAF", "callingCode": "241", "flag": "flag-ga", "name": {"common": "Gabon", "deu": "Gabun", "fra": "Gabon", "hrv": "Gabon", "ita": "Gabon", "jpn": "ガボン", "nld": "Gabon", "por": "Gabão", "rus": "Габон", "spa": "Gabón", "svk": "Gabon", "fin": "Gabon", "zho": "加蓬", "isr": "גבון"}}, "GM": {"currency": "GMD", "callingCode": "220", "flag": "flag-gm", "name": {"common": "Gambia", "deu": "Gambia", "fra": "Gambie", "hrv": "Gambija", "ita": "Gambia", "jpn": "ガンビア", "nld": "Gambia", "por": "Gâmbia", "rus": "Гамбия", "spa": "Gambia", "svk": "Gambia", "fin": "Gambia", "zho": "冈比亚", "isr": "גמביה"}}, "GE": {"currency": "GEL", "callingCode": "995", "flag": "flag-ge", "name": {"common": "Georgia", "deu": "Georgien", "fra": "Géorgie", "hrv": "Gruzija", "ita": "Georgia", "jpn": "グルジア", "nld": "Georgië", "por": "Geórgia", "rus": "Грузия", "spa": "Georgia", "svk": "Gruzínsko", "fin": "Georgia", "zho": "格鲁吉亚", "isr": "גאורגיה"}}, "DE": {"currency": "EUR", "callingCode": "49", "flag": "flag-de", "name": {"common": "Germany", "deu": "Deutschland", "fra": "Allemagne", "hrv": "Njemačka", "ita": "Germania", "jpn": "ドイツ", "nld": "Duitsland", "por": "Alemanha", "rus": "Германия", "spa": "Alemania", "svk": "Nemecko", "fin": "Saksa", "zho": "德国", "isr": "גרמניה"}}, "GH": {"currency": "GHS", "callingCode": "233", "flag": "flag-gh", "name": {"common": "Ghana", "deu": "Ghana", "fra": "Ghana", "hrv": "Gana", "ita": "Ghana", "jpn": "ガーナ", "nld": "Ghana", "por": "Gana", "rus": "Гана", "spa": "Ghana", "svk": "Ghana", "fin": "Ghana", "zho": "加纳", "isr": "גאנה"}}, "GI": {"currency": "GIP", "callingCode": "350", "flag": "flag-gi", "name": {"common": "Gibraltar", "deu": "Gibraltar", "fra": "Gibraltar", "hrv": "Gibraltar", "ita": "Gibilterra", "jpn": "ジブラルタル", "nld": "Gibraltar", "por": "Gibraltar", "rus": "Гибралтар", "spa": "Gibraltar", "svk": "Gibraltár", "fin": "Gibraltar", "zho": "直布罗陀", "isr": "גיברלטר"}}, "GR": {"currency": "EUR", "callingCode": "30", "flag": "flag-gr", "name": {"common": "Greece", "deu": "Griechenland", "fra": "Grèce", "hrv": "Grčka", "ita": "Grecia", "jpn": "ギリシャ", "nld": "Griekenland", "por": "Grécia", "rus": "Греция", "spa": "Grecia", "svk": "Greécko", "fin": "Kreikka", "zho": "希腊", "isr": "יוון"}}, "GL": {"currency": "DKK", "callingCode": "299", "flag": "flag-gl", "name": {"common": "Greenland", "deu": "Grönland", "fra": "Groenland", "hrv": "Grenland", "ita": "Groenlandia", "jpn": "グリーンランド", "nld": "Groenland", "por": "Gronelândia", "rus": "Гренландия", "spa": "Groenlandia", "svk": "Grónsko", "fin": "Groönlanti", "zho": "格陵兰", "isr": "גרינלנד"}}, "GD": {"currency": "XCD", "callingCode": "1473", "flag": "flag-gd", "name": {"common": "Grenada", "deu": "Grenada", "fra": "Grenade", "hrv": "Grenada", "ita": "Grenada", "jpn": "グレナダ", "nld": "Grenada", "por": "Granada", "rus": "Гренада", "spa": "Grenada", "svk": "Grenada", "fin": "Grenada", "zho": "格林纳达", "isr": "גרנדה"}}, "GP": {"currency": "EUR", "callingCode": "590", "flag": "flag-gp", "name": {"common": "Guadeloupe", "deu": "Guadeloupe", "fra": "Guadeloupe", "hrv": "Gvadalupa", "ita": "Guadeloupa", "jpn": "グアドループ", "nld": "Guadeloupe", "por": "Guadalupe", "rus": "Гваделупа", "spa": "Guadalupe", "svk": "Guadeloupe", "fin": "Guadeloupe", "zho": "瓜德罗普岛", "isr": "גוואדלופ"}}, "GU": {"currency": "USD", "callingCode": "1671", "flag": "flag-gu", "name": {"common": "Guam", "deu": "Guam", "fra": "Guam", "hrv": "Guam", "ita": "Guam", "jpn": "グアム", "nld": "Guam", "por": "Guam", "rus": "Гуам", "spa": "Guam", "svk": "Guam", "fin": "Guam", "zho": "关岛", "isr": "גואם"}}, "GT": {"currency": "GTQ", "callingCode": "502", "flag": "flag-gt", "name": {"common": "Guatemala", "deu": "Guatemala", "fra": "Guatemala", "hrv": "Gvatemala", "ita": "Guatemala", "jpn": "グアテマラ", "nld": "Guatemala", "por": "Guatemala", "rus": "Гватемала", "spa": "Guatemala", "svk": "Guatemala", "fin": "Guatemala", "zho": "危地马拉", "isr": "גואטמלה"}}, "GG": {"currency": "GBP", "callingCode": "44", "flag": "flag-gg", "name": {"common": "Guernsey", "deu": "Guernsey", "fra": "Guernesey", "hrv": "Guernsey", "ita": "Guernsey", "jpn": "ガーンジー", "nld": "Guernsey", "por": "Guernsey", "rus": "Гернси", "spa": "Guernsey", "svk": "Guernsey", "fin": "Guernsey", "zho": "根西岛", "isr": "גרנסי"}}, "GN": {"currency": "GNF", "callingCode": "224", "flag": "flag-gn", "name": {"common": "Guinea", "deu": "Guinea", "fra": "Guinée", "hrv": "Gvineja", "ita": "Guinea", "jpn": "ギニア", "nld": "Guinee", "por": "Guiné", "rus": "Гвинея", "spa": "Guinea", "svk": "Guinea", "fin": "Guinea", "zho": "几内亚", "isr": "גינאה"}}, "GW": {"currency": "XOF", "callingCode": "245", "flag": "flag-gw", "name": {"common": "Guinea-Bissau", "deu": "Guinea-Bissau", "fra": "Guinée-Bissau", "hrv": "Gvineja Bisau", "ita": "Guinea-Bissau", "jpn": "ギニアビサウ", "nld": "Guinee-Bissau", "por": "Guiné-Bissau", "rus": "Гвинея-Бисау", "spa": "Guinea-Bisáu", "svk": "Guinea-Bissau", "fin": "Guinea-Bissau", "zho": "几内亚比绍", "isr": "גינאה ביסאו"}}, "GY": {"currency": "GYD", "callingCode": "592", "flag": "flag-gy", "name": {"common": "Guyana", "deu": "Guyana", "fra": "Guyana", "hrv": "Gvajana", "ita": "Guyana", "jpn": "ガイアナ", "nld": "Guyana", "por": "Guiana", "rus": "Гайана", "spa": "Guyana", "svk": "Guyana", "fin": "Guayana", "zho": "圭亚那", "isr": "גיאנה"}}, "HT": {"currency": "HTG", "callingCode": "509", "flag": "flag-ht", "name": {"common": "Haiti", "deu": "Haiti", "fra": "Haïti", "hrv": "Haiti", "ita": "Haiti", "jpn": "ハイチ", "nld": "Haïti", "por": "Haiti", "rus": "Гаити", "spa": "Haiti", "svk": "Haiti", "fin": "Haiti", "zho": "海地", "isr": "האיטי"}}, "HM": {"currency": "AUD", "flag": "flag-hm", "name": {"common": "Heard Island and McDonald Islands", "deu": "Heard und die McDonaldinseln", "fra": "Îles Heard-et-MacDonald", "hrv": "Otok Heard i otočje McDonald", "ita": "Isole Heard e McDonald", "jpn": "ハード島とマクドナルド諸島", "nld": "Heard-en McDonaldeilanden", "por": "Ilha Heard e Ilhas McDonald", "rus": "Остров Херд и острова Макдональд", "spa": "Islas Heard y McDonald", "svk": "Heardov ostrov", "fin": "Heard ja McDonaldinsaaret", "zho": "赫德岛和麦当劳群岛", "isr": "איי הרד ומקדונלד"}}, "HN": {"currency": "HNL", "callingCode": "504", "flag": "flag-hn", "name": {"common": "Honduras", "deu": "Honduras", "fra": "Honduras", "hrv": "Honduras", "ita": "Honduras", "jpn": "ホンジュラス", "nld": "Honduras", "por": "Honduras", "rus": "Гондурас", "spa": "Honduras", "fin": "Honduras", "zho": "洪都拉斯", "isr": "הונדורס"}}, "HK": {"currency": "HKD", "callingCode": "852", "flag": "flag-hk", "name": {"common": "Hong Kong", "deu": "Hongkong", "fra": "Hong Kong", "hrv": "Hong Kong", "ita": "Hong Kong", "jpn": "香港", "nld": "Hongkong", "por": "Hong Kong", "rus": "Гонконг", "spa": "Hong Kong", "svk": "Hongkong", "fin": "Hongkong", "isr": "הונג קונג (מחוז מנהלי מיוחד של סין)"}}, "HU": {"currency": "HUF", "callingCode": "36", "flag": "flag-hu", "name": {"common": "Hungary", "deu": "Ungarn", "fra": "Hongrie", "hrv": "Mađarska", "ita": "Ungheria", "jpn": "ハンガリー", "nld": "Hongarije", "por": "Hungria", "rus": "Венгрия", "spa": "Hungría", "svk": "Maďarsko", "fin": "Unkari", "zho": "匈牙利", "isr": "הונגריה"}}, "IS": {"currency": "ISK", "callingCode": "354", "flag": "flag-is", "name": {"common": "Iceland", "deu": "Island", "fra": "Islande", "hrv": "Island", "ita": "Islanda", "jpn": "アイスランド", "nld": "IJsland", "por": "Islândia", "rus": "Исландия", "spa": "Islandia", "svk": "Island", "fin": "Islanti", "zho": "冰岛", "isr": "איסלנד"}}, "IN": {"currency": "INR", "callingCode": "91", "flag": "flag-in", "name": {"common": "India", "deu": "Indien", "fra": "Inde", "hrv": "Indija", "ita": "India", "jpn": "インド", "nld": "India", "por": "Índia", "rus": "Индия", "spa": "India", "svk": "India", "fin": "Intia", "zho": "印度", "isr": "הודו"}}, "ID": {"currency": "IDR", "callingCode": "62", "flag": "flag-id", "name": {"common": "Indonesia", "deu": "Indonesien", "fra": "Indonésie", "hrv": "Indonezija", "ita": "Indonesia", "jpn": "インドネシア", "nld": "Indonesië", "por": "Indonésia", "rus": "Индонезия", "spa": "Indonesia", "svk": "Indonézia", "fin": "Indonesia", "zho": "印度尼西亚", "isr": "אינדונזיה"}}, "IR": {"currency": "IRR", "callingCode": "98", "flag": "flag-ir", "name": {"common": "Iran", "deu": "Iran", "fra": "Iran", "hrv": "Iran", "ita": "Iran", "jpn": "イラン・イスラム共和国", "nld": "Iran", "por": "Irão", "rus": "Иран", "spa": "Iran", "svk": "Irán", "fin": "Iran", "zho": "伊朗", "isr": "איראן"}}, "IQ": {"currency": "IQD", "callingCode": "964", "flag": "flag-iq", "name": {"common": "Iraq", "deu": "Irak", "fra": "Irak", "hrv": "Irak", "ita": "Iraq", "jpn": "イラク", "nld": "Irak", "por": "Iraque", "rus": "Ирак", "spa": "Irak", "svk": "Irak", "fin": "Irak", "zho": "伊拉克", "isr": "עיראק"}}, "IE": {"currency": "EUR", "callingCode": "353", "flag": "flag-ie", "name": {"common": "Ireland", "deu": "Irland", "fra": "Irlande", "hrv": "Irska", "ita": "Irlanda", "jpn": "アイルランド", "nld": "Ierland", "por": "Irlanda", "rus": "Ирландия", "spa": "Irlanda", "svk": "Írsko", "fin": "Irlanti", "zho": "爱尔兰", "isr": "אירלנד"}}, "IM": {"currency": "GBP", "callingCode": "44", "flag": "flag-im", "name": {"common": "Isle of Man", "deu": "Insel Man", "fra": "Île de Man", "hrv": "Otok Man", "ita": "Isola di Man", "jpn": "マン島", "nld": "Isle of Man", "por": "Ilha de Man", "rus": "Остров Мэн", "spa": "Isla de Man", "svk": "Man", "fin": "Mansaari", "zho": "马恩岛", "isr": "האי מאן"}}, "IL": {"currency": "ILS", "callingCode": "972", "flag": "flag-il", "name": {"common": "Israel", "deu": "Israel", "fra": "Israël", "hrv": "Izrael", "ita": "Israele", "jpn": "イスラエル", "nld": "Israël", "por": "Israel", "rus": "Израиль", "spa": "Israel", "svk": "Izrael", "fin": "Israel", "zho": "以色列", "isr": "ישראל"}}, "IT": {"currency": "EUR", "callingCode": "39", "flag": "flag-it", "name": {"common": "Italy", "deu": "Italien", "fra": "Italie", "hrv": "Italija", "ita": "Italia", "jpn": "イタリア", "nld": "Italië", "por": "Itália", "rus": "Италия", "spa": "Italia", "svk": "Taliansko", "fin": "Italia", "zho": "意大利", "isr": "איטליה"}}, "CI": {"currency": "XOF", "callingCode": "225", "flag": "flag-ci", "name": {"common": "Ivory Coast", "deu": "Elfenbeinküste", "fra": "Côte d'Ivoire", "hrv": "Obala Bjelokosti", "ita": "Costa d'Avorio", "jpn": "コートジボワール", "nld": "Ivoorkust", "por": "Costa do Marfim", "rus": "Кот-д’Ивуар", "spa": "Costa de Marfil", "svk": "Pobržie Slonoviny", "fin": "Norsunluurannikko", "zho": "科特迪瓦", "isr": "חוף השנהב"}}, "JM": {"currency": "JMD", "callingCode": "1876", "flag": "flag-jm", "name": {"common": "Jamaica", "deu": "Jamaika", "fra": "Jamaïque", "hrv": "Jamajka", "ita": "Giamaica", "jpn": "ジャマイカ", "nld": "Jamaica", "por": "Jamaica", "rus": "Ямайка", "spa": "Jamaica", "svk": "Jamajka", "fin": "Jamaika", "zho": "牙买加", "isr": "ג׳מייקה"}}, "JP": {"currency": "JPY", "callingCode": "81", "flag": "flag-jp", "name": {"common": "Japan", "deu": "Japan", "fra": "Japon", "hrv": "Japan", "ita": "Giappone", "jpn": "日本", "nld": "Japan", "por": "Japão", "rus": "Япония", "spa": "Japón", "svk": "Japonsko", "fin": "Japani", "zho": "日本", "isr": "יפן"}}, "JE": {"currency": "GBP", "callingCode": "44", "flag": "flag-je", "name": {"common": "Jersey", "deu": "Jersey", "fra": "Jersey", "hrv": "Jersey", "ita": "Isola di Jersey", "jpn": "ジャージー", "nld": "Jersey", "por": "Jersey", "rus": "Джерси", "spa": "Jersey", "svk": "Jersey", "fin": "Jersey", "zho": "泽西岛", "isr": "ג׳רסי"}}, "JO": {"currency": "JOD", "callingCode": "962", "flag": "flag-jo", "name": {"common": "Jordan", "deu": "Jordanien", "fra": "Jordanie", "hrv": "Jordan", "ita": "Giordania", "jpn": "ヨルダン", "nld": "Jordanië", "por": "Jordânia", "rus": "Иордания", "spa": "Jordania", "svk": "Jordánsko", "fin": "Jordania", "zho": "约旦", "isr": "ירדן"}}, "KZ": {"currency": "KZT", "callingCode": "76", "flag": "flag-kz", "name": {"common": "Kazakhstan", "deu": "Kasachstan", "fra": "Kazakhstan", "hrv": "Kazahstan", "ita": "Kazakistan", "jpn": "カザフスタン", "nld": "Kazachstan", "por": "Cazaquistão", "rus": "Казахстан", "spa": "Kazajistán", "svk": "Kazachstan", "fin": "Kazakstan", "zho": "哈萨克斯坦", "isr": "קזחסטן"}}, "KE": {"currency": "KES", "callingCode": "254", "flag": "flag-ke", "name": {"common": "Kenya", "deu": "Kenia", "fra": "Kenya", "hrv": "Kenija", "ita": "Kenya", "jpn": "ケニア", "nld": "Kenia", "por": "Quénia", "rus": "Кения", "spa": "Kenia", "svk": "Keňa", "fin": "Kenia", "zho": "肯尼亚", "isr": "קניה"}}, "KI": {"currency": "AUD", "callingCode": "686", "flag": "flag-ki", "name": {"common": "Kiribati", "deu": "Kiribati", "fra": "Kiribati", "hrv": "Kiribati", "ita": "Kiribati", "jpn": "キリバス", "nld": "Kiribati", "por": "Kiribati", "rus": "Кирибати", "spa": "Kiribati", "svk": "Kiribati", "fin": "Kiribati", "zho": "基里巴斯", "isr": "קיריבאטי"}}, "XK": {"currency": "EUR", "callingCode": "383", "flag": "flag-xk", "name": {"common": "Kosovo", "deu": "Kosovo", "fra": "Kosovo", "hrv": "Kosovo", "ita": "Kosovo", "nld": "Kosovo", "por": "Kosovo", "rus": "Республика Косово", "spa": "Kosovo", "svk": "Kosovo", "fin": "Kosovo", "zho": "科索沃", "isr": "קוסובו"}}, "KW": {"currency": "KWD", "callingCode": "965", "flag": "flag-kw", "name": {"common": "Kuwait", "deu": "Kuwait", "fra": "Koweït", "hrv": "Kuvajt", "ita": "Kuwait", "jpn": "クウェート", "nld": "Koeweit", "por": "Kuwait", "rus": "Кувейт", "spa": "Kuwait", "svk": "Kuvajt", "fin": "Kuwait", "zho": "科威特", "isr": "כווית"}}, "KG": {"currency": "KGS", "callingCode": "996", "flag": "flag-kg", "name": {"common": "Kyrgyzstan", "deu": "Kirgisistan", "fra": "Kirghizistan", "hrv": "Kirgistan", "ita": "Kirghizistan", "jpn": "キルギス", "nld": "Kirgizië", "por": "Quirguistão", "rus": "Киргизия", "spa": "Kirguizistán", "svk": "Kirgizsko", "fin": "Kirgisia", "zho": "吉尔吉斯斯坦", "isr": "קירגיזסטן"}}, "LA": {"currency": "LAK", "callingCode": "856", "flag": "flag-la", "name": {"common": "Laos", "deu": "Laos", "fra": "Laos", "hrv": "Laos", "ita": "Laos", "jpn": "ラオス人民民主共和国", "nld": "Laos", "por": "Laos", "rus": "Лаос", "spa": "Laos", "svk": "Laos", "fin": "Laos", "zho": "老挝", "isr": "לאוס"}}, "LV": {"currency": "EUR", "callingCode": "371", "flag": "flag-lv", "name": {"common": "Latvia", "deu": "Lettland", "fra": "Lettonie", "hrv": "Latvija", "ita": "Lettonia", "jpn": "ラトビア", "nld": "Letland", "por": "Letónia", "rus": "Латвия", "spa": "Letonia", "svk": "Lotyšsko", "fin": "Latvia", "zho": "拉脱维亚", "isr": "לטביה"}}, "LB": {"currency": "LBP", "callingCode": "961", "flag": "flag-lb", "name": {"common": "Lebanon", "deu": "Libanon", "fra": "Liban", "hrv": "Libanon", "ita": "Libano", "jpn": "レバノン", "nld": "Libanon", "por": "Líbano", "rus": "Ливан", "spa": "Líbano", "svk": "Libanon", "fin": "Libanon", "zho": "黎巴嫩", "isr": "לבנון"}}, "LS": {"currency": "LSL", "callingCode": "266", "flag": "flag-ls", "name": {"common": "Lesotho", "deu": "Lesotho", "fra": "Lesotho", "hrv": "Lesoto", "ita": "Lesotho", "jpn": "レソト", "nld": "Lesotho", "por": "Lesoto", "rus": "Лесото", "spa": "Lesotho", "svk": "Lesotho", "fin": "Lesotho", "zho": "莱索托", "isr": "לסוטו"}}, "LR": {"currency": "LRD", "callingCode": "231", "flag": "flag-lr", "name": {"common": "Liberia", "deu": "Liberia", "fra": "Liberia", "hrv": "Liberija", "ita": "Liberia", "jpn": "リベリア", "nld": "Liberia", "por": "Libéria", "rus": "Либерия", "spa": "Liberia", "svk": "Libéria", "fin": "Liberia", "zho": "利比里亚", "isr": "ליבריה"}}, "LY": {"currency": "LYD", "callingCode": "218", "flag": "flag-ly", "name": {"common": "Libya", "deu": "Libyen", "fra": "Libye", "hrv": "Libija", "ita": "Libia", "jpn": "リビア", "nld": "Libië", "por": "Líbia", "rus": "Ливия", "spa": "Libia", "svk": "Líbya", "fin": "Libya", "zho": "利比亚", "isr": "לוב"}}, "LI": {"currency": "CHF", "callingCode": "423", "flag": "flag-li", "name": {"common": "Liechtenstein", "deu": "Liechtenstein", "fra": "Liechtenstein", "hrv": "Lihtenštajn", "ita": "Liechtenstein", "jpn": "リヒテンシュタイン", "nld": "Liechtenstein", "por": "Liechtenstein", "rus": "Лихтенштейн", "spa": "Liechtenstein", "svk": "Lichtenštajnsko", "fin": "Liechenstein", "zho": "列支敦士登", "isr": "ליכטנשטיין"}}, "LT": {"currency": "EUR", "callingCode": "370", "flag": "flag-lt", "name": {"common": "Lithuania", "deu": "Litauen", "fra": "Lituanie", "hrv": "Litva", "ita": "Lituania", "jpn": "リトアニア", "nld": "Litouwen", "por": "Lituânia", "rus": "Литва", "spa": "Lituania", "svk": "Litva", "fin": "Liettua", "zho": "立陶宛", "isr": "ליטא"}}, "LU": {"currency": "EUR", "callingCode": "352", "flag": "flag-lu", "name": {"common": "Luxembourg", "deu": "Luxemburg", "fra": "Luxembourg", "hrv": "Luksemburg", "ita": "Lussemburgo", "jpn": "ルクセンブルク", "nld": "Luxemburg", "por": "Luxemburgo", "rus": "Люксембург", "spa": "Luxemburgo", "svk": "Luxembursko", "fin": "Luxemburg", "zho": "卢森堡", "isr": "לוקסמבורג"}}, "MO": {"currency": "MOP", "callingCode": "853", "flag": "flag-mo", "name": {"common": "Macau", "deu": "Macao", "fra": "Macao", "hrv": "Makao", "ita": "Macao", "jpn": "マカオ", "nld": "Macao", "por": "Macau", "rus": "Макао", "spa": "Macao", "fin": "Macao", "isr": "מקאו (מחוז מנהלי מיוחד של סין)"}}, "MK": {"currency": "MKD", "callingCode": "389", "flag": "flag-mk", "name": {"common": "Macedonia", "deu": "Mazedonien", "fra": "Macédoine", "hrv": "Makedonija", "ita": "Macedonia", "jpn": "マケドニア旧ユーゴスラビア共和国", "nld": "Macedonië", "por": "Macedónia", "rus": "Республика Македония", "spa": "Macedonia", "svk": "Macedónsko", "fin": "Makedonia", "zho": "马其顿", "isr": "מקדוניה"}}, "MG": {"currency": "MGA", "callingCode": "261", "flag": "flag-mg", "name": {"common": "Madagascar", "deu": "Madagaskar", "fra": "Madagascar", "hrv": "Madagaskar", "ita": "Madagascar", "jpn": "マダガスカル", "nld": "Madagaskar", "por": "Madagáscar", "rus": "Мадагаскар", "spa": "Madagascar", "svk": "Madagaskar", "fin": "Madagaskar", "zho": "马达加斯加", "isr": "מדגסקר"}}, "MW": {"currency": "MWK", "callingCode": "265", "flag": "flag-mw", "name": {"common": "Malawi", "deu": "Malawi", "fra": "Malawi", "hrv": "Malavi", "ita": "Malawi", "jpn": "マラウイ", "nld": "Malawi", "por": "Malawi", "rus": "Малави", "spa": "Malawi", "svk": "Malawi", "fin": "Malawi", "zho": "马拉维", "isr": "מלאווי"}}, "MY": {"currency": "MYR", "callingCode": "60", "flag": "flag-my", "name": {"common": "Malaysia", "deu": "Malaysia", "fra": "Malaisie", "hrv": "Malezija", "ita": "Malesia", "jpn": "マレーシア", "nld": "Maleisië", "por": "Malásia", "rus": "Малайзия", "spa": "Malasia", "svk": "Malajzia", "fin": "Malesia", "zho": "马来西亚", "isr": "מלזיה"}}, "MV": {"currency": "MVR", "callingCode": "960", "flag": "flag-mv", "name": {"common": "Maldives", "deu": "Malediven", "fra": "Maldives", "hrv": "Maldivi", "ita": "Maldive", "jpn": "モルディブ", "nld": "Maldiven", "por": "Maldivas", "rus": "Мальдивы", "spa": "Maldivas", "svk": "Maldivy", "fin": "Malediivit", "zho": "马尔代夫", "isr": "האיים המלדיביים"}}, "ML": {"currency": "XOF", "callingCode": "223", "flag": "flag-ml", "name": {"common": "Mali", "deu": "Mali", "fra": "Mali", "hrv": "Mali", "ita": "Mali", "jpn": "マリ", "nld": "Mali", "por": "Mali", "rus": "Мали", "spa": "Mali", "svk": "Mali", "fin": "Mali", "zho": "马里", "isr": "מאלי"}}, "MT": {"currency": "EUR", "callingCode": "356", "flag": "flag-mt", "name": {"common": "Malta", "deu": "Malta", "fra": "Malte", "hrv": "Malta", "ita": "Malta", "jpn": "マルタ", "nld": "Malta", "por": "Malta", "rus": "Мальта", "spa": "Malta", "svk": "Malta", "fin": "Malta", "zho": "马耳他", "isr": "מלטה"}}, "MH": {"currency": "USD", "callingCode": "692", "flag": "flag-mh", "name": {"common": "Marshall Islands", "deu": "Marshallinseln", "fra": "Îles Marshall", "hrv": "Maršalovi Otoci", "ita": "Isole Marshall", "jpn": "マーシャル諸島", "nld": "Marshalleilanden", "por": "Ilhas Marshall", "rus": "Маршалловы Острова", "spa": "Islas Marshall", "svk": "Marshallove ostrovy", "fin": "Marshallinsaaret", "zho": "马绍尔群岛", "isr": "איי מרשל"}}, "MQ": {"currency": "EUR", "callingCode": "596", "flag": "flag-mq", "name": {"common": "Martinique", "deu": "Martinique", "fra": "Martinique", "hrv": "Martinique", "ita": "Martinica", "jpn": "マルティニーク", "nld": "Martinique", "por": "Martinica", "rus": "Мартиника", "spa": "Martinica", "svk": "Martinik", "fin": "Martinique", "zho": "马提尼克", "isr": "מרטיניק"}}, "MR": {"currency": "MRO", "callingCode": "222", "flag": "flag-mr", "name": {"common": "Mauritania", "deu": "Mauretanien", "fra": "Mauritanie", "hrv": "Mauritanija", "ita": "Mauritania", "jpn": "モーリタニア", "nld": "Mauritanië", "por": "Mauritânia", "rus": "Мавритания", "spa": "Mauritania", "svk": "Mauritánia", "fin": "Mauritania", "zho": "毛里塔尼亚", "isr": "מאוריטניה"}}, "MU": {"currency": "MUR", "callingCode": "230", "flag": "flag-mu", "name": {"common": "Mauritius", "deu": "Mauritius", "fra": "Île Maurice", "hrv": "Mauricijus", "ita": "Mauritius", "jpn": "モーリシャス", "nld": "Mauritius", "por": "Maurício", "rus": "Маврикий", "spa": "Mauricio", "svk": "Maurícius", "fin": "Mauritius", "zho": "毛里求斯", "isr": "מאוריציוס"}}, "YT": {"currency": "EUR", "callingCode": "262", "flag": "flag-yt", "name": {"common": "Mayotte", "deu": "Mayotte", "fra": "Mayotte", "hrv": "Mayotte", "ita": "Mayotte", "jpn": "マヨット", "nld": "Mayotte", "por": "Mayotte", "rus": "Майотта", "spa": "Mayotte", "svk": "Mayotte", "fin": "Mayotte", "zho": "马约特", "isr": "מאיוט"}}, "MX": {"currency": "MXN", "callingCode": "52", "flag": "flag-mx", "name": {"common": "Mexico", "deu": "Mexiko", "fra": "Mexique", "hrv": "Meksiko", "ita": "Messico", "jpn": "メキシコ", "nld": "Mexico", "por": "México", "rus": "Мексика", "spa": "México", "svk": "Mexiko", "fin": "Meksiko", "zho": "墨西哥", "isr": "מקסיקו"}}, "FM": {"currency": "USD", "callingCode": "691", "flag": "flag-fm", "name": {"common": "Micronesia", "deu": "Mikronesien", "fra": "Micronésie", "hrv": "Mikronezija", "ita": "Micronesia", "jpn": "ミクロネシア連邦", "nld": "Micronesië", "por": "Micronésia", "rus": "Федеративные Штаты Микронезии", "spa": "Micronesia", "svk": "Mikronézia", "fin": "Mikronesia", "zho": "密克罗尼西亚", "isr": "מיקרונזיה"}}, "MD": {"currency": "MDL", "callingCode": "373", "flag": "flag-md", "name": {"common": "Moldova", "deu": "Moldawie", "fra": "Moldavie", "hrv": "Moldova", "ita": "Moldavia", "jpn": "モルドバ共和国", "nld": "Moldavië", "por": "Moldávia", "rus": "Молдавия", "spa": "Moldavia", "svk": "Moldavsko", "fin": "Moldova", "zho": "摩尔多瓦", "isr": "מולדובה"}}, "MC": {"currency": "EUR", "callingCode": "377", "flag": "flag-mc", "name": {"common": "Monaco", "deu": "Monaco", "fra": "Monaco", "hrv": "Monako", "ita": "Principato di Monaco", "jpn": "モナコ", "nld": "Monaco", "por": "Mónaco", "rus": "Монако", "spa": "Mónaco", "svk": "Monako", "fin": "Monaco", "zho": "摩纳哥", "isr": "מונקו"}}, "MN": {"currency": "MNT", "callingCode": "976", "flag": "flag-mn", "name": {"common": "Mongolia", "deu": "Mongolei", "fra": "Mongolie", "hrv": "Mongolija", "ita": "Mongolia", "jpn": "モンゴル", "nld": "Mongolië", "por": "Mongólia", "rus": "Монголия", "spa": "Mongolia", "svk": "Mongolsko", "fin": "Mongolia", "zho": "蒙古", "isr": "מונגוליה"}}, "ME": {"currency": "EUR", "callingCode": "382", "flag": "flag-me", "name": {"common": "Montenegro", "deu": "Montenegro", "fra": "Monténégro", "hrv": "Crna Gora", "ita": "Montenegro", "jpn": "モンテネグロ", "nld": "Montenegro", "por": "Montenegro", "rus": "Черногория", "spa": "Montenegro", "svk": "Čierna Hora", "fin": "Montenegro", "zho": "黑山", "isr": "מונטנגרו"}}, "MS": {"currency": "XCD", "callingCode": "1664", "flag": "flag-ms", "name": {"common": "Montserrat", "deu": "Montserrat", "fra": "Montserrat", "hrv": "Montserrat", "ita": "Montserrat", "jpn": "モントセラト", "nld": "Montserrat", "por": "Montserrat", "rus": "Монтсеррат", "spa": "Montserrat", "svk": "Montserrat", "fin": "Montserrat", "zho": "蒙特塞拉特", "isr": "מונסראט"}}, "MA": {"currency": "MAD", "callingCode": "212", "flag": "flag-ma", "name": {"common": "Morocco", "deu": "Marokko", "fra": "Maroc", "hrv": "Maroko", "ita": "Marocco", "jpn": "モロッコ", "nld": "Marokko", "por": "Marrocos", "rus": "Марокко", "spa": "Marruecos", "svk": "Maroko", "fin": "Marokko", "zho": "摩洛哥", "isr": "מרוקו"}}, "MZ": {"currency": "MZN", "callingCode": "258", "flag": "flag-mz", "name": {"common": "Mozambique", "deu": "Mosambik", "fra": "Mozambique", "hrv": "Mozambik", "ita": "Mozambico", "jpn": "モザンビーク", "nld": "Mozambique", "por": "Moçambique", "rus": "Мозамбик", "spa": "Mozambique", "svk": "Mozambik", "fin": "Mosambik", "zho": "莫桑比克", "isr": "מוזמביק"}}, "MM": {"currency": "MMK", "callingCode": "95", "flag": "flag-mm", "name": {"common": "Myanmar", "deu": "Myanmar", "fra": "Birmanie", "hrv": "Mijanmar", "ita": "Birmania", "jpn": "ミャンマー", "nld": "Myanmar", "por": "Myanmar", "rus": "Мьянма", "spa": "Myanmar", "svk": "Mjanmarsko", "fin": "Myanmar", "zho": "缅甸", "isr": "מיאנמר (בורמה)"}}, "NA": {"currency": "NAD", "callingCode": "264", "flag": "flag-na", "name": {"common": "Namibia", "deu": "Namibia", "fra": "Namibie", "hrv": "Namibija", "ita": "Namibia", "jpn": "ナミビア", "nld": "Namibië", "por": "Namíbia", "rus": "Намибия", "spa": "Namibia", "svk": "Namíbia", "fin": "Namibia", "zho": "纳米比亚", "isr": "נמיביה"}}, "NR": {"currency": "AUD", "callingCode": "674", "flag": "flag-nr", "name": {"common": "Nauru", "deu": "Nauru", "fra": "Nauru", "hrv": "Nauru", "ita": "Nauru", "jpn": "ナウル", "nld": "Nauru", "por": "Nauru", "rus": "Науру", "spa": "Nauru", "svk": "Nauru", "fin": "Nauru", "zho": "瑙鲁", "isr": "נאורו"}}, "NP": {"currency": "NPR", "callingCode": "977", "flag": "flag-np", "name": {"common": "Nepal", "deu": "Népal", "fra": "Népal", "hrv": "Nepal", "ita": "Nepal", "jpn": "ネパール", "nld": "Nepal", "por": "Nepal", "rus": "Непал", "spa": "Nepal", "svk": "Nepál", "fin": "Nepal", "zho": "尼泊尔", "isr": "נפאל"}}, "NL": {"currency": "EUR", "callingCode": "31", "flag": "flag-nl", "name": {"common": "Netherlands", "deu": "Niederlande", "fra": "Pays-Bas", "hrv": "Nizozemska", "ita": "Paesi Bassi", "jpn": "オランダ", "nld": "Nederland", "por": "Holanda", "rus": "Нидерланды", "spa": "Países Bajos", "svk": "Holansko", "fin": "Alankomaat", "zho": "荷兰", "isr": "הולנד"}}, "NC": {"currency": "XPF", "callingCode": "687", "flag": "flag-nc", "name": {"common": "New Caledonia", "deu": "Neukaledonien", "fra": "Nouvelle-Calédonie", "hrv": "Nova Kaledonija", "ita": "Nuova Caledonia", "jpn": "ニューカレドニア", "nld": "Nieuw-Caledonië", "por": "Nova Caledónia", "rus": "Новая Каледония", "spa": "Nueva Caledonia", "svk": "Nová Kaledónia", "fin": "Uusi-Kaledonia", "zho": "新喀里多尼亚", "isr": "קלדוניה החדשה"}}, "NZ": {"currency": "NZD", "callingCode": "64", "flag": "flag-nz", "name": {"common": "New Zealand", "deu": "Neuseeland", "fra": "Nouvelle-Zélande", "hrv": "Novi Zeland", "ita": "Nuova Zelanda", "jpn": "ニュージーランド", "nld": "Nieuw-Zeeland", "por": "Nova Zelândia", "rus": "Новая Зеландия", "spa": "Nueva Zelanda", "svk": "Nový Zéland", "fin": "Uusi-Seelanti", "zho": "新西兰", "isr": "ניו זילנד"}}, "NI": {"currency": "NIO", "callingCode": "505", "flag": "flag-ni", "name": {"common": "Nicaragua", "deu": "Nicaragua", "fra": "Nicaragua", "hrv": "Nikaragva", "ita": "Nicaragua", "jpn": "ニカラグア", "nld": "Nicaragua", "por": "Nicarágua", "rus": "Никарагуа", "spa": "Nicaragua", "svk": "Nikaragua", "fin": "Nicaragua", "zho": "尼加拉瓜", "isr": "ניקרגואה"}}, "NE": {"currency": "XOF", "callingCode": "227", "flag": "flag-ne", "name": {"common": "Niger", "deu": "Niger", "fra": "Niger", "hrv": "Niger", "ita": "Niger", "jpn": "ニジェール", "nld": "Niger", "por": "Níger", "rus": "Нигер", "spa": "Níger", "svk": "Niger", "fin": "Niger", "zho": "尼日尔", "isr": "ניז׳ר"}}, "NG": {"currency": "NGN", "callingCode": "234", "flag": "flag-ng", "name": {"common": "Nigeria", "deu": "Nigeria", "fra": "Nigéria", "hrv": "Nigerija", "ita": "Nigeria", "jpn": "ナイジェリア", "nld": "Nigeria", "por": "Nigéria", "rus": "Нигерия", "spa": "Nigeria", "svk": "Nigéria", "fin": "Nigeria", "zho": "尼日利亚", "isr": "ניגריה"}}, "NU": {"currency": "NZD", "callingCode": "683", "flag": "flag-nu", "name": {"common": "Niue", "deu": "Niue", "fra": "Niue", "hrv": "Niue", "ita": "Niue", "jpn": "ニウエ", "nld": "Niue", "por": "Niue", "rus": "Ниуэ", "spa": "Niue", "svk": "Niue", "fin": "Niue", "zho": "纽埃", "isr": "ניווה"}}, "NF": {"currency": "AUD", "callingCode": "672", "flag": "flag-nf", "name": {"common": "Norfolk Island", "deu": "Norfolkinsel", "fra": "Île Norfolk", "hrv": "Otok Norfolk", "ita": "Isola Norfolk", "jpn": "ノーフォーク島", "nld": "Norfolkeiland", "por": "Ilha Norfolk", "rus": "Норфолк", "spa": "Isla de Norfolk", "svk": "Norfolk", "fin": "Norfolkinsaari", "zho": "诺福克岛", "isr": "איי נורפוק"}}, "KP": {"currency": "KPW", "callingCode": "850", "flag": "flag-kp", "name": {"common": "North Korea", "deu": "Nordkorea", "fra": "Corée du Nord", "hrv": "Sjeverna Koreja", "ita": "Corea del Nord", "jpn": "朝鮮民主主義人民共和国", "nld": "Noord-Korea", "por": "Coreia do Norte", "rus": "Северная Корея", "spa": "Corea del Norte", "svk": "Kórejská ľudovodemokratická republika (KĽR, Severná Kó)", "fin": "Pohjois-Korea", "zho": "朝鲜", "isr": "קוריאה הצפונית"}}, "MP": {"currency": "USD", "callingCode": "1670", "flag": "flag-mp", "name": {"common": "Northern Mariana Islands", "deu": "Nördliche Marianen", "fra": "Îles Mariannes du Nord", "hrv": "Sjevernomarijanski otoci", "ita": "Isole Marianne Settentrionali", "jpn": "北マリアナ諸島", "nld": "Noordelijke Marianeneilanden", "por": "Marianas Setentrionais", "rus": "Северные Марианские острова", "spa": "Islas Marianas del Norte", "svk": "Severné Mariány", "fin": "Pohjois-Mariaanit", "zho": "北马里亚纳群岛", "isr": "איי מריאנה הצפוניים"}}, "NO": {"currency": "NOK", "callingCode": "47", "flag": "flag-no", "name": {"common": "Norway", "deu": "Norwegen", "fra": "Norvège", "hrv": "Norveška", "ita": "Norvegia", "jpn": "ノルウェー", "nld": "Noorwegen", "por": "Noruega", "rus": "Норвегия", "spa": "Noruega", "svk": "Nórsko", "fin": "Norja", "zho": "挪威", "isr": "נורווגיה"}}, "OM": {"currency": "OMR", "callingCode": "968", "flag": "flag-om", "name": {"common": "Oman", "deu": "Oman", "fra": "Oman", "hrv": "Oman", "ita": "oman", "jpn": "オマーン", "nld": "Oman", "por": "Omã", "rus": "Оман", "spa": "Omán", "svk": "Omán", "fin": "Oman", "zho": "阿曼", "isr": "עומאן"}}, "PK": {"currency": "PKR", "callingCode": "92", "flag": "flag-pk", "name": {"common": "Pakistan", "deu": "Pakistan", "fra": "Pakistan", "hrv": "Pakistan", "ita": "Pakistan", "jpn": "パキスタン", "nld": "Pakistan", "por": "Paquistão", "rus": "Пакистан", "spa": "Pakistán", "svk": "Pakistan", "fin": "Pakistan", "zho": "巴基斯坦", "isr": "פקיסטן"}}, "PW": {"currency": "USD", "callingCode": "680", "flag": "flag-pw", "name": {"common": "Palau", "deu": "Palau", "fra": "Palaos (Palau)", "hrv": "Palau", "ita": "Palau", "jpn": "パラオ", "nld": "Palau", "por": "Palau", "rus": "Палау", "spa": "Palau", "svk": "Palau", "fin": "Palau", "zho": "帕劳", "isr": "פלאו"}}, "PS": {"currency": "ILS", "callingCode": "970", "flag": "flag-ps", "name": {"common": "Palestine", "deu": "Palästina", "fra": "Palestine", "hrv": "Palestina", "ita": "Palestina", "jpn": "パレスチナ", "nld": "Palestijnse gebieden", "por": "Palestina", "rus": "Палестина", "spa": "Palestina", "svk": "Palestína", "fin": "Palestiina", "zho": "巴勒斯坦", "isr": "השטחים הפלסטיניים"}}, "PA": {"currency": "PAB", "callingCode": "507", "flag": "flag-pa", "name": {"common": "Panama", "deu": "Panama", "fra": "Panama", "hrv": "Panama", "ita": "Panama", "jpn": "パナマ", "nld": "Panama", "por": "Panamá", "rus": "Панама", "spa": "Panamá", "svk": "Panama", "fin": "Panama", "zho": "巴拿马", "isr": "פנמה"}}, "PG": {"currency": "PGK", "callingCode": "675", "flag": "flag-pg", "name": {"common": "Papua New Guinea", "deu": "Papua-Neuguinea", "fra": "Papouasie-Nouvelle-Guinée", "hrv": "Papua Nova Gvineja", "ita": "Papua Nuova Guinea", "jpn": "パプアニューギニア", "nld": "Papoea-Nieuw-Guinea", "por": "Papua Nova Guiné", "rus": "Папуа — Новая Гвинея", "spa": "Papúa Nueva Guinea", "svk": "Papua-Nová Guinea", "fin": "Papua-Uusi-Guinea", "zho": "巴布亚新几内亚", "isr": "פפואה גינאה החדשה"}}, "PY": {"currency": "PYG", "callingCode": "595", "flag": "flag-py", "name": {"common": "Paraguay", "deu": "Paraguay", "fra": "Paraguay", "hrv": "Paragvaj", "ita": "Paraguay", "jpn": "パラグアイ", "nld": "Paraguay", "por": "Paraguai", "rus": "Парагвай", "spa": "Paraguay", "svk": "Paraguaj", "fin": "Paraguay", "zho": "巴拉圭", "isr": "פרגוואי"}}, "PE": {"currency": "PEN", "callingCode": "51", "flag": "flag-pe", "name": {"common": "Peru", "deu": "Peru", "fra": "Pérou", "hrv": "Peru", "ita": "Perù", "jpn": "ペルー", "nld": "Peru", "por": "Perú", "rus": "Перу", "spa": "Perú", "svk": "Peru", "fin": "Peru", "zho": "秘鲁", "isr": "פרו"}}, "PH": {"currency": "PHP", "callingCode": "63", "flag": "flag-ph", "name": {"common": "Philippines", "deu": "Philippinen", "fra": "Philippines", "hrv": "Filipini", "ita": "Filippine", "jpn": "フィリピン", "nld": "Filipijnen", "por": "Filipinas", "rus": "Филиппины", "spa": "Filipinas", "svk": "Filipíny", "fin": "Filippiinit", "zho": "菲律宾", "isr": "הפיליפינים"}}, "PN": {"currency": "NZD", "callingCode": "64", "flag": "flag-pn", "name": {"common": "Pitcairn Islands", "deu": "Pitcairn", "fra": "Îles Pitcairn", "hrv": "Pitcairnovo otočje", "ita": "Isole Pitcairn", "jpn": "ピトケアン", "nld": "Pitcairneilanden", "por": "Ilhas Pitcairn", "rus": "Острова Питкэрн", "spa": "Islas Pitcairn", "svk": "Pitcairnove ostrovy", "fin": "Pitcairn", "zho": "皮特凯恩群岛", "isr": "איי פיטקרן"}}, "PL": {"currency": "PLN", "callingCode": "48", "flag": "flag-pl", "name": {"common": "Poland", "deu": "Polen", "fra": "Pologne", "hrv": "Poljska", "ita": "Polonia", "jpn": "ポーランド", "nld": "Polen", "por": "Polónia", "rus": "Польша", "spa": "Polonia", "svk": "Poľsko", "fin": "Puola", "zho": "波兰", "isr": "פולין"}}, "PT": {"currency": "EUR", "callingCode": "351", "flag": "flag-pt", "name": {"common": "Portugal", "deu": "Portugal", "fra": "Portugal", "hrv": "Portugal", "ita": "Portogallo", "jpn": "ポルトガル", "nld": "Portugal", "por": "Portugal", "rus": "Португалия", "spa": "Portugal", "svk": "Portugalsko", "fin": "Portugali", "zho": "葡萄牙", "isr": "פורטוגל"}}, "PR": {"currency": "USD", "callingCode": "1787", "flag": "flag-pr", "name": {"common": "Puerto Rico", "deu": "Puerto Rico", "fra": "Porto Rico", "hrv": "Portoriko", "ita": "Porto Rico", "jpn": "プエルトリコ", "nld": "Puerto Rico", "por": "Porto Rico", "rus": "Пуэрто-Рико", "spa": "Puerto Rico", "svk": "Portoriko", "fin": "Puerto Rico", "zho": "波多黎各", "isr": "פוארטו ריקו"}}, "QA": {"currency": "QAR", "callingCode": "974", "flag": "flag-qa", "name": {"common": "Qatar", "deu": "Katar", "fra": "Qatar", "hrv": "Katar", "ita": "Qatar", "jpn": "カタール", "nld": "Qatar", "por": "Catar", "rus": "Катар", "spa": "Catar", "svk": "Katar", "fin": "Qatar", "zho": "卡塔尔", "isr": "קטאר"}}, "CG": {"currency": "XAF", "callingCode": "242", "flag": "flag-cg", "name": {"common": "Republic of the Congo", "cym": "Gweriniaeth y Congo", "deu": "Kongo", "fra": "Congo", "hrv": "Kongo", "ita": "Congo", "jpn": "コンゴ共和国", "nld": "Congo", "por": "Congo", "rus": "Республика Конго", "spa": "Congo", "svk": "Kongo", "fin": "Kongo-Brazzaville", "zho": "刚果", "isr": "קונגו - ברזאויל"}}, "RO": {"currency": "RON", "callingCode": "40", "flag": "flag-ro", "name": {"common": "Romania", "deu": "Rumänien", "fra": "Roumanie", "hrv": "Rumunjska", "ita": "Romania", "jpn": "ルーマニア", "nld": "Roemenië", "por": "Roménia", "rus": "Румыния", "spa": "Rumania", "svk": "Rumunsko", "fin": "Romania", "zho": "罗马尼亚", "isr": "רומניה"}}, "RU": {"currency": "RUB", "callingCode": "7", "flag": "flag-ru", "name": {"common": "Russia", "deu": "Russland", "fra": "Russie", "hrv": "Rusija", "ita": "Russia", "jpn": "ロシア連邦", "nld": "Rusland", "por": "Rússia", "rus": "Россия", "spa": "Rusia", "svk": "Rusko", "fin": "Venäjä", "zho": "俄罗斯", "isr": "רוסיה"}}, "RW": {"currency": "RWF", "callingCode": "250", "flag": "flag-rw", "name": {"common": "Rwanda", "deu": "Ruanda", "fra": "Rwanda", "hrv": "Ruanda", "ita": "Ruanda", "jpn": "ルワンダ", "nld": "Rwanda", "por": "Ruanda", "rus": "Руанда", "spa": "Ruanda", "svk": "Rwanda", "fin": "Ruanda", "zho": "卢旺达", "isr": "רואנדה"}}, "RE": {"currency": "EUR", "callingCode": "262", "flag": "flag-re", "name": {"common": "Réunion", "deu": "Réunion", "fra": "Réunion", "hrv": "Réunion", "ita": "Riunione", "jpn": "レユニオン", "nld": "Réunion", "por": "Reunião", "rus": "Реюньон", "spa": "Reunión", "svk": "Réunion", "fin": "Réunion", "zho": "留尼旺岛", "isr": "ראוניון"}}, "BL": {"currency": "EUR", "callingCode": "590", "flag": "flag-bl", "name": {"common": "Saint Barthélemy", "deu": "Saint-Barthélemy", "fra": "Saint-Barthélemy", "hrv": "Saint Barthélemy", "ita": "Antille Francesi", "jpn": "サン・バルテルミー", "nld": "Saint Barthélemy", "por": "São Bartolomeu", "rus": "Сен-Бартелеми", "spa": "San Bartolomé", "svk": "Svätý Bartolomej", "fin": "Saint-Barthélemy", "zho": "圣巴泰勒米", "isr": "סנט ברתולומיאו"}}, "KN": {"currency": "XCD", "callingCode": "1869", "flag": "flag-kn", "name": {"common": "Saint Kitts and Nevis", "deu": "Saint Christopher und Nevis", "fra": "Saint-Christophe-et-Niévès", "hrv": "Sveti Kristof i Nevis", "ita": "Saint Kitts e Nevis", "jpn": "セントクリストファー・ネイビス", "nld": "Saint Kitts en Nevis", "por": "São Cristóvão e Nevis", "rus": "Сент-Китс и Невис", "spa": "San Cristóbal y Nieves", "svk": "Svätý Krištof a Nevis", "fin": "Saint Kitts ja Nevis", "zho": "圣基茨和尼维斯", "isr": "סנט קיטס ונוויס"}}, "LC": {"currency": "XCD", "callingCode": "1758", "flag": "flag-lc", "name": {"common": "Saint Lucia", "deu": "Saint Lucia", "fra": "Sainte-Lucie", "hrv": "Sveta Lucija", "ita": "Santa Lucia", "jpn": "セントルシア", "nld": "Saint Lucia", "por": "Santa Lúcia", "rus": "Сент-Люсия", "spa": "Santa Lucía", "svk": "Svätá Lucia", "fin": "Saint Lucia", "zho": "圣卢西亚", "isr": "סנט לוסיה"}}, "MF": {"currency": "EUR", "callingCode": "590", "flag": "flag-mf", "name": {"common": "Saint Martin", "deu": "Saint Martin", "fra": "Saint-Martin", "hrv": "Sveti Martin", "ita": "Saint Martin", "jpn": "サン・マルタン(フランス領)", "nld": "Saint-Martin", "por": "São Martinho", "rus": "Сен-Мартен", "spa": "Saint Martin", "svk": "Svätý Martin", "fin": "Saint-Martin", "zho": "圣马丁", "isr": "סן מרטן"}}, "PM": {"currency": "EUR", "callingCode": "508", "flag": "flag-pm", "name": {"common": "Saint Pierre and Miquelon", "deu": "Saint-Pierre und Miquelon", "fra": "Saint-Pierre-et-Miquelon", "hrv": "Sveti Petar i Mikelon", "ita": "Saint-Pierre e Miquelon", "jpn": "サンピエール島・ミクロン島", "nld": "Saint Pierre en Miquelon", "por": "Saint-Pierre e Miquelon", "rus": "Сен-Пьер и Микелон", "spa": "San Pedro y Miquelón", "svk": "Saint Pierre a Miquelon", "fin": "Saint-Pierre ja Miquelon", "zho": "圣皮埃尔和密克隆", "isr": "סנט פייר ומיקלון"}}, "VC": {"currency": "XCD", "callingCode": "1784", "flag": "flag-vc", "name": {"common": "Saint Vincent and the Grenadines", "deu": "Saint Vincent und die Grenadinen", "fra": "Saint-Vincent-et-les-Grenadines", "hrv": "Sveti Vincent i Grenadini", "ita": "Saint Vincent e Grenadine", "jpn": "セントビンセントおよびグレナディーン諸島", "nld": "Saint Vincent en de Grenadines", "por": "São Vincente e Granadinas", "rus": "Сент-Винсент и Гренадины", "spa": "San Vicente y Granadinas", "svk": "Svätý Vincent a Grenadíny", "fin": "Saint Vincent ja Grenadiinit", "zho": "圣文森特和格林纳丁斯", "isr": "סנט וינסנט והגרנדינים"}}, "WS": {"currency": "WST", "callingCode": "685", "flag": "flag-ws", "name": {"common": "Samoa", "deu": "Samoa", "fra": "Samoa", "hrv": "Samoa", "ita": "Samoa", "jpn": "サモア", "nld": "Samoa", "por": "Samoa", "rus": "Самоа", "spa": "Samoa", "fin": "Samoa", "zho": "萨摩亚", "isr": "סמואה"}}, "SM": {"currency": "EUR", "callingCode": "378", "flag": "flag-sm", "name": {"common": "San Marino", "deu": "San Marino", "fra": "Saint-Marin", "hrv": "San Marino", "ita": "San Marino", "jpn": "サンマリノ", "nld": "San Marino", "por": "San Marino", "rus": "Сан-Марино", "spa": "San Marino", "svk": "San Maríno", "fin": "San Marino", "zho": "圣马力诺", "isr": "סן מרינו"}}, "SA": {"currency": "SAR", "callingCode": "966", "flag": "flag-sa", "name": {"common": "Saudi Arabia", "deu": "Saudi-Arabien", "fra": "Arabie Saoudite", "hrv": "Saudijska Arabija", "ita": "Arabia Saudita", "jpn": "サウジアラビア", "nld": "Saoedi-Arabië", "por": "Arábia Saudita", "rus": "Саудовская Аравия", "spa": "Arabia Saudí", "svk": "Saudská Arábia", "fin": "Saudi-Arabia", "zho": "沙特阿拉伯", "isr": "ערב הסעודית"}}, "SN": {"currency": "XOF", "callingCode": "221", "flag": "flag-sn", "name": {"common": "Senegal", "deu": "Senegal", "fra": "Sénégal", "hrv": "Senegal", "ita": "Senegal", "jpn": "セネガル", "nld": "Senegal", "por": "Senegal", "rus": "Сенегал", "spa": "Senegal", "svk": "Senegal", "fin": "Senegal", "zho": "塞内加尔", "isr": "סנגל"}}, "RS": {"currency": "RSD", "callingCode": "381", "flag": "flag-rs", "name": {"common": "Serbia", "deu": "Serbien", "fra": "Serbie", "hrv": "Srbija", "ita": "Serbia", "jpn": "セルビア", "nld": "Servië", "por": "Sérvia", "rus": "Сербия", "spa": "Serbia", "svk": "Srbsko", "fin": "Serbia", "zho": "塞尔维亚", "isr": "סרביה"}}, "SC": {"currency": "SCR", "callingCode": "248", "flag": "flag-sc", "name": {"common": "Seychelles", "deu": "Seychellen", "fra": "Seychelles", "hrv": "Sejšeli", "ita": "Seychelles", "jpn": "セーシェル", "nld": "Seychellen", "por": "Seicheles", "rus": "Сейшельские Острова", "spa": "Seychelles", "svk": "Seychely", "fin": "Seychellit", "zho": "塞舌尔", "isr": "איי סיישל"}}, "SL": {"currency": "SLL", "callingCode": "232", "flag": "flag-sl", "name": {"common": "Sierra Leone", "deu": "Sierra Leone", "fra": "Sierra Leone", "hrv": "Sijera Leone", "ita": "Sierra Leone", "jpn": "シエラレオネ", "nld": "Sierra Leone", "por": "Serra Leoa", "rus": "Сьерра-Леоне", "spa": "Sierra Leone", "svk": "Sierra Leone", "fin": "Sierra Leone", "zho": "塞拉利昂", "isr": "סיירה לאונה"}}, "SG": {"currency": "SGD", "callingCode": "65", "flag": "flag-sg", "name": {"common": "Singapore", "deu": "Singapur", "fra": "Singapour", "hrv": "Singapur", "ita": "Singapore", "jpn": "シンガポール", "nld": "Singapore", "por": "Singapura", "rus": "Сингапур", "spa": "Singapur", "svk": "Singapur", "fin": "Singapore", "isr": "סינגפור"}}, "SX": {"currency": "ANG", "callingCode": "1721", "flag": "flag-sx", "name": {"common": "Sint Maarten", "deu": "Sint Maarten", "fra": "Saint-Martin", "ita": "Sint Maarten", "jpn": "シント・マールテン", "nld": "Sint Maarten", "por": "São Martinho", "rus": "Синт-Мартен", "spa": "Sint Maarten", "svk": "Svätý Martin", "fin": "Sint Maarten", "zho": "圣马丁岛", "isr": "סנט מארטן"}}, "SK": {"currency": "EUR", "callingCode": "421", "flag": "flag-sk", "name": {"common": "Slovakia", "deu": "Slowakei", "fra": "Slovaquie", "hrv": "Slovačka", "ita": "Slovacchia", "jpn": "スロバキア", "nld": "Slowakije", "por": "Eslováquia", "rus": "Словакия", "spa": "República Eslovaca", "svk": "Slovensko", "fin": "Slovakia", "zho": "斯洛伐克", "isr": "סלובקיה"}}, "SI": {"currency": "EUR", "callingCode": "386", "flag": "flag-si", "name": {"common": "Slovenia", "deu": "Slowenien", "fra": "Slovénie", "hrv": "Slovenija", "ita": "Slovenia", "jpn": "スロベニア", "nld": "Slovenië", "por": "Eslovénia", "rus": "Словения", "spa": "Eslovenia", "svk": "Slovinsko", "fin": "Slovenia", "zho": "斯洛文尼亚", "isr": "סלובניה"}}, "SB": {"currency": "SBD", "callingCode": "677", "flag": "flag-sb", "name": {"common": "Solomon Islands", "deu": "Salomonen", "fra": "Îles Salomon", "hrv": "Solomonski Otoci", "ita": "Isole Salomone", "jpn": "ソロモン諸島", "nld": "Salomonseilanden", "por": "Ilhas Salomão", "rus": "Соломоновы Острова", "spa": "Islas Salomón", "svk": "Salomonove ostrovy", "fin": "Salomonsaaret", "zho": "所罗门群岛", "isr": "איי שלמה"}}, "SO": {"currency": "SOS", "callingCode": "252", "flag": "flag-so", "name": {"common": "Somalia", "deu": "Somalia", "fra": "Somalie", "hrv": "Somalija", "ita": "Somalia", "jpn": "ソマリア", "nld": "Somalië", "por": "Somália", "rus": "Сомали", "spa": "Somalia", "svk": "Somálsko", "fin": "Somalia", "zho": "索马里", "isr": "סומליה"}}, "ZA": {"currency": "ZAR", "callingCode": "27", "flag": "flag-za", "name": {"common": "South Africa", "deu": "Republik Südafrika", "fra": "Afrique du Sud", "hrv": "Južnoafrička Republika", "ita": "Sud Africa", "jpn": "南アフリカ", "nld": "Zuid-Afrika", "por": "África do Sul", "rus": "Южно-Африканская Республика", "spa": "República de Sudáfrica", "svk": "Juhoafrická republika", "fin": "Etelä-Afrikka", "zho": "南非", "isr": "דרום אפריקה"}}, "GS": {"currency": "GBP", "callingCode": "500", "flag": "flag-gs", "name": {"common": "South Georgia", "deu": "Südgeorgien und die Südlichen Sandwichinseln", "fra": "Géorgie du Sud-et-les Îles Sandwich du Sud", "hrv": "Južna Georgija i otočje Južni Sandwich", "ita": "Georgia del Sud e Isole Sandwich Meridionali", "jpn": "サウスジョージア・サウスサンドウィッチ諸島", "nld": "Zuid-Georgia en Zuidelijke Sandwicheilanden", "por": "Ilhas Geórgia do Sul e Sandwich do Sul", "rus": "Южная Георгия и Южные Сандвичевы острова", "spa": "Islas Georgias del Sur y Sandwich del Sur", "svk": "Južná Georgia a Južné Sandwichove ostrovy", "fin": "Etelä-Georgia ja Eteläiset Sandwichsaaret", "zho": "南乔治亚", "isr": "ג׳ורג׳יה הדרומית ואיי סנדוויץ׳ הדרומיים"}}, "KR": {"currency": "KRW", "callingCode": "82", "flag": "flag-kr", "name": {"common": "South Korea", "deu": "Südkorea", "fra": "Corée du Sud", "hrv": "Južna Koreja", "ita": "Corea del Sud", "jpn": "大韓民国", "nld": "Zuid-Korea", "por": "Coreia do Sul", "rus": "Южная Корея", "spa": "Corea del Sur", "svk": "Južná Kórea", "fin": "Etelä-Korea", "zho": "韩国", "isr": "קוריאה הדרומית"}}, "SS": {"currency": "SSP", "callingCode": "211", "flag": "flag-ss", "name": {"common": "South Sudan", "deu": "Südsudan", "fra": "Soudan du Sud", "hrv": "Južni Sudan", "ita": "Sudan del sud", "jpn": "南スーダン", "nld": "Zuid-Soedan", "por": "Sudão do Sul", "rus": "Южный Судан", "spa": "Sudán del Sur", "svk": "Južný Sudán", "fin": "Etelä-Sudan", "zho": "南苏丹", "isr": "דרום סודן"}}, "ES": {"currency": "EUR", "callingCode": "34", "flag": "flag-es", "name": {"common": "Spain", "deu": "Spanien", "fra": "Espagne", "hrv": "Španjolska", "ita": "Spagna", "jpn": "スペイン", "nld": "Spanje", "por": "Espanha", "rus": "Испания", "spa": "España", "svk": "Španielsko", "fin": "Espanja", "zho": "西班牙", "isr": "ספרד"}}, "LK": {"currency": "LKR", "callingCode": "94", "flag": "flag-lk", "name": {"common": "Sri Lanka", "deu": "Sri Lanka", "fra": "Sri Lanka", "hrv": "Šri Lanka", "ita": "Sri Lanka", "jpn": "スリランカ", "nld": "Sri Lanka", "por": "Sri Lanka", "rus": "Шри-Ланка", "spa": "Sri Lanka", "svk": "Srí Lanka", "fin": "Sri Lanka", "zho": "斯里兰卡", "isr": "סרי לנקה"}}, "SD": {"currency": "SDG", "callingCode": "249", "flag": "flag-sd", "name": {"common": "Sudan", "deu": "Sudan", "fra": "Soudan", "hrv": "Sudan", "ita": "Sudan", "jpn": "スーダン", "nld": "Soedan", "por": "Sudão", "rus": "Судан", "spa": "Sudán", "svk": "Sudán", "fin": "Sudan", "zho": "苏丹", "isr": "סודן"}}, "SR": {"currency": "SRD", "callingCode": "597", "flag": "flag-sr", "name": {"common": "Suriname", "deu": "Suriname", "fra": "Surinam", "hrv": "Surinam", "ita": "Suriname", "jpn": "スリナム", "nld": "Suriname", "por": "Suriname", "rus": "Суринам", "spa": "Surinam", "svk": "Surinam", "fin": "Suriname", "zho": "苏里南", "isr": "סורינם"}}, "SJ": {"currency": "NOK", "callingCode": "4779", "flag": "flag-sj", "name": {"common": "Svalbard and Jan Mayen", "deu": "Spitzbergen", "fra": "Svalbard et Jan Mayen", "hrv": "Svalbard i Jan Mayen", "ita": "Svalbard e Jan Mayen", "jpn": "スヴァールバル諸島およびヤンマイエン島", "nld": "Svalbard en Jan Mayen", "por": "Ilhas Svalbard e Jan Mayen", "rus": "Шпицберген и Ян-Майен", "spa": "Islas Svalbard y Jan Mayen", "svk": "Svalbard a Jan Mayen", "fin": "Huippuvuoret", "zho": "斯瓦尔巴特", "isr": "סוולבארד ויאן מאיין"}}, "SZ": {"currency": "SZL", "callingCode": "268", "flag": "flag-sz", "name": {"common": "Swaziland", "deu": "Swasiland", "fra": "Swaziland", "hrv": "Svazi", "ita": "Swaziland", "jpn": "スワジランド", "nld": "Swaziland", "por": "Suazilândia", "rus": "Свазиленд", "spa": "Suazilandia", "svk": "Svazijsko", "fin": "Swazimaa", "zho": "斯威士兰", "isr": "סווזילנד"}}, "SE": {"currency": "SEK", "callingCode": "46", "flag": "flag-se", "name": {"common": "Sweden", "deu": "Schweden", "fra": "Suède", "hrv": "Švedska", "ita": "Svezia", "jpn": "スウェーデン", "nld": "Zweden", "por": "Suécia", "rus": "Швеция", "spa": "Suecia", "svk": "šveédsko", "fin": "Ruotsi", "zho": "瑞典", "isr": "שוודיה"}}, "CH": {"currency": "CHE", "callingCode": "41", "flag": "flag-ch", "name": {"common": "Switzerland", "deu": "Schweiz", "fra": "Suisse", "hrv": "Švicarska", "ita": "Svizzera", "jpn": "スイス", "nld": "Zwitserland", "por": "Suíça", "rus": "Швейцария", "spa": "Suiza", "svk": "Švajčiarsko", "fin": "Sveitsi", "zho": "瑞士", "isr": "שווייץ"}}, "SY": {"currency": "SYP", "callingCode": "963", "flag": "flag-sy", "name": {"common": "Syria", "deu": "Syrien", "fra": "Syrie", "hrv": "Sirija", "ita": "Siria", "jpn": "シリア・アラブ共和国", "nld": "Syrië", "por": "Síria", "rus": "Сирия", "spa": "Siria", "svk": "Sýria", "fin": "Syyria", "zho": "叙利亚", "isr": "סוריה"}}, "ST": {"currency": "STD", "callingCode": "239", "flag": "flag-st", "name": {"common": "São Tomé and Príncipe", "deu": "São Tomé und Príncipe", "fra": "São Tomé et Príncipe", "hrv": "Sveti Toma i Princip", "ita": "São Tomé e Príncipe", "jpn": "サントメ・プリンシペ", "nld": "Sao Tomé en Principe", "por": "São Tomé e Príncipe", "rus": "Сан-Томе и Принсипи", "spa": "Santo Tomé y Príncipe", "svk": "Svätý Tomáš a Princov ostrov", "fin": "São Téme ja Príncipe", "zho": "圣多美和普林西比", "isr": "סאו טומה ופרינסיפה"}}, "TW": {"currency": "TWD", "callingCode": "886", "flag": "flag-tw", "name": {"common": "Taiwan", "deu": "Taiwan", "fra": "Taïwan", "hrv": "Tajvan", "ita": "Taiwan", "jpn": "台湾(台湾省/中華民国)", "nld": "Taiwan", "por": "Ilha Formosa", "rus": "Тайвань", "spa": "Taiwán", "svk": "Taiwan", "fin": "Taiwan", "isr": "טייוואן"}}, "TJ": {"currency": "TJS", "callingCode": "992", "flag": "flag-tj", "name": {"common": "Tajikistan", "deu": "Tadschikistan", "fra": "Tadjikistan", "hrv": "Tađikistan", "ita": "Tagikistan", "jpn": "タジキスタン", "nld": "Tadzjikistan", "por": "Tajiquistão", "rus": "Таджикистан", "spa": "Tayikistán", "svk": "Tadžikistan", "fin": "Tadžikistan", "zho": "塔吉克斯坦", "isr": "טג׳יקיסטן"}}, "TZ": {"currency": "TZS", "callingCode": "255", "flag": "flag-tz", "name": {"common": "Tanzania", "deu": "Tansania", "fra": "Tanzanie", "hrv": "Tanzanija", "ita": "Tanzania", "jpn": "タンザニア", "nld": "Tanzania", "por": "Tanzânia", "rus": "Танзания", "spa": "Tanzania", "svk": "Tanzánia", "fin": "Tansania", "zho": "坦桑尼亚", "isr": "טנזניה"}}, "TH": {"currency": "THB", "callingCode": "66", "flag": "flag-th", "name": {"common": "Thailand", "deu": "Thailand", "fra": "Thaïlande", "hrv": "Tajland", "ita": "Tailandia", "jpn": "タイ", "nld": "Thailand", "por": "Tailândia", "rus": "Таиланд", "spa": "Tailandia", "svk": "Thajsko", "fin": "Thaimaa", "zho": "泰国", "isr": "תאילנד"}}, "TL": {"currency": "USD", "callingCode": "670", "flag": "flag-tl", "name": {"common": "Timor-Leste", "deu": "Timor-Leste", "fra": "Timor oriental", "hrv": "Istočni Timor", "ita": "Timor Est", "jpn": "東ティモール", "nld": "Oost-Timor", "por": "Timor-Leste", "rus": "Восточный Тимор", "spa": "Timor Oriental", "svk": "Východný Timor", "fin": "Itä-Timor", "zho": "东帝汶", "isr": "טימור לסטה"}}, "TG": {"currency": "XOF", "callingCode": "228", "flag": "flag-tg", "name": {"common": "Togo", "deu": "Togo", "fra": "Togo", "hrv": "Togo", "ita": "Togo", "jpn": "トーゴ", "nld": "Togo", "por": "Togo", "rus": "Того", "spa": "Togo", "svk": "Togo", "fin": "Togo", "zho": "多哥", "isr": "טוגו"}}, "TK": {"currency": "NZD", "callingCode": "690", "flag": "flag-tk", "name": {"common": "Tokelau", "deu": "Tokelau", "fra": "Tokelau", "hrv": "Tokelau", "ita": "Isole Tokelau", "jpn": "トケラウ", "nld": "Tokelau", "por": "Tokelau", "rus": "Токелау", "spa": "Islas Tokelau", "svk": "Tokelau", "fin": "Tokelau", "zho": "托克劳", "isr": "טוקלאו"}}, "TO": {"currency": "TOP", "callingCode": "676", "flag": "flag-to", "name": {"common": "Tonga", "deu": "Tonga", "fra": "Tonga", "hrv": "Tonga", "ita": "Tonga", "jpn": "トンガ", "nld": "Tonga", "por": "Tonga", "rus": "Тонга", "spa": "Tonga", "svk": "Tonga", "fin": "Tonga", "zho": "汤加", "isr": "טונגה"}}, "TT": {"currency": "TTD", "callingCode": "1868", "flag": "flag-tt", "name": {"common": "Trinidad and Tobago", "deu": "Trinidad und Tobago", "fra": "Trinité-et-Tobago", "hrv": "Trinidad i Tobago", "ita": "Trinidad e Tobago", "jpn": "トリニダード・トバゴ", "nld": "Trinidad en Tobago", "por": "Trinidade e Tobago", "rus": "Тринидад и Тобаго", "spa": "Trinidad y Tobago", "svk": "Trinidad a Tobago", "fin": "Trinidad ja Tobago", "zho": "特立尼达和多巴哥", "isr": "טרינידד וטובגו"}}, "TN": {"currency": "TND", "callingCode": "216", "flag": "flag-tn", "name": {"common": "Tunisia", "deu": "Tunesien", "fra": "Tunisie", "hrv": "Tunis", "ita": "Tunisia", "jpn": "チュニジア", "nld": "Tunesië", "por": "Tunísia", "rus": "Тунис", "spa": "Túnez", "svk": "Tunisko", "fin": "Tunisia", "zho": "突尼斯", "isr": "טוניסיה"}}, "TR": {"currency": "TRY", "callingCode": "90", "flag": "flag-tr", "name": {"common": "Turkey", "deu": "Türkei", "fra": "Turquie", "hrv": "Turska", "ita": "Turchia", "jpn": "トルコ", "nld": "Turkije", "por": "Turquia", "rus": "Турция", "spa": "Turquía", "svk": "Turecko", "fin": "Turkki", "zho": "土耳其", "isr": "טורקיה"}}, "TM": {"currency": "TMT", "callingCode": "993", "flag": "flag-tm", "name": {"common": "Turkmenistan", "deu": "Turkmenistan", "fra": "Turkménistan", "hrv": "Turkmenistan", "ita": "Turkmenistan", "jpn": "トルクメニスタン", "nld": "Turkmenistan", "por": "Turquemenistão", "rus": "Туркмения", "spa": "Turkmenistán", "svk": "Turkménsko", "fin": "Turkmenistan", "zho": "土库曼斯坦", "isr": "טורקמניסטן"}}, "TC": {"currency": "USD", "callingCode": "1649", "flag": "flag-tc", "name": {"common": "Turks and Caicos Islands", "deu": "Turks-und Caicosinseln", "fra": "Îles Turques-et-Caïques", "hrv": "Otoci Turks i Caicos", "ita": "Isole Turks e Caicos", "jpn": "タークス・カイコス諸島", "nld": "Turks-en Caicoseilanden", "por": "Ilhas Turks e Caicos", "rus": "Теркс и Кайкос", "spa": "Islas Turks y Caicos", "svk": "Turks a Caicos", "fin": "Turks-ja Caicossaaret", "zho": "特克斯和凯科斯群岛", "isr": "איי טורקס וקאיקוס"}}, "TV": {"currency": "AUD", "callingCode": "688", "flag": "flag-tv", "name": {"common": "Tuvalu", "deu": "Tuvalu", "fra": "Tuvalu", "hrv": "Tuvalu", "ita": "Tuvalu", "jpn": "ツバル", "nld": "Tuvalu", "por": "Tuvalu", "rus": "Тувалу", "spa": "Tuvalu", "svk": "Tuvalu", "fin": "Tuvalu", "zho": "图瓦卢", "isr": "טובאלו"}}, "UG": {"currency": "UGX", "callingCode": "256", "flag": "flag-ug", "name": {"common": "Uganda", "deu": "Uganda", "fra": "Ouganda", "hrv": "Uganda", "ita": "Uganda", "jpn": "ウガンダ", "nld": "Oeganda", "por": "Uganda", "rus": "Уганда", "spa": "Uganda", "svk": "Uganda", "fin": "Uganda", "zho": "乌干达", "isr": "אוגנדה"}}, "UA": {"currency": "UAH", "callingCode": "380", "flag": "flag-ua", "name": {"common": "Ukraine", "deu": "Ukraine", "fra": "Ukraine", "hrv": "Ukrajina", "ita": "Ucraina", "jpn": "ウクライナ", "nld": "Oekraïne", "por": "Ucrânia", "rus": "Украина", "spa": "Ucrania", "svk": "Ukrajina", "fin": "Ukraina", "zho": "乌克兰", "isr": "אוקראינה"}}, "AE": {"currency": "AED", "callingCode": "971", "flag": "flag-ae", "name": {"common": "United Arab Emirates", "deu": "Vereinigte Arabische Emirate", "fra": "Émirats arabes unis", "hrv": "Ujedinjeni Arapski Emirati", "ita": "Emirati Arabi Uniti", "jpn": "アラブ首長国連邦", "nld": "Verenigde Arabische Emiraten", "por": "Emirados Árabes Unidos", "rus": "Объединённые Арабские Эмираты", "spa": "Emiratos Árabes Unidos", "svk": "Spojené arabské emiráty", "fin": "Arabiemiraatit", "zho": "阿拉伯联合酋长国", "isr": "איחוד האמירויות הערביות"}}, "GB": {"currency": "GBP", "callingCode": "44", "flag": "flag-gb", "name": {"common": "United Kingdom", "deu": "Vereinigtes Königreich", "fra": "Royaume-Uni", "hrv": "Ujedinjeno Kraljevstvo", "ita": "Regno Unito", "jpn": "イギリス", "nld": "Verenigd Koninkrijk", "por": "Reino Unido", "rus": "Великобритания", "spa": "Reino Unido", "svk": "Veľká Británia (Spojené kráľovstvo)", "fin": "Yhdistynyt kuningaskunta", "zho": "英国", "isr": "הממלכה המאוחדת"}}, "US": {"currency": "USD", "callingCode": "1", "flag": "flag-us", "name": {"common": "United States", "deu": "Vereinigte Staaten von Amerika", "fra": "États-Unis", "hrv": "Sjedinjene Američke Države", "ita": "Stati Uniti d'America", "jpn": "アメリカ合衆国", "nld": "Verenigde Staten", "por": "Estados Unidos", "rus": "Соединённые Штаты Америки", "spa": "Estados Unidos", "svk": "Spojené štáty", "fin": "Yhdysvallat", "zho": "美国", "isr": "ארצות הברית"}}, "UM": {"currency": "USD", "flag": "flag-um", "name": {"common": "United States Minor Outlying Islands", "deu": "Kleinere Inselbesitzungen der Vereinigten Staaten", "fra": "Îles mineures éloignées des États-Unis", "hrv": "Mali udaljeni otoci SAD-a", "ita": "Isole minori esterne degli Stati Uniti d'America", "jpn": "合衆国領有小離島", "nld": "Kleine afgelegen eilanden van de Verenigde Staten", "por": "Ilhas Menores Distantes dos Estados Unidos", "rus": "Внешние малые острова США", "spa": "Islas Ultramarinas Menores de Estados Unidos", "svk": "Menšie odľahlé ostrovy USA", "fin": "Yhdysvaltain asumattomat saaret", "zho": "美国本土外小岛屿", "isr": "האיים המרוחקים הקטנים של ארה״ב"}}, "VI": {"currency": "USD", "callingCode": "1340", "flag": "flag-vi", "name": {"common": "United States Virgin Islands", "deu": "Amerikanische Jungferninseln", "fra": "Îles Vierges des États-Unis", "hrv": "Američki Djevičanski Otoci", "ita": "Isole Vergini americane", "jpn": "アメリカ領ヴァージン諸島", "nld": "Amerikaanse Maagdeneilanden", "por": "Ilhas Virgens dos Estados Unidos", "rus": "Виргинские Острова", "spa": "Islas Vírgenes de los Estados Unidos", "svk": "Americké Panenské ostrovy", "fin": "Neitsytsaaret", "zho": "美属维尔京群岛", "isr": "איי הבתולה של ארצות הברית"}}, "UY": {"currency": "UYI", "callingCode": "598", "flag": "flag-uy", "name": {"common": "Uruguay", "deu": "Uruguay", "fra": "Uruguay", "hrv": "Urugvaj", "ita": "Uruguay", "jpn": "ウルグアイ", "nld": "Uruguay", "por": "Uruguai", "rus": "Уругвай", "spa": "Uruguay", "svk": "Uruguaj", "fin": "Uruguay", "zho": "乌拉圭", "isr": "אורוגוואי"}}, "UZ": {"currency": "UZS", "callingCode": "998", "flag": "flag-uz", "name": {"common": "Uzbekistan", "deu": "Usbekistan", "fra": "Ouzbékistan", "hrv": "Uzbekistan", "ita": "Uzbekistan", "jpn": "ウズベキスタン", "nld": "Oezbekistan", "por": "Uzbequistão", "rus": "Узбекистан", "spa": "Uzbekistán", "svk": "Uzbekistan", "fin": "Uzbekistan", "zho": "乌兹别克斯坦", "isr": "אוזבקיסטן"}}, "VU": {"currency": "VUV", "callingCode": "678", "flag": "flag-vu", "name": {"common": "Vanuatu", "deu": "Vanuatu", "fra": "Vanuatu", "hrv": "Vanuatu", "ita": "Vanuatu", "jpn": "バヌアツ", "nld": "Vanuatu", "por": "Vanuatu", "rus": "Вануату", "spa": "Vanuatu", "svk": "Vanuatu", "fin": "Vanuatu", "zho": "瓦努阿图", "isr": "ונואטו"}}, "VA": {"currency": "EUR", "callingCode": "3906698", "flag": "flag-va", "name": {"common": "Vatican City", "deu": "Vatikanstadt", "fra": "Cité du Vatican", "hrv": "Vatikan", "ita": "Città del Vaticano", "jpn": "バチカン市国", "nld": "Vaticaanstad", "por": "Cidade do Vaticano", "rus": "Ватикан", "spa": "Ciudad del Vaticano", "svk": "Vatikán", "fin": "Vatikaani", "zho": "梵蒂冈", "isr": "הוותיקן"}}, "VE": {"currency": "VEF", "callingCode": "58", "flag": "flag-ve", "name": {"common": "Venezuela", "deu": "Venezuela", "fra": "Venezuela", "hrv": "Venezuela", "ita": "Venezuela", "jpn": "ベネズエラ・ボリバル共和国", "nld": "Venezuela", "por": "Venezuela", "rus": "Венесуэла", "spa": "Venezuela", "svk": "Venezuela", "fin": "Venezuela", "zho": "委内瑞拉", "isr": "ונצואלה"}}, "VN": {"currency": "VND", "callingCode": "84", "flag": "flag-vn", "name": {"common": "Vietnam", "deu": "Vietnam", "fra": "Viêt Nam", "hrv": "Vijetnam", "ita": "Vietnam", "jpn": "ベトナム", "nld": "Vietnam", "por": "Vietname", "rus": "Вьетнам", "spa": "Vietnam", "svk": "Vietnam", "fin": "Vietnam", "zho": "越南", "isr": "וייטנאם"}}, "WF": {"currency": "XPF", "callingCode": "681", "flag": "flag-wf", "name": {"common": "Wallis and Futuna", "deu": "Wallis und Futuna", "fra": "Wallis-et-Futuna", "hrv": "Wallis i Fortuna", "ita": "Wallis e Futuna", "jpn": "ウォリス・フツナ", "nld": "Wallis en Futuna", "por": "Wallis e Futuna", "rus": "Уоллис и Футуна", "spa": "Wallis y Futuna", "svk": "Wallis a Futuna", "fin": "Wallis ja Futuna", "zho": "瓦利斯和富图纳群岛", "isr": "איי ווליס ופוטונה"}}, "EH": {"currency": "MAD", "callingCode": "212", "flag": "flag-eh", "name": {"common": "Western Sahara", "deu": "Westsahara", "fra": "Sahara Occidental", "hrv": "Zapadna Sahara", "ita": "Sahara Occidentale", "jpn": "西サハラ", "nld": "Westelijke Sahara", "por": "Saara Ocidental", "rus": "Западная Сахара", "spa": "Sahara Occidental", "svk": "Západná Sahara", "fin": "Länsi-Sahara", "zho": "西撒哈拉", "isr": "סהרה המערבית"}}, "YE": {"currency": "YER", "callingCode": "967", "flag": "flag-ye", "name": {"common": "Yemen", "deu": "Jemen", "fra": "Yémen", "hrv": "Jemen", "ita": "Yemen", "jpn": "イエメン", "nld": "Jemen", "por": "Iémen", "rus": "Йемен", "spa": "Yemen", "svk": "Jemen", "fin": "Jemen", "zho": "也门", "isr": "תימן"}}, "ZM": {"currency": "ZMW", "callingCode": "260", "flag": "flag-zm", "name": {"common": "Zambia", "deu": "Sambia", "fra": "Zambie", "hrv": "Zambija", "ita": "Zambia", "jpn": "ザンビア", "nld": "Zambia", "por": "Zâmbia", "rus": "Замбия", "spa": "Zambia", "svk": "Zambia", "fin": "Sambia", "zho": "赞比亚", "isr": "זמביה"}}, "ZW": {"currency": "ZWL", "callingCode": "263", "flag": "flag-zw", "name": {"common": "Zimbabwe", "deu": "Simbabwe", "fra": "Zimbabwe", "hrv": "Zimbabve", "ita": "Zimbabwe", "jpn": "ジンバブエ", "nld": "Zimbabwe", "por": "Zimbabwe", "rus": "Зимбабве", "spa": "Zimbabue", "svk": "Zimbabwe", "fin": "Zimbabwe", "zho": "津巴布韦", "isr": "זימבבואה"}}, "AX": {"currency": "EUR", "callingCode": "358", "flag": "flag-ax", "name": {"common": "Åland Islands", "deu": "Åland", "fra": "Ahvenanmaa", "hrv": "Ålandski otoci", "ita": "Isole Aland", "jpn": "オーランド諸島", "nld": "Ålandeilanden", "por": "Alândia", "rus": "Аландские острова", "spa": "Alandia", "svk": "Alandy", "fin": "Ahvenanmaa", "zho": "奥兰群岛", "isr": "איי אולנד"}}} \ No newline at end of file diff --git a/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_lib_assets_data_countriesemoji.json b/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_lib_assets_data_countriesemoji.json deleted file mode 100644 index dcea665e..00000000 --- a/app/src/main/res/raw/node_modules_reactnativecountrypickermodal_lib_assets_data_countriesemoji.json +++ /dev/null @@ -1 +0,0 @@ -{"AF":{"currency":["AFN"],"callingCode":["93"],"region":"Asia","subregion":"Southern Asia","flag":"flag-af","name":{"common":"Afghanistan","ces":"Afghánistán","cym":"Affganistan","deu":"Afghanistan","fra":"Afghanistan","hrv":"Afganistan","ita":"Afghanistan","jpn":"アフガニスタン","nld":"Afghanistan","por":"Afeganistão","rus":"Афганистан","slk":"Afganistan","spa":"Afganistán","fin":"Afganistan","est":"Afganistan","zho":"阿富汗","pol":"Afganistan","urd":"افغانستان","kor":"아프가니스탄"}},"AL":{"currency":["ALL"],"callingCode":["355"],"region":"Europe","subregion":"Southern Europe","flag":"flag-al","name":{"common":"Albania","ces":"Albánie","cym":"Albania","deu":"Albanien","fra":"Albanie","hrv":"Albanija","ita":"Albania","jpn":"アルバニア","nld":"Albanië","por":"Albânia","rus":"Албания","slk":"Albánsko","spa":"Albania","fin":"Albania","est":"Albaania","zho":"阿尔巴尼亚","pol":"Albania","urd":"البانیا","kor":"알바니아"}},"DZ":{"currency":["DZD"],"callingCode":["213"],"region":"Africa","subregion":"Northern Africa","flag":"flag-dz","name":{"common":"Algeria","ces":"Alžírsko","cym":"Algeria","deu":"Algerien","fra":"Algérie","hrv":"Alžir","ita":"Algeria","jpn":"アルジェリア","nld":"Algerije","por":"Argélia","rus":"Алжир","slk":"Alžírsko","spa":"Argelia","fin":"Algeria","est":"Alžeeria","zho":"阿尔及利亚","pol":"Algieria","urd":"الجزائر","kor":"알제리"}},"AS":{"currency":["USD"],"callingCode":["1684"],"region":"Oceania","subregion":"Polynesia","flag":"flag-as","name":{"common":"American Samoa","ces":"Americká Samoa","deu":"Amerikanisch-Samoa","fra":"Samoa américaines","hrv":"Američka Samoa","ita":"Samoa Americane","jpn":"アメリカ領サモア","nld":"Amerikaans Samoa","por":"Samoa Americana","rus":"Американское Самоа","slk":"Americká Samoa","spa":"Samoa Americana","fin":"Amerikan Samoa","est":"Ameerika Samoa","zho":"美属萨摩亚","pol":"Samoa Amerykańskie","urd":"امریکی سمووا","kor":"아메리칸사모아"}},"AD":{"currency":["EUR"],"callingCode":["376"],"region":"Europe","subregion":"Southern Europe","flag":"flag-ad","name":{"common":"Andorra","ces":"Andorra","cym":"Andorra","deu":"Andorra","fra":"Andorre","hrv":"Andora","ita":"Andorra","jpn":"アンドラ","nld":"Andorra","por":"Andorra","rus":"Андорра","slk":"Andorra","spa":"Andorra","fin":"Andorra","est":"Andorra","zho":"安道尔","pol":"Andora","urd":"انڈورا","kor":"안도라"}},"AO":{"currency":["AOA"],"callingCode":["244"],"region":"Africa","subregion":"Middle Africa","flag":"flag-ao","name":{"common":"Angola","ces":"Angola","cym":"Angola","deu":"Angola","fra":"Angola","hrv":"Angola","ita":"Angola","jpn":"アンゴラ","nld":"Angola","por":"Angola","rus":"Ангола","slk":"Angola","spa":"Angola","fin":"Angola","est":"Angola","zho":"安哥拉","pol":"Angola","urd":"انگولہ","kor":"앙골라"}},"AI":{"currency":["XCD"],"callingCode":["1264"],"region":"Americas","subregion":"Caribbean","flag":"flag-ai","name":{"common":"Anguilla","ces":"Anguilla","deu":"Anguilla","fra":"Anguilla","hrv":"Angvila","ita":"Anguilla","jpn":"アンギラ","nld":"Anguilla","por":"Anguilla","rus":"Ангилья","slk":"Anguilla","spa":"Anguilla","fin":"Anguilla","est":"Anguilla","zho":"安圭拉","pol":"Anguilla","urd":"اینگویلا","kor":"앵귈라"}},"AQ":{"currency":[],"callingCode":[],"region":"Antarctic","subregion":"","flag":"flag-aq","name":{"common":"Antarctica","ces":"Antarktida","cym":"Yr Antarctig","deu":"Antarktis","fra":"Antarctique","hrv":"Antarktika","ita":"Antartide","jpn":"南極","nld":"Antarctica","por":"Antártida","rus":"Антарктида","slk":"Antarktída","spa":"Antártida","fin":"Etelämanner","est":"Antarktika","zho":"南极洲","pol":"Antarktyka","urd":"انٹارکٹکا","kor":"남극"}},"AG":{"currency":["XCD"],"callingCode":["1268"],"region":"Americas","subregion":"Caribbean","flag":"flag-ag","name":{"common":"Antigua and Barbuda","ces":"Antigua a Barbuda","cym":"Antigwa a Barbiwda","deu":"Antigua und Barbuda","fra":"Antigua-et-Barbuda","hrv":"Antigva i Barbuda","ita":"Antigua e Barbuda","jpn":"アンティグア・バーブーダ","nld":"Antigua en Barbuda","por":"Antígua e Barbuda","rus":"Антигуа и Барбуда","slk":"Antigua a Barbuda","spa":"Antigua y Barbuda","fin":"Antigua ja Barbuda","est":"Antigua ja Barbuda","zho":"安提瓜和巴布达","pol":"Antigua i Barbuda","urd":"اینٹیگوا و باربوڈا","kor":"앤티가 바부다"}},"AR":{"currency":["ARS"],"callingCode":["54"],"region":"Americas","subregion":"South America","flag":"flag-ar","name":{"common":"Argentina","ces":"Argentina","cym":"Ariannin","deu":"Argentinien","fra":"Argentine","hrv":"Argentina","ita":"Argentina","jpn":"アルゼンチン","nld":"Argentinië","por":"Argentina","rus":"Аргентина","slk":"Argentína","spa":"Argentina","fin":"Argentiina","est":"Argentina","zho":"阿根廷","pol":"Argentyna","urd":"ارجنٹائن","kor":"아르헨티나"}},"AM":{"currency":["AMD"],"callingCode":["374"],"region":"Asia","subregion":"Western Asia","flag":"flag-am","name":{"common":"Armenia","ces":"Arménie","cym":"Armenia","deu":"Armenien","fra":"Arménie","hrv":"Armenija","ita":"Armenia","jpn":"アルメニア","nld":"Armenië","por":"Arménia","rus":"Армения","slk":"Arménsko","spa":"Armenia","fin":"Armenia","est":"Armeenia","zho":"亚美尼亚","pol":"Armenia","urd":"آرمینیا","kor":"아르메니아"}},"AW":{"currency":["AWG"],"callingCode":["297"],"region":"Americas","subregion":"Caribbean","flag":"flag-aw","name":{"common":"Aruba","ces":"Aruba","deu":"Aruba","fra":"Aruba","hrv":"Aruba","ita":"Aruba","jpn":"アルバ","nld":"Aruba","por":"Aruba","rus":"Аруба","slk":"Aruba","spa":"Aruba","fin":"Aruba","est":"Aruba","zho":"阿鲁巴","pol":"Aruba","urd":"اروبا","kor":"아루바"}},"AU":{"currency":["AUD"],"callingCode":["61"],"region":"Oceania","subregion":"Australia and New Zealand","flag":"flag-au","name":{"common":"Australia","ces":"Austrálie","cym":"Awstralia","deu":"Australien","fra":"Australie","hrv":"Australija","ita":"Australia","jpn":"オーストラリア","nld":"Australië","por":"Austrália","rus":"Австралия","slk":"Austrália","spa":"Australia","fin":"Australia","est":"Austraalia","zho":"澳大利亚","pol":"Australia","urd":"آسٹریلیا","kor":"호주"}},"AT":{"currency":["EUR"],"callingCode":["43"],"region":"Europe","subregion":"Western Europe","flag":"flag-at","name":{"common":"Austria","ces":"Rakousko","cym":"Awstria","deu":"Österreich","fra":"Autriche","hrv":"Austrija","ita":"Austria","jpn":"オーストリア","nld":"Oostenrijk","por":"Áustria","rus":"Австрия","slk":"Rakúsko","spa":"Austria","fin":"Itävalta","est":"Austria","zho":"奥地利","pol":"Austria","urd":"آسٹریا","kor":"오스트리아"}},"AZ":{"currency":["AZN"],"callingCode":["994"],"region":"Asia","subregion":"Western Asia","flag":"flag-az","name":{"common":"Azerbaijan","ces":"Ázerbájdžán","cym":"Aserbaijan","deu":"Aserbaidschan","fra":"Azerbaïdjan","hrv":"Azerbajdžan","ita":"Azerbaijan","jpn":"アゼルバイジャン","nld":"Azerbeidzjan","por":"Azerbeijão","rus":"Азербайджан","slk":"AzerbajLJan","spa":"Azerbaiyán","fin":"Azerbaidzan","est":"Aserbaidžaan","zho":"阿塞拜疆","pol":"Azerbejdżan","urd":"آذربائیجان","kor":"아제르바이잔"}},"BS":{"currency":["BSD"],"callingCode":["1242"],"region":"Americas","subregion":"Caribbean","flag":"flag-bs","name":{"common":"Bahamas","ces":"Bahamy","cym":"Bahamas","deu":"Bahamas","fra":"Bahamas","hrv":"Bahami","ita":"Bahamas","jpn":"バハマ","nld":"Bahama’s","por":"Bahamas","rus":"Багамские Острова","slk":"Bahamy","spa":"Bahamas","fin":"Bahamasaaret","est":"Bahama","zho":"巴哈马","pol":"Bahamy","urd":"بہاماس","kor":"바하마"}},"BH":{"currency":["BHD"],"callingCode":["973"],"region":"Asia","subregion":"Western Asia","flag":"flag-bh","name":{"common":"Bahrain","ces":"Bahrajn","cym":"Bahrain","deu":"Bahrain","fra":"Bahreïn","hrv":"Bahrein","ita":"Bahrein","jpn":"バーレーン","nld":"Bahrein","por":"Bahrein","rus":"Бахрейн","slk":"Bahrajn","spa":"Bahrein","fin":"Bahrain","est":"Bahrein","zho":"巴林","pol":"Bahrajn","urd":"بحرین","kor":"바레인"}},"BD":{"currency":["BDT"],"callingCode":["880"],"region":"Asia","subregion":"Southern Asia","flag":"flag-bd","name":{"common":"Bangladesh","ces":"Bangladéš","cym":"Bangladesh","deu":"Bangladesch","fra":"Bangladesh","hrv":"Bangladeš","ita":"Bangladesh","jpn":"バングラデシュ","nld":"Bangladesh","por":"Bangladesh","rus":"Бангладеш","slk":"Bangladéš","spa":"Bangladesh","fin":"Bangladesh","est":"Bangladesh","zho":"孟加拉国","pol":"Bangladesz","urd":"بنگلہ دیش","kor":"방글라데시"}},"BB":{"currency":["BBD"],"callingCode":["1246"],"region":"Americas","subregion":"Caribbean","flag":"flag-bb","name":{"common":"Barbados","ces":"Barbados","cym":"Barbados","deu":"Barbados","fra":"Barbade","hrv":"Barbados","ita":"Barbados","jpn":"バルバドス","nld":"Barbados","por":"Barbados","rus":"Барбадос","slk":"Barbados","spa":"Barbados","fin":"Barbados","est":"Barbados","zho":"巴巴多斯","pol":"Barbados","urd":"بارباڈوس","kor":"바베이도스"}},"BY":{"currency":["BYN"],"callingCode":["375"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-by","name":{"common":"Belarus","ces":"Bělorusko","cym":"Belarws","deu":"Weißrussland","fra":"Biélorussie","hrv":"Bjelorusija","ita":"Bielorussia","jpn":"ベラルーシ","nld":"Wit-Rusland","por":"Bielorússia","rus":"Беларусь","slk":"Bielorusko","spa":"Bielorrusia","fin":"Valko-Venäjä","est":"Valgevene","zho":"白俄罗斯","pol":"Białoruś","urd":"بیلاروس","kor":"벨라루스"}},"BE":{"currency":["EUR"],"callingCode":["32"],"region":"Europe","subregion":"Western Europe","flag":"flag-be","name":{"common":"Belgium","ces":"Belgie","cym":"Gwlad Belg","deu":"Belgien","fra":"Belgique","hrv":"Belgija","ita":"Belgio","jpn":"ベルギー","nld":"België","por":"Bélgica","rus":"Бельгия","slk":"Belgicko","spa":"Bélgica","fin":"Belgia","est":"Belgia","zho":"比利时","pol":"Belgia","urd":"بلجئیم","kor":"벨기에"}},"BZ":{"currency":["BZD"],"callingCode":["501"],"region":"Americas","subregion":"Central America","flag":"flag-bz","name":{"common":"Belize","ces":"Belize","cym":"Belîs","deu":"Belize","fra":"Belize","hrv":"Belize","ita":"Belize","jpn":"ベリーズ","nld":"Belize","por":"Belize","rus":"Белиз","slk":"Belize","spa":"Belice","fin":"Belize","est":"Belize","zho":"伯利兹","pol":"Belize","urd":"بیلیز","kor":"벨리즈"}},"BJ":{"currency":["XOF"],"callingCode":["229"],"region":"Africa","subregion":"Western Africa","flag":"flag-bj","name":{"common":"Benin","ces":"Benin","cym":"Benin","deu":"Benin","fra":"Bénin","hrv":"Benin","ita":"Benin","jpn":"ベナン","nld":"Benin","por":"Benin","rus":"Бенин","slk":"Benin","spa":"Benín","fin":"Benin","est":"Benin","zho":"贝宁","pol":"Benin","urd":"بینن","kor":"베냉"}},"BM":{"currency":["BMD"],"callingCode":["1441"],"region":"Americas","subregion":"North America","flag":"flag-bm","name":{"common":"Bermuda","ces":"Bermudy","cym":"Bermiwda","deu":"Bermuda","fra":"Bermudes","hrv":"Bermudi","ita":"Bermuda","jpn":"バミューダ","nld":"Bermuda","por":"Bermudas","rus":"Бермудские Острова","slk":"Bermudy","spa":"Bermudas","fin":"Bermuda","est":"Bermuda","zho":"百慕大","pol":"Bermudy","urd":"برمودا","kor":"버뮤다"}},"BT":{"currency":["BTN","INR"],"callingCode":["975"],"region":"Asia","subregion":"Southern Asia","flag":"flag-bt","name":{"common":"Bhutan","ces":"Bhútán","cym":"Bhwtan","deu":"Bhutan","fra":"Bhoutan","hrv":"Butan","ita":"Bhutan","jpn":"ブータン","nld":"Bhutan","por":"Butão","rus":"Бутан","slk":"Bhután","spa":"Bután","fin":"Bhutan","est":"Bhutan","zho":"不丹","pol":"Bhutan","urd":"بھوٹان","kor":"부탄"}},"BO":{"currency":["BOB"],"callingCode":["591"],"region":"Americas","subregion":"South America","flag":"flag-bo","name":{"common":"Bolivia","ces":"Bolívie","cym":"Bolifia","deu":"Bolivien","fra":"Bolivie","hrv":"Bolivija","ita":"Bolivia","jpn":"ボリビア多民族国","nld":"Bolivia","por":"Bolívia","rus":"Боливия","slk":"Bolívia","spa":"Bolivia","fin":"Bolivia","est":"Boliivia","zho":"玻利维亚","pol":"Boliwia","urd":"بولیویا","kor":"볼리비아"}},"BA":{"currency":["BAM"],"callingCode":["387"],"region":"Europe","subregion":"Southern Europe","flag":"flag-ba","name":{"common":"Bosnia and Herzegovina","ces":"Bosna a Hercegovina","cym":"Bosnia a Hercegovina","deu":"Bosnien und Herzegowina","fra":"Bosnie-Herzégovine","hrv":"Bosna i Hercegovina","ita":"Bosnia ed Erzegovina","jpn":"ボスニア・ヘルツェゴビナ","nld":"Bosnië en Herzegovina","por":"Bósnia e Herzegovina","rus":"Босния и Герцеговина","slk":"Bosna a Hercegovina","spa":"Bosnia y Herzegovina","fin":"Bosnia ja Hertsegovina","est":"Bosnia ja Hertsegoviina","zho":"波斯尼亚和黑塞哥维那","pol":"Bośnia i Hercegowina","urd":"بوسنیا و ہرزیگووینا","kor":"보스니아 헤르체고비나"}},"BW":{"currency":["BWP"],"callingCode":["267"],"region":"Africa","subregion":"Southern Africa","flag":"flag-bw","name":{"common":"Botswana","ces":"Botswana","deu":"Botswana","fra":"Botswana","hrv":"Bocvana","ita":"Botswana","jpn":"ボツワナ","nld":"Botswana","por":"Botswana","rus":"Ботсвана","slk":"Botswana","spa":"Botswana","fin":"Botswana","est":"Botswana","zho":"博茨瓦纳","pol":"Botswana","urd":"بوٹسوانا","kor":"보츠와나"}},"BV":{"currency":["NOK"],"callingCode":[],"region":"Antarctic","subregion":"","flag":"flag-bv","name":{"common":"Bouvet Island","ces":"Bouvetův ostrov","deu":"Bouvetinsel","fra":"Île Bouvet","hrv":"Otok Bouvet","ita":"Isola Bouvet","jpn":"ブーベ島","nld":"Bouveteiland","por":"Ilha Bouvet","rus":"Остров Буве","slk":"Bouvetov ostrov","spa":"Isla Bouvet","fin":"Bouvet'nsaari","est":"Bouvet’ saar","zho":"布维岛","pol":"Wyspa Bouveta","urd":"جزیرہ بووہ","kor":"부베 섬"}},"BR":{"currency":["BRL"],"callingCode":["55"],"region":"Americas","subregion":"South America","flag":"flag-br","name":{"common":"Brazil","ces":"Brazílie","cym":"Brasil","deu":"Brasilien","fra":"Brésil","hrv":"Brazil","ita":"Brasile","jpn":"ブラジル","nld":"Brazilië","por":"Brasil","rus":"Бразилия","slk":"Brazília","spa":"Brasil","fin":"Brasilia","est":"Brasiilia","zho":"巴西","pol":"Brazylia","urd":"برازیل","kor":"브라질"}},"IO":{"currency":["USD"],"callingCode":["246"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-io","name":{"common":"British Indian Ocean Territory","ces":"Britské indickooceánské území","cym":"Tiriogaeth Brydeinig Cefnfor India","deu":"Britisches Territorium im Indischen Ozean","fra":"Territoire britannique de l'océan Indien","hrv":"Britanski Indijskooceanski teritorij","ita":"Territorio britannico dell'oceano indiano","jpn":"イギリス領インド洋地域","nld":"Britse Gebieden in de Indische Oceaan","por":"Território Britânico do Oceano Índico","rus":"Британская территория в Индийском океане","slk":"Britské indickooceánske územie","spa":"Territorio Británico del Océano Índico","fin":"Brittiläinen Intian valtameren alue","est":"Briti India ookeani ala","zho":"英属印度洋领地","pol":"Brytyjskie Terytorium Oceanu Indyjskiego","urd":"برطانوی بحرہند خطہ","kor":"인도"}},"VG":{"currency":["USD"],"callingCode":["1284"],"region":"Americas","subregion":"Caribbean","flag":"flag-vg","name":{"common":"British Virgin Islands","ces":"Britské Panenské ostrovy","deu":"Britische Jungferninseln","fra":"Îles Vierges britanniques","hrv":"Britanski Djevičanski Otoci","ita":"Isole Vergini Britanniche","jpn":"イギリス領ヴァージン諸島","nld":"Britse Maagdeneilanden","por":"Ilhas Virgens","rus":"Британские Виргинские острова","slk":"Panenské ostrovy","spa":"Islas Vírgenes del Reino Unido","fin":"Neitsytsaaret","est":"Briti Neitsisaared","zho":"英属维尔京群岛","pol":"Brytyjskie Wyspy Dziewicze","urd":"برطانوی جزائر ورجن","kor":"영국령 버진아일랜드"}},"BN":{"currency":["BND"],"callingCode":["673"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-bn","name":{"common":"Brunei","ces":"Brunej","cym":"Brunei","deu":"Brunei","fra":"Brunei","hrv":"Brunej","ita":"Brunei","jpn":"ブルネイ・ダルサラーム","nld":"Brunei","por":"Brunei","rus":"Бруней","slk":"Brunej","spa":"Brunei","fin":"Brunei","est":"Brunei","zho":"文莱","pol":"Brunei","urd":"برونائی","kor":"브루나이"}},"BG":{"currency":["BGN"],"callingCode":["359"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-bg","name":{"common":"Bulgaria","ces":"Bulharsko","cym":"Bwlgaria","deu":"Bulgarien","fra":"Bulgarie","hrv":"Bugarska","ita":"Bulgaria","jpn":"ブルガリア","nld":"Bulgarije","por":"Bulgária","rus":"Болгария","slk":"Bulharsko","spa":"Bulgaria","fin":"Bulgaria","est":"Bulgaaria","zho":"保加利亚","pol":"Bułgaria","urd":"بلغاریہ","kor":"불가리아"}},"BF":{"currency":["XOF"],"callingCode":["226"],"region":"Africa","subregion":"Western Africa","flag":"flag-bf","name":{"common":"Burkina Faso","ces":"Burkina Faso","cym":"Bwrcina Ffaso","deu":"Burkina Faso","fra":"Burkina Faso","hrv":"Burkina Faso","ita":"Burkina Faso","jpn":"ブルキナファソ","nld":"Burkina Faso","por":"Burkina Faso","rus":"Буркина-Фасо","slk":"Burkina Faso","spa":"Burkina Faso","fin":"Burkina Faso","est":"Burkina Faso","zho":"布基纳法索","pol":"Burkina Faso","urd":"برکینا فاسو","kor":"부르키나파소"}},"BI":{"currency":["BIF"],"callingCode":["257"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-bi","name":{"common":"Burundi","ces":"Burundi","cym":"Bwrwndi","deu":"Burundi","fra":"Burundi","hrv":"Burundi","ita":"Burundi","jpn":"ブルンジ","nld":"Burundi","por":"Burundi","rus":"Бурунди","slk":"Burundi","spa":"Burundi","fin":"Burundi","est":"Burundi","zho":"布隆迪","pol":"Burundi","urd":"برونڈی","kor":"부룬디"}},"KH":{"currency":["KHR"],"callingCode":["855"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-kh","name":{"common":"Cambodia","ces":"Kambodža","cym":"Cambodia","deu":"Kambodscha","fra":"Cambodge","hrv":"Kambodža","ita":"Cambogia","jpn":"カンボジア","nld":"Cambodja","por":"Camboja","rus":"Камбоджа","slk":"Kambodža","spa":"Camboya","fin":"Kambodža","est":"Kambodža","zho":"柬埔寨","pol":"Kambodża","urd":"کمبوڈیا","kor":"캄보디아"}},"CM":{"currency":["XAF"],"callingCode":["237"],"region":"Africa","subregion":"Middle Africa","flag":"flag-cm","name":{"common":"Cameroon","ces":"Kamerun","cym":"Camerŵn","deu":"Kamerun","fra":"Cameroun","hrv":"Kamerun","ita":"Camerun","jpn":"カメルーン","nld":"Kameroen","por":"Camarões","rus":"Камерун","slk":"Kamerun","spa":"Camerún","fin":"Kamerun","est":"Kamerun","zho":"喀麦隆","pol":"WybrzeŻe Kości Słoniowej","urd":"کیمرون","kor":"카메룬"}},"CA":{"currency":["CAD"],"callingCode":["1"],"region":"Americas","subregion":"North America","flag":"flag-ca","name":{"common":"Canada","ces":"Kanada","cym":"Canada","deu":"Kanada","fra":"Canada","hrv":"Kanada","ita":"Canada","jpn":"カナダ","nld":"Canada","por":"Canadá","rus":"Канада","slk":"Kanada","spa":"Canadá","fin":"Kanada","est":"Kanada","zho":"加拿大","pol":"Kanada","urd":"کینیڈا","kor":"캐나다"}},"CV":{"currency":["CVE"],"callingCode":["238"],"region":"Africa","subregion":"Western Africa","flag":"flag-cv","name":{"common":"Cape Verde","ces":"Kapverdy","cym":"Penrhyn Verde","deu":"Kap Verde","fra":"Îles du Cap-Vert","hrv":"Zelenortska Republika","ita":"Capo Verde","jpn":"カーボベルデ","nld":"Kaapverdië","por":"Cabo Verde","rus":"Кабо-Верде","slk":"Kapverdy","spa":"Cabo Verde","fin":"Kap Verde","est":"Roheneemesaared","zho":"佛得角","pol":"Republika Zielonego Przylądka","urd":"کیپ ورڈی","kor":"카보베르데"}},"BQ":{"currency":["USD"],"callingCode":["599"],"region":"Americas","subregion":"Caribbean","flag":"flag-bq","name":{"common":"Caribbean Netherlands","ces":"Karibské Nizozemsko","deu":"Karibische Niederlande","fra":"Pays-Bas caribéens","hrv":"Bonaire, Sint Eustatius i Saba","ita":"Paesi Bassi caraibici","jpn":"ボネール、シント・ユースタティウスおよびサバ","nld":"Caribisch Nederland","por":"Países Baixos Caribenhos","rus":"Карибские Нидерланды","slk":"Bonaire, Sint Eustatius a Saba","spa":"Caribe Neerlandés","fin":"Bonaire, Sint Eustatius ja Saba","est":"Bonaire, Sint Eustatius ja Saba","zho":"荷蘭加勒比區","pol":"Antyle Holenderskie","urd":"کیریبین نیدرلینڈز","kor":"카리브 네덜란드"}},"KY":{"currency":["KYD"],"callingCode":["1345"],"region":"Americas","subregion":"Caribbean","flag":"flag-ky","name":{"common":"Cayman Islands","ces":"Kajmanské ostrovy","cym":"Ynysoedd Cayman","deu":"Kaimaninseln","fra":"Îles Caïmans","hrv":"Kajmanski otoci","ita":"Isole Cayman","jpn":"ケイマン諸島","nld":"Caymaneilanden","por":"Ilhas Caimão","rus":"Каймановы острова","slk":"Kajmanie ostrovy","spa":"Islas Caimán","fin":"Caymansaaret","est":"Kaimanisaared","zho":"开曼群岛","pol":"Kajmany","urd":"جزائر کیمین","kor":"케이맨 제도"}},"CF":{"currency":["XAF"],"callingCode":["236"],"region":"Africa","subregion":"Middle Africa","flag":"flag-cf","name":{"common":"Central African Republic","ces":"Středoafrická republika","cym":"Gweriniaeth Canolbarth Affrica","deu":"Zentralafrikanische Republik","fra":"République centrafricaine","hrv":"Srednjoafrička Republika","ita":"Repubblica Centrafricana","jpn":"中央アフリカ共和国","nld":"Centraal-Afrikaanse Republiek","por":"República Centro-Africana","rus":"Центральноафриканская Республика","slk":"Stredoafrická republika","spa":"República Centroafricana","fin":"Keski-Afrikan tasavalta","est":"Kesk-Aafrika Vabariik","zho":"中非共和国","pol":"Republika Środkowoafrykańska","urd":"وسطی افریقی جمہوریہ","kor":"중앙아프리카 공화국"}},"TD":{"currency":["XAF"],"callingCode":["235"],"region":"Africa","subregion":"Middle Africa","flag":"flag-td","name":{"common":"Chad","ces":"Čad","cym":"Tsiad","deu":"Tschad","fra":"Tchad","hrv":"Čad","ita":"Ciad","jpn":"チャド","nld":"Tsjaad","por":"Chade","rus":"Чад","slk":"Čad","spa":"Chad","fin":"Tšad","est":"Tšaad","zho":"乍得","pol":"Czad","urd":"چاڈ","kor":"차드"}},"CL":{"currency":["CLP"],"callingCode":["56"],"region":"Americas","subregion":"South America","flag":"flag-cl","name":{"common":"Chile","ces":"Chile","cym":"Chile","deu":"Chile","fra":"Chili","hrv":"Čile","ita":"Cile","jpn":"チリ","nld":"Chili","por":"Chile","rus":"Чили","slk":"Čile","spa":"Chile","fin":"Chile","est":"Tšiili","zho":"智利","pol":"Chile","urd":"چلی","kor":"칠레"}},"CN":{"currency":["CNY"],"callingCode":["86"],"region":"Asia","subregion":"Eastern Asia","flag":"flag-cn","name":{"common":"China","ces":"Čína","cym":"Tsieina","deu":"China","fra":"Chine","hrv":"Kina","ita":"Cina","jpn":"中国","nld":"China","por":"China","rus":"Китай","slk":"Čína","spa":"China","fin":"Kiina","est":"Hiina","pol":"Chiny","urd":"چین","kor":"중국"}},"CX":{"currency":["AUD"],"callingCode":["61"],"region":"Oceania","subregion":"Australia and New Zealand","flag":"flag-cx","name":{"common":"Christmas Island","ces":"Vánoční ostrov","cym":"Ynys y Nadolig","deu":"Weihnachtsinsel","fra":"Île Christmas","hrv":"Božićni otok","ita":"Isola di Natale","jpn":"クリスマス島","nld":"Christmaseiland","por":"Ilha do Natal","rus":"Остров Рождества","slk":"Vianočnú ostrov","spa":"Isla de Navidad","fin":"Joulusaari","est":"Jõulusaar","zho":"圣诞岛","pol":"Wyspa Bożego Narodzenia","urd":"جزیرہ کرسمس","kor":"크리스마스 섬"}},"CC":{"currency":["AUD"],"callingCode":["61"],"region":"Oceania","subregion":"Australia and New Zealand","flag":"flag-cc","name":{"common":"Cocos (Keeling) Islands","ces":"Kokosové ostrovy","cym":"Ynysoedd Cocos","deu":"Kokosinseln","fra":"Îles Cocos","hrv":"Kokosovi Otoci","ita":"Isole Cocos e Keeling","jpn":"ココス(キーリング)諸島","nld":"Cocoseilanden","por":"Ilhas Cocos (Keeling)","rus":"Кокосовые острова","slk":"Kokosové ostrovy","spa":"Islas Cocos o Islas Keeling","fin":"Kookossaaret","est":"Kookossaared","zho":"科科斯","pol":"Wyspy Kokosowe","urd":"جزائر کوکوس","kor":"코코스 제도"}},"CO":{"currency":["COP"],"callingCode":["57"],"region":"Americas","subregion":"South America","flag":"flag-co","name":{"common":"Colombia","ces":"Kolumbie","cym":"Colombia","deu":"Kolumbien","fra":"Colombie","hrv":"Kolumbija","ita":"Colombia","jpn":"コロンビア","nld":"Colombia","por":"Colômbia","rus":"Колумбия","slk":"Kolumbia","spa":"Colombia","fin":"Kolumbia","est":"Colombia","zho":"哥伦比亚","pol":"Kolumbia","urd":"کولمبیا","kor":"콜롬비아"}},"KM":{"currency":["KMF"],"callingCode":["269"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-km","name":{"common":"Comoros","ces":"Komory","cym":"Y Comoros","deu":"Komoren","fra":"Comores","hrv":"Komori","ita":"Comore","jpn":"コモロ","nld":"Comoren","por":"Comores","rus":"Коморы","slk":"Komory","spa":"Comoras","fin":"Komorit","est":"Komoorid","zho":"科摩罗","pol":"Komory","urd":"القمری","kor":"코모로"}},"CK":{"currency":["NZD","CKD"],"callingCode":["682"],"region":"Oceania","subregion":"Polynesia","flag":"flag-ck","name":{"common":"Cook Islands","ces":"Cookovy ostrovy","cym":"Ynysoedd Cook","deu":"Cookinseln","fra":"Îles Cook","hrv":"Cookovo Otočje","ita":"Isole Cook","jpn":"クック諸島","nld":"Cookeilanden","por":"Ilhas Cook","rus":"Острова Кука","slk":"Cookove ostrovy","spa":"Islas Cook","fin":"Cookinsaaret","est":"Cooki saared","zho":"库克群岛","pol":"Wyspy Cooka","urd":"جزائر کک","kor":"쿡 제도"}},"CR":{"currency":["CRC"],"callingCode":["506"],"region":"Americas","subregion":"Central America","flag":"flag-cr","name":{"common":"Costa Rica","ces":"Kostarika","cym":"Costa Rica","deu":"Costa Rica","fra":"Costa Rica","hrv":"Kostarika","ita":"Costa Rica","jpn":"コスタリカ","nld":"Costa Rica","por":"Costa Rica","rus":"Коста-Рика","slk":"Kostarika","spa":"Costa Rica","fin":"Costa Rica","est":"Costa Rica","zho":"哥斯达黎加","pol":"Kostaryka","urd":"کوسٹاریکا","kor":"코스타리카"}},"HR":{"currency":["HRK"],"callingCode":["385"],"region":"Europe","subregion":"Southern Europe","flag":"flag-hr","name":{"common":"Croatia","ces":"Chorvatsko","cym":"Croatia","deu":"Kroatien","fra":"Croatie","hrv":"Hrvatska","ita":"Croazia","jpn":"クロアチア","nld":"Kroatië","por":"Croácia","rus":"Хорватия","slk":"Chorvátsko","spa":"Croacia","fin":"Kroatia","est":"Horvaatia","zho":"克罗地亚","pol":"Chorwacja","urd":"کرویئشا","kor":"크로아티아"}},"CU":{"currency":["CUC","CUP"],"callingCode":["53"],"region":"Americas","subregion":"Caribbean","flag":"flag-cu","name":{"common":"Cuba","ces":"Kuba","cym":"Ciwba","deu":"Kuba","fra":"Cuba","hrv":"Kuba","ita":"Cuba","jpn":"キューバ","nld":"Cuba","por":"Cuba","rus":"Куба","slk":"Kuba","spa":"Cuba","fin":"Kuuba","est":"Kuuba","zho":"古巴","pol":"Kuba","urd":"کیوبا","kor":"쿠바"}},"CW":{"currency":["ANG"],"callingCode":["5999"],"region":"Americas","subregion":"Caribbean","flag":"flag-cw","name":{"common":"Curaçao","ces":"Curaçao","deu":"Curaçao","fra":"Curaçao","nld":"Curaçao","por":"ilha da Curação","rus":"Кюрасао","slk":"Curacao","spa":"Curazao","fin":"Curaçao","est":"Curaçao","zho":"库拉索","pol":"Curaçao","urd":"کیوراساؤ","kor":"퀴라소"}},"CY":{"currency":["EUR"],"callingCode":["357"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-cy","name":{"common":"Cyprus","ces":"Kypr","cym":"Cyprus","deu":"Zypern","fra":"Chypre","hrv":"Cipar","ita":"Cipro","jpn":"キプロス","nld":"Cyprus","por":"Chipre","rus":"Кипр","slk":"Cyprus","spa":"Chipre","fin":"Kypros","est":"Küpros","zho":"塞浦路斯","pol":"Cypr","urd":"قبرص","kor":"키프로스"}},"CZ":{"currency":["CZK"],"callingCode":["420"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-cz","name":{"common":"Czechia","ces":"Česko","cym":"Y Weriniaeth Tsiec","deu":"Tschechien","fra":"Tchéquie","hrv":"Češka","ita":"Cechia","jpn":"チェコ","nld":"Tsjechië","por":"Chéquia","rus":"Чехия","slk":"Česko","spa":"Chequia","fin":"Tšekki","est":"Tšehhi","zho":"捷克","pol":"Czechy","urd":"چيک","kor":"체코"}},"CD":{"currency":["CDF"],"callingCode":["243"],"region":"Africa","subregion":"Middle Africa","flag":"flag-cd","name":{"common":"DR Congo","ces":"DR Kongo","cym":"Gweriniaeth Ddemocrataidd Congo","deu":"Kongo (Dem. Rep.)","fra":"Congo (Rép. dém.)","hrv":"Kongo, Demokratska Republika","ita":"Congo (Rep. Dem.)","jpn":"コンゴ民主共和国","nld":"Congo (DRC)","por":"República Democrática do Congo","rus":"Демократическая Республика Конго","slk":"Kongo","spa":"Congo (Rep. Dem.)","fin":"Kongon demokraattinen tasavalta","est":"Kongo DV","zho":"民主刚果","pol":"Demokratyczna Republika Konga","urd":"\nکانگو","kor":"콩고 민주 공화국"}},"DK":{"currency":["DKK"],"callingCode":["45"],"region":"Europe","subregion":"Northern Europe","flag":"flag-dk","name":{"common":"Denmark","ces":"Dánsko","cym":"Denmarc","deu":"Dänemark","fra":"Danemark","hrv":"Danska","ita":"Danimarca","jpn":"デンマーク","nld":"Denemarken","por":"Dinamarca","rus":"Дания","slk":"Dánsko","spa":"Dinamarca","fin":"Tanska","est":"Taani","zho":"丹麦","pol":"Dania","urd":"ڈنمارک","kor":"덴마크"}},"DJ":{"currency":["DJF"],"callingCode":["253"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-dj","name":{"common":"Djibouti","ces":"Džibutsko","cym":"Jibwti","deu":"Dschibuti","fra":"Djibouti","hrv":"Džibuti","ita":"Gibuti","jpn":"ジブチ","nld":"Djibouti","por":"Djibouti","rus":"Джибути","slk":"Džibutsko","spa":"Djibouti","fin":"Dijibouti","est":"Djibouti","zho":"吉布提","pol":"Dżibuti","urd":"جبوتی","kor":"지부티"}},"DM":{"currency":["XCD"],"callingCode":["1767"],"region":"Americas","subregion":"Caribbean","flag":"flag-dm","name":{"common":"Dominica","ces":"Dominika","cym":"Dominica","deu":"Dominica","fra":"Dominique","hrv":"Dominika","ita":"Dominica","jpn":"ドミニカ国","nld":"Dominica","por":"Dominica","rus":"Доминика","slk":"Dominika","spa":"Dominica","fin":"Dominica","est":"Dominica","zho":"多米尼加","pol":"Dominika","urd":"ڈومینیکا","kor":"도미니카 공화국"}},"DO":{"currency":["DOP"],"callingCode":["1809","1829","1849"],"region":"Americas","subregion":"Caribbean","flag":"flag-do","name":{"common":"Dominican Republic","ces":"Dominikánská republika","cym":"Gweriniaeth Dominica","deu":"Dominikanische Republik","fra":"République dominicaine","hrv":"Dominikanska Republika","ita":"Repubblica Dominicana","jpn":"ドミニカ共和国","nld":"Dominicaanse Republiek","por":"República Dominicana","rus":"Доминиканская Республика","slk":"Dominikánska republika","spa":"República Dominicana","fin":"Dominikaaninen tasavalta","est":"Dominikaani Vabariik","zho":"多明尼加","pol":"Dominikana","urd":"ڈومینیکن","kor":"도미니카 공화국"}},"EC":{"currency":["USD"],"callingCode":["593"],"region":"Americas","subregion":"South America","flag":"flag-ec","name":{"common":"Ecuador","ces":"Ekvádor","cym":"Ecwador","deu":"Ecuador","fra":"Équateur","hrv":"Ekvador","ita":"Ecuador","jpn":"エクアドル","nld":"Ecuador","por":"Equador","rus":"Эквадор","slk":"Ekvádor","spa":"Ecuador","fin":"Ecuador","est":"Ecuador","zho":"厄瓜多尔","pol":"Ekwador","urd":"ایکواڈور","kor":"에콰도르"}},"EG":{"currency":["EGP"],"callingCode":["20"],"region":"Africa","subregion":"Northern Africa","flag":"flag-eg","name":{"common":"Egypt","ces":"Egypt","cym":"Yr Aifft","deu":"Ägypten","fra":"Égypte","hrv":"Egipat","ita":"Egitto","jpn":"エジプト","nld":"Egypte","por":"Egito","rus":"Египет","slk":"Egypt","spa":"Egipto","fin":"Egypti","est":"Egiptus","zho":"埃及","pol":"Egipt","urd":"مصر","kor":"이집트"}},"SV":{"currency":["SVC","USD"],"callingCode":["503"],"region":"Americas","subregion":"Central America","flag":"flag-sv","name":{"common":"El Salvador","ces":"Salvador","cym":"El Salfador","deu":"El Salvador","fra":"Salvador","hrv":"Salvador","ita":"El Salvador","jpn":"エルサルバドル","nld":"El Salvador","por":"El Salvador","rus":"Сальвадор","slk":"Salvádor","spa":"El Salvador","fin":"El Salvador","est":"El Salvador","zho":"萨尔瓦多","pol":"Salwador","urd":"ایل سیلواڈور","kor":"엘살바도르"}},"GQ":{"currency":["XAF"],"callingCode":["240"],"region":"Africa","subregion":"Middle Africa","flag":"flag-gq","name":{"common":"Equatorial Guinea","ces":"Rovníková Guinea","cym":"Gini Gyhydeddol","deu":"Äquatorialguinea","fra":"Guinée équatoriale","hrv":"Ekvatorijalna Gvineja","ita":"Guinea Equatoriale","jpn":"赤道ギニア","nld":"Equatoriaal-Guinea","por":"Guiné Equatorial","rus":"Экваториальная Гвинея","slk":"Rovníková Guinea","spa":"Guinea Ecuatorial","fin":"Päiväntasaajan Guinea","est":"Ekvatoriaal-Guinea","zho":"赤道几内亚","pol":"Gwinea Równikowa","urd":"استوائی گنی","kor":"적도 기니"}},"ER":{"currency":["ERN"],"callingCode":["291"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-er","name":{"common":"Eritrea","ces":"Eritrea","cym":"Eritrea","deu":"Eritrea","fra":"Érythrée","hrv":"Eritreja","ita":"Eritrea","jpn":"エリトリア","nld":"Eritrea","por":"Eritreia","rus":"Эритрея","slk":"Eritrea","spa":"Eritrea","fin":"Eritrea","est":"Eritrea","zho":"厄立特里亚","pol":"Erytrea","urd":"ارتریا","kor":"에리트레아"}},"EE":{"currency":["EUR"],"callingCode":["372"],"region":"Europe","subregion":"Northern Europe","flag":"flag-ee","name":{"common":"Estonia","ces":"Estonsko","cym":"Estonia","deu":"Estland","fra":"Estonie","hrv":"Estonija","ita":"Estonia","jpn":"エストニア","nld":"Estland","por":"Estónia","rus":"Эстония","slk":"Estónsko","spa":"Estonia","fin":"Viro","est":"Eesti","zho":"爱沙尼亚","pol":"Estonia","urd":"اسٹونیا","kor":"에스토니아"}},"SZ":{"currency":["SZL"],"callingCode":["268"],"region":"Africa","subregion":"Southern Africa","flag":"flag-sz","name":{"common":"Eswatini","ces":"Svazijsko","deu":"Swasiland","fra":"Swaziland","hrv":"Svazi","ita":"Swaziland","jpn":"スワジランド","nld":"Swaziland","por":"Suazilândia","rus":"Свазиленд","slk":"Svazijsko","spa":"Suazilandia","fin":"Swazimaa","est":"Svaasimaa","pol":"Suazi","zho":"斯威士兰","urd":"سوازی لینڈ","kor":"에스와티니"}},"ET":{"currency":["ETB"],"callingCode":["251"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-et","name":{"common":"Ethiopia","ces":"Etiopie","cym":"Ethiopia","deu":"Äthiopien","fra":"Éthiopie","hrv":"Etiopija","ita":"Etiopia","jpn":"エチオピア","nld":"Ethiopië","por":"Etiópia","rus":"Эфиопия","slk":"Etiópia","spa":"Etiopía","fin":"Etiopia","est":"Etioopia","zho":"埃塞俄比亚","pol":"Etiopia","urd":"ایتھوپیا","kor":"에티오피아"}},"FK":{"currency":["FKP"],"callingCode":["500"],"region":"Americas","subregion":"South America","flag":"flag-fk","name":{"common":"Falkland Islands","ces":"Falklandy","deu":"Falklandinseln","fra":"Îles Malouines","hrv":"Falklandski Otoci","ita":"Isole Falkland o Isole Malvine","jpn":"フォークランド(マルビナス)諸島","nld":"Falklandeilanden","por":"Ilhas Malvinas","rus":"Фолклендские острова","slk":"Falklandy","spa":"Islas Malvinas","fin":"Falkandinsaaret","est":"Falklandi saared","zho":"福克兰群岛","pol":"Falklandy","urd":"جزائر فاکلینڈ","kor":"포클랜드 제도"}},"FO":{"currency":["DKK"],"callingCode":["298"],"region":"Europe","subregion":"Northern Europe","flag":"flag-fo","name":{"common":"Faroe Islands","ces":"Faerské ostrovy","deu":"Färöer-Inseln","fra":"Îles Féroé","hrv":"Farski Otoci","ita":"Isole Far Oer","jpn":"フェロー諸島","nld":"Faeröer","por":"Ilhas Faroé","rus":"Фарерские острова","slk":"Faerské ostrovy","spa":"Islas Faroe","fin":"Färsaaret","est":"Fääri saared","zho":"法罗群岛","pol":"Wyspy Owcze","urd":"جزائر فارو","kor":"페로 제도"}},"FJ":{"currency":["FJD"],"callingCode":["679"],"region":"Oceania","subregion":"Melanesia","flag":"flag-fj","name":{"common":"Fiji","ces":"Fidži","deu":"Fidschi","fra":"Fidji","hrv":"Fiđi","ita":"Figi","jpn":"フィジー","nld":"Fiji","por":"Fiji","rus":"Фиджи","slk":"Fidži","spa":"Fiyi","fin":"Fidži","est":"Fidži","zho":"斐济","pol":"Fidżi","urd":"فجی","kor":"피지"}},"FI":{"currency":["EUR"],"callingCode":["358"],"region":"Europe","subregion":"Northern Europe","flag":"flag-fi","name":{"common":"Finland","ces":"Finsko","deu":"Finnland","fra":"Finlande","hrv":"Finska","ita":"Finlandia","jpn":"フィンランド","nld":"Finland","por":"Finlândia","rus":"Финляндия","slk":"Fínsko","spa":"Finlandia","fin":"Suomi","est":"Soome","zho":"芬兰","pol":"Finlandia","urd":"فن لینڈ","kor":"핀란드"}},"FR":{"currency":["EUR"],"callingCode":["33"],"region":"Europe","subregion":"Western Europe","flag":"flag-fr","name":{"common":"France","ces":"Francie","deu":"Frankreich","fra":"France","hrv":"Francuska","ita":"Francia","jpn":"フランス","nld":"Frankrijk","por":"França","rus":"Франция","slk":"Francúzsko","spa":"Francia","fin":"Ranska","est":"Prantsusmaa","zho":"法国","pol":"Francja","urd":"فرانس","kor":"프랑스"}},"GF":{"currency":["EUR"],"callingCode":["594"],"region":"Americas","subregion":"South America","flag":"flag-gf","name":{"common":"French Guiana","ces":"Francouzská Guyana","deu":"Französisch-Guayana","fra":"Guyane","hrv":"Francuska Gvajana","ita":"Guyana francese","jpn":"フランス領ギアナ","nld":"Frans-Guyana","por":"Guiana Francesa","rus":"Французская Гвиана","slk":"Guyana","spa":"Guayana Francesa","fin":"Ranskan Guayana","est":"Prantsuse Guajaana","zho":"法属圭亚那","pol":"Gujana Francuska","urd":"فرانسیسی گیانا","kor":"프랑스령 기아나"}},"PF":{"currency":["XPF"],"callingCode":["689"],"region":"Oceania","subregion":"Polynesia","flag":"flag-pf","name":{"common":"French Polynesia","ces":"Francouzská Polynésie","deu":"Französisch-Polynesien","fra":"Polynésie française","hrv":"Francuska Polinezija","ita":"Polinesia Francese","jpn":"フランス領ポリネシア","nld":"Frans-Polynesië","por":"Polinésia Francesa","rus":"Французская Полинезия","slk":"Francúzska Polynézia","spa":"Polinesia Francesa","fin":"Ranskan Polynesia","est":"Prantsuse Polüneesia","zho":"法属波利尼西亚","pol":"Polinezja Francuska","urd":"فرانسیسی پولینیشیا","kor":"프랑스령 폴리네시아"}},"TF":{"currency":["EUR"],"callingCode":[],"region":"Antarctic","subregion":"","flag":"flag-tf","name":{"common":"French Southern and Antarctic Lands","ces":"Francouzská jižní a antarktická území","deu":"Französische Süd- und Antarktisgebiete","fra":"Terres australes et antarctiques françaises","hrv":"Francuski južni i antarktički teritoriji","ita":"Territori Francesi del Sud","jpn":"フランス領南方・南極地域","nld":"Franse Gebieden in de zuidelijke Indische Oceaan","por":"Terras Austrais e Antárticas Francesas","rus":"Французские Южные и Антарктические территории","slk":"Francúzske juŽné a antarktické územia","spa":"Tierras Australes y Antárticas Francesas","fin":"Ranskan eteläiset ja antarktiset alueet","est":"Prantsuse Lõunaalad","zho":"法国南部和南极土地","pol":"Francuskie Terytoria Południowe i Antarktyczne","urd":"سرزمین جنوبی فرانسیسیہ و انٹارکٹیکا","kor":"프랑스령 남부와 남극 지역"}},"GA":{"currency":["XAF"],"callingCode":["241"],"region":"Africa","subregion":"Middle Africa","flag":"flag-ga","name":{"common":"Gabon","ces":"Gabon","deu":"Gabun","fra":"Gabon","hrv":"Gabon","ita":"Gabon","jpn":"ガボン","nld":"Gabon","por":"Gabão","rus":"Габон","slk":"Gabon","spa":"Gabón","fin":"Gabon","est":"Gabon","zho":"加蓬","pol":"Gabon","urd":"گیبون","kor":"가봉"}},"GM":{"currency":["GMD"],"callingCode":["220"],"region":"Africa","subregion":"Western Africa","flag":"flag-gm","name":{"common":"Gambia","ces":"Gambie","deu":"Gambia","fra":"Gambie","hrv":"Gambija","ita":"Gambia","jpn":"ガンビア","nld":"Gambia","por":"Gâmbia","rus":"Гамбия","slk":"Gambia","spa":"Gambia","fin":"Gambia","est":"Gambia","zho":"冈比亚","pol":"Gambia","urd":"گیمبیا","kor":"감비아"}},"GE":{"currency":["GEL"],"callingCode":["995"],"region":"Asia","subregion":"Western Asia","flag":"flag-ge","name":{"common":"Georgia","ces":"Gruzie","deu":"Georgien","fra":"Géorgie","hrv":"Gruzija","ita":"Georgia","jpn":"グルジア","nld":"Georgië","por":"Geórgia","rus":"Грузия","slk":"Gruzínsko","spa":"Georgia","fin":"Georgia","est":"Gruusia","zho":"格鲁吉亚","pol":"Gruzja","urd":"جارجیا","kor":"조지아"}},"DE":{"currency":["EUR"],"callingCode":["49"],"region":"Europe","subregion":"Western Europe","flag":"flag-de","name":{"common":"Germany","ces":"Německo","deu":"Deutschland","fra":"Allemagne","hrv":"Njemačka","ita":"Germania","jpn":"ドイツ","nld":"Duitsland","por":"Alemanha","rus":"Германия","slk":"Nemecko","spa":"Alemania","fin":"Saksa","est":"Saksamaa","zho":"德国","pol":"Niemcy","urd":"جرمنی","kor":"독일"}},"GH":{"currency":["GHS"],"callingCode":["233"],"region":"Africa","subregion":"Western Africa","flag":"flag-gh","name":{"common":"Ghana","ces":"Ghana","deu":"Ghana","fra":"Ghana","hrv":"Gana","ita":"Ghana","jpn":"ガーナ","nld":"Ghana","por":"Gana","rus":"Гана","slk":"Ghana","spa":"Ghana","fin":"Ghana","est":"Ghana","zho":"加纳","pol":"Ghana","urd":"گھانا","kor":"가나"}},"GI":{"currency":["GIP"],"callingCode":["350"],"region":"Europe","subregion":"Southern Europe","flag":"flag-gi","name":{"common":"Gibraltar","ces":"Gibraltar","deu":"Gibraltar","fra":"Gibraltar","hrv":"Gibraltar","ita":"Gibilterra","jpn":"ジブラルタル","nld":"Gibraltar","por":"Gibraltar","rus":"Гибралтар","slk":"Gibraltár","spa":"Gibraltar","fin":"Gibraltar","est":"Gibraltar","zho":"直布罗陀","pol":"Gibraltar","urd":"جبل الطارق","kor":"지브롤터"}},"GR":{"currency":["EUR"],"callingCode":["30"],"region":"Europe","subregion":"Southern Europe","flag":"flag-gr","name":{"common":"Greece","ces":"Řecko","deu":"Griechenland","fra":"Grèce","hrv":"Grčka","ita":"Grecia","jpn":"ギリシャ","nld":"Griekenland","por":"Grécia","rus":"Греция","slk":"Greécko","spa":"Grecia","fin":"Kreikka","est":"Kreeka","zho":"希腊","pol":"Grecja","urd":"یونان","kor":"그리스"}},"GL":{"currency":["DKK"],"callingCode":["299"],"region":"Americas","subregion":"North America","flag":"flag-gl","name":{"common":"Greenland","ces":"Grónsko","deu":"Grönland","fra":"Groenland","hrv":"Grenland","ita":"Groenlandia","jpn":"グリーンランド","nld":"Groenland","por":"Gronelândia","rus":"Гренландия","slk":"Grónsko","spa":"Groenlandia","fin":"Groönlanti","est":"Gröönimaa","zho":"格陵兰","pol":"Grenlandia","urd":"گرین لینڈ","kor":"그린란드"}},"GD":{"currency":["XCD"],"callingCode":["1473"],"region":"Americas","subregion":"Caribbean","flag":"flag-gd","name":{"common":"Grenada","ces":"Grenada","deu":"Grenada","fra":"Grenade","hrv":"Grenada","ita":"Grenada","jpn":"グレナダ","nld":"Grenada","por":"Granada","rus":"Гренада","slk":"Grenada","spa":"Grenada","fin":"Grenada","est":"Grenada","zho":"格林纳达","pol":"Grenada","urd":"گریناڈا","kor":"그레나다"}},"GP":{"currency":["EUR"],"callingCode":["590"],"region":"Americas","subregion":"Caribbean","flag":"flag-gp","name":{"common":"Guadeloupe","ces":"Guadeloupe","deu":"Guadeloupe","fra":"Guadeloupe","hrv":"Gvadalupa","ita":"Guadeloupa","jpn":"グアドループ","nld":"Guadeloupe","por":"Guadalupe","rus":"Гваделупа","slk":"Guadeloupe","spa":"Guadalupe","fin":"Guadeloupe","est":"Guadeloupe","zho":"瓜德罗普岛","pol":"Gwadelupa","urd":"گواڈیلوپ","kor":"과들루프"}},"GU":{"currency":["USD"],"callingCode":["1671"],"region":"Oceania","subregion":"Micronesia","flag":"flag-gu","name":{"common":"Guam","ces":"Guam","deu":"Guam","fra":"Guam","hrv":"Guam","ita":"Guam","jpn":"グアム","nld":"Guam","por":"Guam","rus":"Гуам","slk":"Guam","spa":"Guam","fin":"Guam","est":"Guam","zho":"关岛","pol":"Guam","urd":"گوام","kor":"괌"}},"GT":{"currency":["GTQ"],"callingCode":["502"],"region":"Americas","subregion":"Central America","flag":"flag-gt","name":{"common":"Guatemala","ces":"Guatemala","deu":"Guatemala","fra":"Guatemala","hrv":"Gvatemala","ita":"Guatemala","jpn":"グアテマラ","nld":"Guatemala","por":"Guatemala","rus":"Гватемала","slk":"Guatemala","spa":"Guatemala","fin":"Guatemala","est":"Guatemala","zho":"危地马拉","pol":"Gwatemala","urd":"گواتیمالا","kor":"과테말라"}},"GG":{"currency":["GBP"],"callingCode":["44"],"region":"Europe","subregion":"Northern Europe","flag":"flag-gg","name":{"common":"Guernsey","ces":"Guernsey","deu":"Guernsey","fra":"Guernesey","hrv":"Guernsey","ita":"Guernsey","jpn":"ガーンジー","nld":"Guernsey","por":"Guernsey","rus":"Гернси","slk":"Guernsey","spa":"Guernsey","fin":"Guernsey","est":"Guernsey","zho":"根西岛","pol":"Guernsey","urd":"گرنزی","kor":"건지 섬"}},"GN":{"currency":["GNF"],"callingCode":["224"],"region":"Africa","subregion":"Western Africa","flag":"flag-gn","name":{"common":"Guinea","ces":"Guinea","deu":"Guinea","fra":"Guinée","hrv":"Gvineja","ita":"Guinea","jpn":"ギニア","nld":"Guinee","por":"Guiné","rus":"Гвинея","slk":"Guinea","spa":"Guinea","fin":"Guinea","est":"Guinea","zho":"几内亚","pol":"Gwinea","urd":"گنی","kor":"기니"}},"GW":{"currency":["XOF"],"callingCode":["245"],"region":"Africa","subregion":"Western Africa","flag":"flag-gw","name":{"common":"Guinea-Bissau","ces":"Guinea-Bissau","deu":"Guinea-Bissau","fra":"Guinée-Bissau","hrv":"Gvineja Bisau","ita":"Guinea-Bissau","jpn":"ギニアビサウ","nld":"Guinee-Bissau","por":"Guiné-Bissau","rus":"Гвинея-Бисау","slk":"Guinea-Bissau","spa":"Guinea-Bisáu","fin":"Guinea-Bissau","est":"Guinea-Bissau","zho":"几内亚比绍","pol":"Gwinea Bissau","urd":"گنی بساؤ","kor":"기니비사우"}},"GY":{"currency":["GYD"],"callingCode":["592"],"region":"Americas","subregion":"South America","flag":"flag-gy","name":{"common":"Guyana","ces":"Guyana","deu":"Guyana","fra":"Guyana","hrv":"Gvajana","ita":"Guyana","jpn":"ガイアナ","nld":"Guyana","por":"Guiana","rus":"Гайана","slk":"Guyana","spa":"Guyana","fin":"Guayana","est":"Guyana","zho":"圭亚那","pol":"Gujana","urd":"گیانا","kor":"가이아나"}},"HT":{"currency":["HTG","USD"],"callingCode":["509"],"region":"Americas","subregion":"Caribbean","flag":"flag-ht","name":{"common":"Haiti","ces":"Haiti","deu":"Haiti","fra":"Haïti","hrv":"Haiti","ita":"Haiti","jpn":"ハイチ","nld":"Haïti","por":"Haiti","rus":"Гаити","slk":"Haiti","spa":"Haiti","fin":"Haiti","est":"Haiti","zho":"海地","pol":"Haiti","urd":"ہیٹی","kor":"아이티"}},"HM":{"currency":["AUD"],"callingCode":[],"region":"Antarctic","subregion":"","flag":"flag-hm","name":{"common":"Heard Island and McDonald Islands","ces":"Heardův ostrov a McDonaldovy ostrovy","deu":"Heard und die McDonaldinseln","fra":"Îles Heard-et-MacDonald","hrv":"Otok Heard i otočje McDonald","ita":"Isole Heard e McDonald","jpn":"ハード島とマクドナルド諸島","nld":"Heard-en McDonaldeilanden","por":"Ilha Heard e Ilhas McDonald","rus":"Остров Херд и острова Макдональд","slk":"Heardov ostrov","spa":"Islas Heard y McDonald","fin":"Heard ja McDonaldinsaaret","est":"Heard ja McDonald","zho":"赫德岛和麦当劳群岛","pol":"Wyspy Heard i McDonalda","urd":"جزیرہ ہرڈ و جزائر مکڈونلڈ","kor":"허드 맥도널드 제도"}},"HN":{"currency":["HNL"],"callingCode":["504"],"region":"Americas","subregion":"Central America","flag":"flag-hn","name":{"common":"Honduras","ces":"Honduras","deu":"Honduras","fra":"Honduras","hrv":"Honduras","ita":"Honduras","jpn":"ホンジュラス","nld":"Honduras","por":"Honduras","rus":"Гондурас","slk":"Honduras","spa":"Honduras","fin":"Honduras","est":"Honduras","zho":"洪都拉斯","pol":"Honduras","urd":"ہونڈوراس","kor":"온두라스"}},"HK":{"currency":["HKD"],"callingCode":["852"],"region":"Asia","subregion":"Eastern Asia","flag":"flag-hk","name":{"common":"Hong Kong","ces":"Hongkong","deu":"Hongkong","fra":"Hong Kong","hrv":"Hong Kong","ita":"Hong Kong","jpn":"香港","nld":"Hongkong","por":"Hong Kong","rus":"Гонконг","slk":"Hongkong","spa":"Hong Kong","fin":"Hongkong","est":"Hongkong","pol":"Hongkong","urd":"ہانگ کانگ","kor":"홍콩"}},"HU":{"currency":["HUF"],"callingCode":["36"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-hu","name":{"common":"Hungary","ces":"Maďarsko","deu":"Ungarn","fra":"Hongrie","hrv":"Mađarska","ita":"Ungheria","jpn":"ハンガリー","nld":"Hongarije","por":"Hungria","rus":"Венгрия","slk":"Maďarsko","spa":"Hungría","fin":"Unkari","est":"Ungari","zho":"匈牙利","pol":"Węgry","urd":"مجارستان","kor":"헝가리"}},"IS":{"currency":["ISK"],"callingCode":["354"],"region":"Europe","subregion":"Northern Europe","flag":"flag-is","name":{"common":"Iceland","ces":"Island","deu":"Island","fra":"Islande","hrv":"Island","ita":"Islanda","jpn":"アイスランド","nld":"IJsland","por":"Islândia","rus":"Исландия","slk":"Island","spa":"Islandia","fin":"Islanti","est":"Island","zho":"冰岛","pol":"Islandia","urd":"آئس لینڈ","kor":"아이슬란드"}},"IN":{"currency":["INR"],"callingCode":["91"],"region":"Asia","subregion":"Southern Asia","flag":"flag-in","name":{"common":"India","ces":"Indie","deu":"Indien","fra":"Inde","hrv":"Indija","ita":"India","jpn":"インド","nld":"India","por":"Índia","rus":"Индия","slk":"India","spa":"India","fin":"Intia","est":"India","zho":"印度","pol":"Indie","urd":"بھارت","kor":"인도"}},"ID":{"currency":["IDR"],"callingCode":["62"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-id","name":{"common":"Indonesia","ces":"Indonésie","deu":"Indonesien","fra":"Indonésie","hrv":"Indonezija","ita":"Indonesia","jpn":"インドネシア","nld":"Indonesië","por":"Indonésia","rus":"Индонезия","slk":"Indonézia","spa":"Indonesia","fin":"Indonesia","est":"Indoneesia","zho":"印度尼西亚","pol":"Indonezja","urd":"انڈونیشیا","kor":"인도네시아"}},"IR":{"currency":["IRR"],"callingCode":["98"],"region":"Asia","subregion":"Southern Asia","flag":"flag-ir","name":{"common":"Iran","ces":"Írán","deu":"Iran","fra":"Iran","hrv":"Iran","ita":"Iran","jpn":"イラン・イスラム共和国","nld":"Iran","por":"Irão","rus":"Иран","slk":"Irán","spa":"Iran","fin":"Iran","est":"Iraan","zho":"伊朗","pol":"Iran","urd":"ایران","kor":"이란"}},"IQ":{"currency":["IQD"],"callingCode":["964"],"region":"Asia","subregion":"Western Asia","flag":"flag-iq","name":{"common":"Iraq","ces":"Irák","deu":"Irak","fra":"Irak","hrv":"Irak","ita":"Iraq","jpn":"イラク","nld":"Irak","por":"Iraque","rus":"Ирак","slk":"Irak","spa":"Irak","fin":"Irak","est":"Iraak","zho":"伊拉克","pol":"Irak","urd":"عراق","kor":"이라크"}},"IE":{"currency":["EUR"],"callingCode":["353"],"region":"Europe","subregion":"Northern Europe","flag":"flag-ie","name":{"common":"Ireland","ces":"Irsko","deu":"Irland","fra":"Irlande","hrv":"Irska","ita":"Irlanda","jpn":"アイルランド","nld":"Ierland","por":"Irlanda","rus":"Ирландия","slk":"Írsko","spa":"Irlanda","fin":"Irlanti","est":"Iirimaa","zho":"爱尔兰","pol":"Irlandia","urd":"جزیرہ آئرلینڈ","kor":"아일랜드"}},"IM":{"currency":["GBP"],"callingCode":["44"],"region":"Europe","subregion":"Northern Europe","flag":"flag-im","name":{"common":"Isle of Man","ces":"Ostrov Man","deu":"Insel Man","fra":"Île de Man","hrv":"Otok Man","ita":"Isola di Man","jpn":"マン島","nld":"Isle of Man","por":"Ilha de Man","rus":"Остров Мэн","slk":"Man","spa":"Isla de Man","fin":"Mansaari","est":"Mani saar","zho":"马恩岛","pol":"Wyspa Man","urd":"آئل آف مین","kor":"맨섬"}},"IL":{"currency":["ILS"],"callingCode":["972"],"region":"Asia","subregion":"Western Asia","flag":"flag-il","name":{"common":"Israel","ces":"Izrael","deu":"Israel","fra":"Israël","hrv":"Izrael","ita":"Israele","jpn":"イスラエル","nld":"Israël","por":"Israel","rus":"Израиль","slk":"Izrael","spa":"Israel","fin":"Israel","est":"Iisrael","zho":"以色列","pol":"Izrael","urd":"اسرائیل","kor":"이스라엘"}},"IT":{"currency":["EUR"],"callingCode":["39"],"region":"Europe","subregion":"Southern Europe","flag":"flag-it","name":{"common":"Italy","ces":"Itálie","deu":"Italien","fra":"Italie","hrv":"Italija","ita":"Italia","jpn":"イタリア","nld":"Italië","por":"Itália","rus":"Италия","slk":"Taliansko","spa":"Italia","fin":"Italia","est":"Itaalia","zho":"意大利","pol":"Włochy","urd":"اطالیہ","kor":"이탈리아"}},"CI":{"currency":["XOF"],"callingCode":["225"],"region":"Africa","subregion":"Western Africa","flag":"flag-ci","name":{"common":"Ivory Coast","ces":"Pobřeží slonoviny","deu":"Elfenbeinküste","fra":"Côte d'Ivoire","hrv":"Obala Bjelokosti","ita":"Costa d'Avorio","jpn":"コートジボワール","nld":"Ivoorkust","por":"Costa do Marfim","rus":"Кот-д’Ивуар","slk":"Pobržie Slonoviny","spa":"Costa de Marfil","fin":"Norsunluurannikko","est":"Elevandiluurannik","zho":"科特迪瓦","pol":"WybrzeŻe Kości Słoniowej","urd":"آئیوری کوسٹ","kor":"코트디부아르"}},"JM":{"currency":["JMD"],"callingCode":["1876"],"region":"Americas","subregion":"Caribbean","flag":"flag-jm","name":{"common":"Jamaica","ces":"Jamajka","deu":"Jamaika","fra":"Jamaïque","hrv":"Jamajka","ita":"Giamaica","jpn":"ジャマイカ","nld":"Jamaica","por":"Jamaica","rus":"Ямайка","slk":"Jamajka","spa":"Jamaica","fin":"Jamaika","est":"Jamaica","zho":"牙买加","pol":"Jamajka","urd":"جمیکا","kor":"자메이카"}},"JP":{"currency":["JPY"],"callingCode":["81"],"region":"Asia","subregion":"Eastern Asia","flag":"flag-jp","name":{"common":"Japan","ces":"Japonsko","deu":"Japan","fra":"Japon","hrv":"Japan","ita":"Giappone","jpn":"日本","nld":"Japan","por":"Japão","rus":"Япония","slk":"Japonsko","spa":"Japón","fin":"Japani","est":"Jaapan","zho":"日本","pol":"Japonia","urd":"جاپان","kor":"일본"}},"JE":{"currency":["GBP"],"callingCode":["44"],"region":"Europe","subregion":"Northern Europe","flag":"flag-je","name":{"common":"Jersey","ces":"Jersey","deu":"Jersey","fra":"Jersey","hrv":"Jersey","ita":"Isola di Jersey","jpn":"ジャージー","nld":"Jersey","por":"Jersey","rus":"Джерси","slk":"Jersey","spa":"Jersey","fin":"Jersey","est":"Jersey","zho":"泽西岛","pol":"Jersey","urd":"جرزی","kor":"저지 섬"}},"JO":{"currency":["JOD"],"callingCode":["962"],"region":"Asia","subregion":"Western Asia","flag":"flag-jo","name":{"common":"Jordan","ces":"Jordánsko","deu":"Jordanien","fra":"Jordanie","hrv":"Jordan","ita":"Giordania","jpn":"ヨルダン","nld":"Jordanië","por":"Jordânia","rus":"Иордания","slk":"Jordánsko","spa":"Jordania","fin":"Jordania","est":"Jordaania","zho":"约旦","pol":"Jordania","urd":"اردن","kor":"요르단"}},"KZ":{"currency":["KZT"],"callingCode":["7"],"region":"Asia","subregion":"Central Asia","flag":"flag-kz","name":{"common":"Kazakhstan","ces":"Kazachstán","deu":"Kasachstan","fra":"Kazakhstan","hrv":"Kazahstan","ita":"Kazakistan","jpn":"カザフスタン","nld":"Kazachstan","por":"Cazaquistão","rus":"Казахстан","slk":"Kazachstan","spa":"Kazajistán","fin":"Kazakstan","est":"Kasahstan","zho":"哈萨克斯坦","pol":"Kazachstan","urd":"قازقستان","kor":"카자흐스탄"}},"KE":{"currency":["KES"],"callingCode":["254"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-ke","name":{"common":"Kenya","ces":"Keňa","deu":"Kenia","fra":"Kenya","hrv":"Kenija","ita":"Kenya","jpn":"ケニア","nld":"Kenia","por":"Quénia","rus":"Кения","slk":"Keňa","spa":"Kenia","fin":"Kenia","est":"Keenia","zho":"肯尼亚","pol":"Kenia","urd":"کینیا","kor":"케냐"}},"KI":{"currency":["AUD"],"callingCode":["686"],"region":"Oceania","subregion":"Micronesia","flag":"flag-ki","name":{"common":"Kiribati","ces":"Kiribati","deu":"Kiribati","fra":"Kiribati","hrv":"Kiribati","ita":"Kiribati","jpn":"キリバス","nld":"Kiribati","por":"Kiribati","rus":"Кирибати","slk":"Kiribati","spa":"Kiribati","fin":"Kiribati","est":"Kiribati","zho":"基里巴斯","pol":"Kiribati","urd":"کیریباتی","kor":"키리바시"}},"XK":{"currency":["EUR"],"callingCode":["383"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-xk","name":{"common":"Kosovo","ces":"Kosovo","deu":"Kosovo","fra":"Kosovo","hrv":"Kosovo","ita":"Kosovo","nld":"Kosovo","por":"Kosovo","rus":"Республика Косово","slk":"Kosovo","spa":"Kosovo","fin":"Kosovo","est":"Kosovo","zho":"科索沃","pol":"Kosowo","urd":"کوسووہ","kor":"코소보"}},"KW":{"currency":["KWD"],"callingCode":["965"],"region":"Asia","subregion":"Western Asia","flag":"flag-kw","name":{"common":"Kuwait","ces":"Kuvajt","deu":"Kuwait","fra":"Koweït","hrv":"Kuvajt","ita":"Kuwait","jpn":"クウェート","nld":"Koeweit","por":"Kuwait","rus":"Кувейт","slk":"Kuvajt","spa":"Kuwait","fin":"Kuwait","est":"Kuveit","zho":"科威特","pol":"Kuwejt","urd":"کویت","kor":"쿠웨이트"}},"KG":{"currency":["KGS"],"callingCode":["996"],"region":"Asia","subregion":"Central Asia","flag":"flag-kg","name":{"common":"Kyrgyzstan","ces":"Kyrgyzstán","deu":"Kirgisistan","fra":"Kirghizistan","hrv":"Kirgistan","ita":"Kirghizistan","jpn":"キルギス","nld":"Kirgizië","por":"Quirguistão","rus":"Киргизия","slk":"Kirgizsko","spa":"Kirguizistán","fin":"Kirgisia","est":"Kõrgõzstan","zho":"吉尔吉斯斯坦","pol":"Kirgistan","urd":"کرغیزستان","kor":"키르기스스탄"}},"LA":{"currency":["LAK"],"callingCode":["856"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-la","name":{"common":"Laos","ces":"Laos","deu":"Laos","fra":"Laos","hrv":"Laos","ita":"Laos","jpn":"ラオス人民民主共和国","nld":"Laos","por":"Laos","rus":"Лаос","slk":"Laos","spa":"Laos","fin":"Laos","est":"Laos","zho":"老挝","pol":"Laos","urd":"لاؤس","kor":"라오스"}},"LV":{"currency":["EUR"],"callingCode":["371"],"region":"Europe","subregion":"Northern Europe","flag":"flag-lv","name":{"common":"Latvia","ces":"Lotyšsko","deu":"Lettland","fra":"Lettonie","hrv":"Latvija","ita":"Lettonia","jpn":"ラトビア","nld":"Letland","por":"Letónia","rus":"Латвия","slk":"Lotyšsko","spa":"Letonia","fin":"Latvia","est":"Läti","zho":"拉脱维亚","pol":"Łotwa","urd":"لٹویا","kor":"라트비아"}},"LB":{"currency":["LBP"],"callingCode":["961"],"region":"Asia","subregion":"Western Asia","flag":"flag-lb","name":{"common":"Lebanon","ces":"Libanon","deu":"Libanon","fra":"Liban","hrv":"Libanon","ita":"Libano","jpn":"レバノン","nld":"Libanon","por":"Líbano","rus":"Ливан","slk":"Libanon","spa":"Líbano","fin":"Libanon","est":"Liibanon","zho":"黎巴嫩","pol":"Liban","urd":"لبنان","kor":"레바논"}},"LS":{"currency":["LSL","ZAR"],"callingCode":["266"],"region":"Africa","subregion":"Southern Africa","flag":"flag-ls","name":{"common":"Lesotho","ces":"Lesotho","deu":"Lesotho","fra":"Lesotho","hrv":"Lesoto","ita":"Lesotho","jpn":"レソト","nld":"Lesotho","por":"Lesoto","rus":"Лесото","slk":"Lesotho","spa":"Lesotho","fin":"Lesotho","est":"Lesotho","zho":"莱索托","pol":"Lesotho","urd":"لیسوتھو","kor":"레소토"}},"LR":{"currency":["LRD"],"callingCode":["231"],"region":"Africa","subregion":"Western Africa","flag":"flag-lr","name":{"common":"Liberia","ces":"Libérie","deu":"Liberia","fra":"Liberia","hrv":"Liberija","ita":"Liberia","jpn":"リベリア","nld":"Liberia","por":"Libéria","rus":"Либерия","slk":"Libéria","spa":"Liberia","fin":"Liberia","est":"Libeeria","zho":"利比里亚","pol":"Liberia","urd":"لائبیریا","kor":"라이베리아"}},"LY":{"currency":["LYD"],"callingCode":["218"],"region":"Africa","subregion":"Northern Africa","flag":"flag-ly","name":{"common":"Libya","ces":"Libye","deu":"Libyen","fra":"Libye","hrv":"Libija","ita":"Libia","jpn":"リビア","nld":"Libië","por":"Líbia","rus":"Ливия","slk":"Líbya","spa":"Libia","fin":"Libya","est":"Liibüa","zho":"利比亚","pol":"Libia","urd":"لیبیا","kor":"리비아"}},"LI":{"currency":["CHF"],"callingCode":["423"],"region":"Europe","subregion":"Western Europe","flag":"flag-li","name":{"common":"Liechtenstein","ces":"Lichtenštejnsko","deu":"Liechtenstein","fra":"Liechtenstein","hrv":"Lihtenštajn","ita":"Liechtenstein","jpn":"リヒテンシュタイン","nld":"Liechtenstein","por":"Liechtenstein","rus":"Лихтенштейн","slk":"Lichtenštajnsko","spa":"Liechtenstein","fin":"Liechenstein","est":"Liechtenstein","zho":"列支敦士登","pol":"Liechtenstein","urd":"لیختینستائن","kor":"리히텐슈타인"}},"LT":{"currency":["EUR"],"callingCode":["370"],"region":"Europe","subregion":"Northern Europe","flag":"flag-lt","name":{"common":"Lithuania","ces":"Litva","deu":"Litauen","fra":"Lituanie","hrv":"Litva","ita":"Lituania","jpn":"リトアニア","nld":"Litouwen","por":"Lituânia","rus":"Литва","slk":"Litva","spa":"Lituania","fin":"Liettua","est":"Leedu","zho":"立陶宛","pol":"Litwa","urd":"لتھووینیا","kor":"리투아니아"}},"LU":{"currency":["EUR"],"callingCode":["352"],"region":"Europe","subregion":"Western Europe","flag":"flag-lu","name":{"common":"Luxembourg","ces":"Lucembursko","deu":"Luxemburg","fra":"Luxembourg","hrv":"Luksemburg","ita":"Lussemburgo","jpn":"ルクセンブルク","nld":"Luxemburg","por":"Luxemburgo","rus":"Люксембург","slk":"Luxembursko","spa":"Luxemburgo","fin":"Luxemburg","est":"Luksemburg","zho":"卢森堡","pol":"Luksemburg","urd":"لکسمبرگ","kor":"룩셈부르크"}},"MO":{"currency":["MOP"],"callingCode":["853"],"region":"Asia","subregion":"Eastern Asia","flag":"flag-mo","name":{"common":"Macau","ces":"Macao","deu":"Macao","fra":"Macao","hrv":"Makao","ita":"Macao","jpn":"マカオ","nld":"Macao","por":"Macau","rus":"Макао","slk":"Macao","spa":"Macao","fin":"Macao","est":"Macau","pol":"Makau","urd":"مکاؤ","kor":"마카오"}},"MK":{"currency":["MKD"],"callingCode":["389"],"region":"Europe","subregion":"Southern Europe","flag":"flag-mk","name":{"common":"Macedonia","ces":"Makedonie","deu":"Mazedonien","fra":"Macédoine","hrv":"Makedonija","ita":"Macedonia","jpn":"マケドニア旧ユーゴスラビア共和国","nld":"Macedonië","por":"Macedónia","rus":"Республика Македония","slk":"Macedónsko","spa":"Macedonia","fin":"Makedonia","est":"Makedoonia","zho":"马其顿","pol":"Macedonia","urd":"مقدونیہ","kor":"마케도니아"}},"MG":{"currency":["MGA"],"callingCode":["261"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-mg","name":{"common":"Madagascar","ces":"Madagaskar","deu":"Madagaskar","fra":"Madagascar","hrv":"Madagaskar","ita":"Madagascar","jpn":"マダガスカル","nld":"Madagaskar","por":"Madagáscar","rus":"Мадагаскар","slk":"Madagaskar","spa":"Madagascar","fin":"Madagaskar","est":"Madagaskar","zho":"马达加斯加","pol":"Madagaskar","urd":"مڈغاسکر","kor":"마다가스카르"}},"MW":{"currency":["MWK"],"callingCode":["265"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-mw","name":{"common":"Malawi","ces":"Malawi","deu":"Malawi","fra":"Malawi","hrv":"Malavi","ita":"Malawi","jpn":"マラウイ","nld":"Malawi","por":"Malawi","rus":"Малави","slk":"Malawi","spa":"Malawi","fin":"Malawi","est":"Malawi","zho":"马拉维","pol":"Malawi","urd":"ملاوی","kor":"말라위"}},"MY":{"currency":["MYR"],"callingCode":["60"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-my","name":{"common":"Malaysia","ces":"Malajsie","deu":"Malaysia","fra":"Malaisie","hrv":"Malezija","ita":"Malesia","jpn":"マレーシア","nld":"Maleisië","por":"Malásia","rus":"Малайзия","slk":"Malajzia","spa":"Malasia","fin":"Malesia","est":"Malaisia","zho":"马来西亚","pol":"Malezja","urd":"ملائیشیا","kor":"말레이시아"}},"MV":{"currency":["MVR"],"callingCode":["960"],"region":"Asia","subregion":"Southern Asia","flag":"flag-mv","name":{"common":"Maldives","ces":"Maledivy","deu":"Malediven","fra":"Maldives","hrv":"Maldivi","ita":"Maldive","jpn":"モルディブ","nld":"Maldiven","por":"Maldivas","spa":"Maldivas","rus":"Мальдивы","slk":"Maldivy","fin":"Malediivit","est":"Maldiivid","zho":"马尔代夫","pol":"Malediwy","urd":"مالدیپ","kor":"몰디브"}},"ML":{"currency":["XOF"],"callingCode":["223"],"region":"Africa","subregion":"Western Africa","flag":"flag-ml","name":{"common":"Mali","ces":"Mali","deu":"Mali","fra":"Mali","hrv":"Mali","ita":"Mali","jpn":"マリ","nld":"Mali","por":"Mali","rus":"Мали","slk":"Mali","spa":"Mali","fin":"Mali","est":"Mali","zho":"马里","pol":"Mali","urd":"مالی","kor":"말리"}},"MT":{"currency":["EUR"],"callingCode":["356"],"region":"Europe","subregion":"Southern Europe","flag":"flag-mt","name":{"common":"Malta","ces":"Malta","deu":"Malta","fra":"Malte","hrv":"Malta","ita":"Malta","jpn":"マルタ","nld":"Malta","por":"Malta","rus":"Мальта","slk":"Malta","spa":"Malta","fin":"Malta","est":"Malta","zho":"马耳他","pol":"Malta","urd":"مالٹا","kor":"몰타"}},"MH":{"currency":["USD"],"callingCode":["692"],"region":"Oceania","subregion":"Micronesia","flag":"flag-mh","name":{"common":"Marshall Islands","ces":"Marshallovy ostrovy","deu":"Marshallinseln","fra":"Îles Marshall","hrv":"Maršalovi Otoci","ita":"Isole Marshall","jpn":"マーシャル諸島","nld":"Marshalleilanden","por":"Ilhas Marshall","rus":"Маршалловы Острова","slk":"Marshallove ostrovy","spa":"Islas Marshall","fin":"Marshallinsaaret","est":"Marshalli Saared","zho":"马绍尔群岛","pol":"Wyspy Marshalla","urd":"جزائر مارشل","kor":"마셜 제도"}},"MQ":{"currency":["EUR"],"callingCode":["596"],"region":"Americas","subregion":"Caribbean","flag":"flag-mq","name":{"common":"Martinique","ces":"Martinik","deu":"Martinique","fra":"Martinique","hrv":"Martinique","ita":"Martinica","jpn":"マルティニーク","nld":"Martinique","por":"Martinica","rus":"Мартиника","spa":"Martinica","slk":"Martinique","fin":"Martinique","est":"Martinique","zho":"马提尼克","pol":"Martynika","urd":"مارٹینیک","kor":"마르티니크"}},"MR":{"currency":["MRO"],"callingCode":["222"],"region":"Africa","subregion":"Western Africa","flag":"flag-mr","name":{"common":"Mauritania","ces":"Mauritánie","deu":"Mauretanien","fra":"Mauritanie","hrv":"Mauritanija","ita":"Mauritania","jpn":"モーリタニア","nld":"Mauritanië","por":"Mauritânia","rus":"Мавритания","slk":"Mauritánia","spa":"Mauritania","fin":"Mauritania","est":"Mauritaania","zho":"毛里塔尼亚","pol":"Mauretania","urd":"موریتانیہ","kor":"모리타니"}},"MU":{"currency":["MUR"],"callingCode":["230"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-mu","name":{"common":"Mauritius","ces":"Mauricius","deu":"Mauritius","fra":"Île Maurice","hrv":"Mauricijus","ita":"Mauritius","jpn":"モーリシャス","nld":"Mauritius","por":"Maurício","rus":"Маврикий","slk":"Maurícius","spa":"Mauricio","fin":"Mauritius","est":"Mauritius","zho":"毛里求斯","pol":"Mauritius","urd":"موریشس","kor":"모리셔스"}},"YT":{"currency":["EUR"],"callingCode":["262"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-yt","name":{"common":"Mayotte","ces":"Mayotte","deu":"Mayotte","fra":"Mayotte","hrv":"Mayotte","ita":"Mayotte","jpn":"マヨット","nld":"Mayotte","por":"Mayotte","rus":"Майотта","slk":"Mayotte","spa":"Mayotte","fin":"Mayotte","est":"Mayotte","zho":"马约特","pol":"Majotta","urd":"مایوٹ","kor":"마요트"}},"MX":{"currency":["MXN"],"callingCode":["52"],"region":"Americas","subregion":"North America","flag":"flag-mx","name":{"common":"Mexico","ces":"Mexiko","deu":"Mexiko","fra":"Mexique","hrv":"Meksiko","ita":"Messico","jpn":"メキシコ","nld":"Mexico","por":"México","rus":"Мексика","slk":"Mexiko","spa":"México","fin":"Meksiko","est":"Mehhiko","zho":"墨西哥","pol":"Meksyk","urd":"میکسیکو","kor":"멕시코"}},"FM":{"currency":["USD"],"callingCode":["691"],"region":"Oceania","subregion":"Micronesia","flag":"flag-fm","name":{"common":"Micronesia","ces":"Mikronésie","deu":"Mikronesien","fra":"Micronésie","hrv":"Mikronezija","ita":"Micronesia","jpn":"ミクロネシア連邦","nld":"Micronesië","por":"Micronésia","rus":"Федеративные Штаты Микронезии","slk":"Mikronézia","spa":"Micronesia","fin":"Mikronesia","est":"Mikroneesia","zho":"密克罗尼西亚","pol":"Mikronezja","urd":"مائکرونیشیا","kor":"미크로네시아"}},"MD":{"currency":["MDL"],"callingCode":["373"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-md","name":{"common":"Moldova","ces":"Moldavsko","deu":"Moldawien","fra":"Moldavie","hrv":"Moldova","ita":"Moldavia","jpn":"モルドバ共和国","nld":"Moldavië","por":"Moldávia","rus":"Молдавия","slk":"Moldavsko","spa":"Moldavia","fin":"Moldova","est":"Moldova","zho":"摩尔多瓦","pol":"Mołdawia","urd":"مالدووا","kor":"몰도바"}},"MC":{"currency":["EUR"],"callingCode":["377"],"region":"Europe","subregion":"Western Europe","flag":"flag-mc","name":{"common":"Monaco","ces":"Monako","deu":"Monaco","fra":"Monaco","hrv":"Monako","ita":"Principato di Monaco","jpn":"モナコ","nld":"Monaco","por":"Mónaco","rus":"Монако","slk":"Monako","spa":"Mónaco","fin":"Monaco","est":"Monaco","zho":"摩纳哥","pol":"Monako","urd":"موناکو","kor":"모나코"}},"MN":{"currency":["MNT"],"callingCode":["976"],"region":"Asia","subregion":"Eastern Asia","flag":"flag-mn","name":{"common":"Mongolia","ces":"Mongolsko","deu":"Mongolei","fra":"Mongolie","hrv":"Mongolija","ita":"Mongolia","jpn":"モンゴル","nld":"Mongolië","por":"Mongólia","rus":"Монголия","slk":"Mongolsko","spa":"Mongolia","fin":"Mongolia","est":"Mongoolia","zho":"蒙古","pol":"Mongolia","urd":"منگولیا","kor":"몽골국"}},"ME":{"currency":["EUR"],"callingCode":["382"],"region":"Europe","subregion":"Southern Europe","flag":"flag-me","name":{"common":"Montenegro","ces":"Černá Hora","deu":"Montenegro","fra":"Monténégro","hrv":"Crna Gora","ita":"Montenegro","jpn":"モンテネグロ","nld":"Montenegro","por":"Montenegro","rus":"Черногория","slk":"Čierna Hora","spa":"Montenegro","fin":"Montenegro","est":"Montenegro","zho":"黑山","pol":"Czarnogóra","urd":"مونٹینیگرو","kor":"몬테네그로"}},"MS":{"currency":["XCD"],"callingCode":["1664"],"region":"Americas","subregion":"Caribbean","flag":"flag-ms","name":{"common":"Montserrat","ces":"Montserrat","deu":"Montserrat","fra":"Montserrat","hrv":"Montserrat","ita":"Montserrat","jpn":"モントセラト","nld":"Montserrat","por":"Montserrat","rus":"Монтсеррат","slk":"Montserrat","spa":"Montserrat","fin":"Montserrat","est":"Montserrat","zho":"蒙特塞拉特","pol":"Montserrat","urd":"مانٹسریٹ","kor":"몬트세랫"}},"MA":{"currency":["MAD"],"callingCode":["212"],"region":"Africa","subregion":"Northern Africa","flag":"flag-ma","name":{"common":"Morocco","ces":"Maroko","deu":"Marokko","fra":"Maroc","hrv":"Maroko","ita":"Marocco","jpn":"モロッコ","nld":"Marokko","por":"Marrocos","rus":"Марокко","slk":"Maroko","spa":"Marruecos","fin":"Marokko","est":"Maroko","zho":"摩洛哥","pol":"Maroko","urd":"مراکش","kor":"모로코"}},"MZ":{"currency":["MZN"],"callingCode":["258"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-mz","name":{"common":"Mozambique","ces":"Mosambik","deu":"Mosambik","fra":"Mozambique","hrv":"Mozambik","ita":"Mozambico","jpn":"モザンビーク","nld":"Mozambique","por":"Moçambique","rus":"Мозамбик","slk":"Mozambik","spa":"Mozambique","fin":"Mosambik","est":"Mosambiik","zho":"莫桑比克","pol":"Mozambik","urd":"موزمبیق","kor":"모잠비크"}},"MM":{"currency":["MMK"],"callingCode":["95"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-mm","name":{"common":"Myanmar","ces":"Myanmar","deu":"Myanmar","fra":"Birmanie","hrv":"Mijanmar","ita":"Birmania","jpn":"ミャンマー","nld":"Myanmar","por":"Myanmar","rus":"Мьянма","slk":"Mjanmarsko","spa":"Myanmar","fin":"Myanmar","est":"Myanmar","zho":"缅甸","pol":"Mjanma","urd":"میانمار","kor":"미얀마"}},"NA":{"currency":["NAD","ZAR"],"callingCode":["264"],"region":"Africa","subregion":"Southern Africa","flag":"flag-na","name":{"common":"Namibia","ces":"Namibie","deu":"Namibia","fra":"Namibie","hrv":"Namibija","ita":"Namibia","jpn":"ナミビア","nld":"Namibië","por":"Namíbia","rus":"Намибия","slk":"Namíbia","spa":"Namibia","fin":"Namibia","est":"Namiibia","zho":"纳米比亚","pol":"Namibia","urd":"نمیبیا","kor":"나미비아"}},"NR":{"currency":["AUD"],"callingCode":["674"],"region":"Oceania","subregion":"Micronesia","flag":"flag-nr","name":{"common":"Nauru","ces":"Nauru","deu":"Nauru","fra":"Nauru","hrv":"Nauru","ita":"Nauru","jpn":"ナウル","nld":"Nauru","por":"Nauru","rus":"Науру","slk":"Nauru","spa":"Nauru","fin":"Nauru","est":"Nauru","zho":"瑙鲁","pol":"Nauru","urd":"ناورو","kor":"나우루"}},"NP":{"currency":["NPR"],"callingCode":["977"],"region":"Asia","subregion":"Southern Asia","flag":"flag-np","name":{"common":"Nepal","ces":"Nepál","deu":"Nepal","fra":"Népal","hrv":"Nepal","ita":"Nepal","jpn":"ネパール","nld":"Nepal","por":"Nepal","rus":"Непал","slk":"Nepál","spa":"Nepal","fin":"Nepal","est":"Nepal","zho":"尼泊尔","pol":"Nepal","urd":"نیپال","kor":"네팔"}},"NL":{"currency":["EUR"],"callingCode":["31"],"region":"Europe","subregion":"Western Europe","flag":"flag-nl","name":{"common":"Netherlands","ces":"Nizozemsko","deu":"Niederlande","fra":"Pays-Bas","hrv":"Nizozemska","ita":"Paesi Bassi","jpn":"オランダ","nld":"Nederland","por":"Holanda","rus":"Нидерланды","slk":"Holansko","spa":"Países Bajos","fin":"Alankomaat","est":"Holland","zho":"荷兰","pol":"Holandia","urd":"نیدرلینڈز","kor":"네덜란드"}},"NC":{"currency":["XPF"],"callingCode":["687"],"region":"Oceania","subregion":"Melanesia","flag":"flag-nc","name":{"common":"New Caledonia","ces":"Nová Kaledonie","deu":"Neukaledonien","fra":"Nouvelle-Calédonie","hrv":"Nova Kaledonija","ita":"Nuova Caledonia","jpn":"ニューカレドニア","nld":"Nieuw-Caledonië","por":"Nova Caledónia","rus":"Новая Каледония","slk":"Nová Kaledónia","spa":"Nueva Caledonia","fin":"Uusi-Kaledonia","est":"Uus-Kaledoonia","zho":"新喀里多尼亚","pol":"Nowa Kaledonia","urd":"نیو کیلیڈونیا","kor":"누벨칼레도니"}},"NZ":{"currency":["NZD"],"callingCode":["64"],"region":"Oceania","subregion":"Australia and New Zealand","flag":"flag-nz","name":{"common":"New Zealand","ces":"Nový Zéland","deu":"Neuseeland","fra":"Nouvelle-Zélande","hrv":"Novi Zeland","ita":"Nuova Zelanda","jpn":"ニュージーランド","nld":"Nieuw-Zeeland","por":"Nova Zelândia","rus":"Новая Зеландия","slk":"Nový Zéland","spa":"Nueva Zelanda","fin":"Uusi-Seelanti","est":"Uus-Meremaa","zho":"新西兰","pol":"Nowa Zelandia","urd":"نیوزی لینڈ","kor":"뉴질랜드"}},"NI":{"currency":["NIO"],"callingCode":["505"],"region":"Americas","subregion":"Central America","flag":"flag-ni","name":{"common":"Nicaragua","ces":"Nikaragua","deu":"Nicaragua","fra":"Nicaragua","hrv":"Nikaragva","ita":"Nicaragua","jpn":"ニカラグア","nld":"Nicaragua","por":"Nicarágua","rus":"Никарагуа","slk":"Nikaragua","spa":"Nicaragua","fin":"Nicaragua","est":"Nicaragua","zho":"尼加拉瓜","pol":"Nikaragua","urd":"نکاراگوا","kor":"니카라과"}},"NE":{"currency":["XOF"],"callingCode":["227"],"region":"Africa","subregion":"Western Africa","flag":"flag-ne","name":{"common":"Niger","ces":"Niger","deu":"Niger","fra":"Niger","hrv":"Niger","ita":"Niger","jpn":"ニジェール","nld":"Niger","por":"Níger","rus":"Нигер","slk":"Niger","spa":"Níger","fin":"Niger","est":"Niger","zho":"尼日尔","pol":"Niger","urd":"نائجر","kor":"니제르"}},"NG":{"currency":["NGN"],"callingCode":["234"],"region":"Africa","subregion":"Western Africa","flag":"flag-ng","name":{"common":"Nigeria","ces":"Nigérie","deu":"Nigeria","fra":"Nigéria","hrv":"Nigerija","ita":"Nigeria","jpn":"ナイジェリア","nld":"Nigeria","por":"Nigéria","rus":"Нигерия","slk":"Nigéria","spa":"Nigeria","fin":"Nigeria","est":"Nigeeria","zho":"尼日利亚","pol":"Nigeria","urd":"نائجیریا","kor":"나이지리아"}},"NU":{"currency":["NZD"],"callingCode":["683"],"region":"Oceania","subregion":"Polynesia","flag":"flag-nu","name":{"common":"Niue","ces":"Niue","deu":"Niue","fra":"Niue","hrv":"Niue","ita":"Niue","jpn":"ニウエ","nld":"Niue","por":"Niue","rus":"Ниуэ","slk":"Niue","spa":"Niue","fin":"Niue","est":"Niue","zho":"纽埃","pol":"Niue","urd":"نیووے","kor":"니우에"}},"NF":{"currency":["AUD"],"callingCode":["672"],"region":"Oceania","subregion":"Australia and New Zealand","flag":"flag-nf","name":{"common":"Norfolk Island","ces":"Norfolk","deu":"Norfolkinsel","fra":"Île Norfolk","hrv":"Otok Norfolk","ita":"Isola Norfolk","jpn":"ノーフォーク島","nld":"Norfolkeiland","por":"Ilha Norfolk","rus":"Норфолк","slk":"Norfolk","spa":"Isla de Norfolk","fin":"Norfolkinsaari","est":"Norfolk","zho":"诺福克岛","pol":"Wyspa Norfolk","urd":"جزیرہ نورفک","kor":"노퍽 섬"}},"KP":{"currency":["KPW"],"callingCode":["850"],"region":"Asia","subregion":"Eastern Asia","flag":"flag-kp","name":{"common":"North Korea","ces":"Severní Korea","deu":"Nordkorea","fra":"Corée du Nord","hrv":"Sjeverna Koreja","ita":"Corea del Nord","jpn":"朝鮮民主主義人民共和国","nld":"Noord-Korea","por":"Coreia do Norte","rus":"Северная Корея","slk":"Kórejská ľudovodemokratická republika (KĽR, Severná Kórea)","spa":"Corea del Norte","fin":"Pohjois-Korea","est":"Põhja-Korea","zho":"朝鲜","pol":"Korea Północna","urd":"شمالی کوریا","kor":"조선"}},"MP":{"currency":["USD"],"callingCode":["1670"],"region":"Oceania","subregion":"Micronesia","flag":"flag-mp","name":{"common":"Northern Mariana Islands","ces":"Severní Mariany","deu":"Nördliche Marianen","fra":"Îles Mariannes du Nord","hrv":"Sjevernomarijanski otoci","ita":"Isole Marianne Settentrionali","jpn":"北マリアナ諸島","nld":"Noordelijke Marianeneilanden","por":"Marianas Setentrionais","rus":"Северные Марианские острова","slk":"Severné Mariány","spa":"Islas Marianas del Norte","fin":"Pohjois-Mariaanit","est":"Põhja-Mariaanid","zho":"北马里亚纳群岛","pol":"Mariany Północne","urd":"جزائر شمالی ماریانا","kor":"북마리아나 제도"}},"NO":{"currency":["NOK"],"callingCode":["47"],"region":"Europe","subregion":"Northern Europe","flag":"flag-no","name":{"common":"Norway","ces":"Norsko","deu":"Norwegen","fra":"Norvège","hrv":"Norveška","ita":"Norvegia","jpn":"ノルウェー","nld":"Noorwegen","por":"Noruega","rus":"Норвегия","slk":"Nórsko","spa":"Noruega","fin":"Norja","est":"Norra","zho":"挪威","pol":"Norwegia","urd":"ناروے","kor":"노르웨이"}},"OM":{"currency":["OMR"],"callingCode":["968"],"region":"Asia","subregion":"Western Asia","flag":"flag-om","name":{"common":"Oman","ces":"Omán","deu":"Oman","fra":"Oman","hrv":"Oman","ita":"oman","jpn":"オマーン","nld":"Oman","por":"Omã","rus":"Оман","slk":"Omán","spa":"Omán","fin":"Oman","est":"Omaan","zho":"阿曼","pol":"Oman","urd":"عمان","kor":"오만"}},"PK":{"currency":["PKR"],"callingCode":["92"],"region":"Asia","subregion":"Southern Asia","flag":"flag-pk","name":{"common":"Pakistan","ces":"Pákistán","deu":"Pakistan","fra":"Pakistan","hrv":"Pakistan","ita":"Pakistan","jpn":"パキスタン","nld":"Pakistan","por":"Paquistão","rus":"Пакистан","slk":"Pakistan","spa":"Pakistán","fin":"Pakistan","est":"Pakistan","zho":"巴基斯坦","pol":"Pakistan","urd":"پاکستان","kor":"파키스탄"}},"PW":{"currency":["USD"],"callingCode":["680"],"region":"Oceania","subregion":"Micronesia","flag":"flag-pw","name":{"common":"Palau","ces":"Palau","deu":"Palau","fra":"Palaos (Palau)","hrv":"Palau","ita":"Palau","jpn":"パラオ","nld":"Palau","por":"Palau","rus":"Палау","slk":"Palau","spa":"Palau","fin":"Palau","est":"Belau","zho":"帕劳","pol":"Palau","urd":"پلاؤ","kor":"팔라우"}},"PS":{"currency":["ILS"],"callingCode":["970"],"region":"Asia","subregion":"Western Asia","flag":"flag-ps","name":{"common":"Palestine","ces":"Palestina","deu":"Palästina","fra":"Palestine","hrv":"Palestina","ita":"Palestina","jpn":"パレスチナ","nld":"Palestijnse gebieden","por":"Palestina","rus":"Палестина","slk":"Palestína","spa":"Palestina","fin":"Palestiina","est":"Palestiina","zho":"巴勒斯坦","pol":"Palestyna","urd":"فلسطین","kor":"팔레스타인"}},"PA":{"currency":["PAB","USD"],"callingCode":["507"],"region":"Americas","subregion":"Central America","flag":"flag-pa","name":{"common":"Panama","ces":"Panama","deu":"Panama","fra":"Panama","hrv":"Panama","ita":"Panama","jpn":"パナマ","nld":"Panama","por":"Panamá","rus":"Панама","slk":"Panama","spa":"Panamá","fin":"Panama","est":"Panama","zho":"巴拿马","pol":"Panama","urd":"پاناما","kor":"파나마"}},"PG":{"currency":["PGK"],"callingCode":["675"],"region":"Oceania","subregion":"Melanesia","flag":"flag-pg","name":{"common":"Papua New Guinea","ces":"Papua-Nová Guinea","deu":"Papua-Neuguinea","fra":"Papouasie-Nouvelle-Guinée","hrv":"Papua Nova Gvineja","ita":"Papua Nuova Guinea","jpn":"パプアニューギニア","nld":"Papoea-Nieuw-Guinea","por":"Papua Nova Guiné","rus":"Папуа — Новая Гвинея","slk":"Papua-Nová Guinea","spa":"Papúa Nueva Guinea","fin":"Papua-Uusi-Guinea","est":"Paapua Uus-Guinea","zho":"巴布亚新几内亚","pol":"Papua-Nowa Gwinea","urd":"پاپوا نیو گنی","kor":"파푸아뉴기니"}},"PY":{"currency":["PYG"],"callingCode":["595"],"region":"Americas","subregion":"South America","flag":"flag-py","name":{"common":"Paraguay","ces":"Paraguay","deu":"Paraguay","fra":"Paraguay","hrv":"Paragvaj","ita":"Paraguay","jpn":"パラグアイ","nld":"Paraguay","por":"Paraguai","rus":"Парагвай","slk":"Paraguaj","spa":"Paraguay","fin":"Paraguay","est":"Paraguay","zho":"巴拉圭","pol":"Paragwaj","urd":"پیراگوئے","kor":"파라과이"}},"PE":{"currency":["PEN"],"callingCode":["51"],"region":"Americas","subregion":"South America","flag":"flag-pe","name":{"common":"Peru","ces":"Peru","deu":"Peru","fra":"Pérou","hrv":"Peru","ita":"Perù","jpn":"ペルー","nld":"Peru","por":"Perú","rus":"Перу","slk":"Peru","spa":"Perú","fin":"Peru","est":"Peruu","zho":"秘鲁","pol":"Peru","urd":"پیرو","kor":"페루"}},"PH":{"currency":["PHP"],"callingCode":["63"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-ph","name":{"common":"Philippines","ces":"Filipíny","deu":"Philippinen","fra":"Philippines","hrv":"Filipini","ita":"Filippine","jpn":"フィリピン","nld":"Filipijnen","por":"Filipinas","rus":"Филиппины","slk":"Filipíny","spa":"Filipinas","fin":"Filippiinit","est":"Filipiinid","zho":"菲律宾","pol":"Filipiny","urd":"فلپائن","kor":"필리핀"}},"PN":{"currency":["NZD"],"callingCode":["64"],"region":"Oceania","subregion":"Polynesia","flag":"flag-pn","name":{"common":"Pitcairn Islands","ces":"Pitcairnovy ostrovy","deu":"Pitcairninseln","fra":"Îles Pitcairn","hrv":"Pitcairnovo otočje","ita":"Isole Pitcairn","jpn":"ピトケアン","nld":"Pitcairneilanden","por":"Ilhas Pitcairn","rus":"Острова Питкэрн","slk":"Pitcairnove ostrovy","spa":"Islas Pitcairn","fin":"Pitcairn","est":"Pitcairn","zho":"皮特凯恩群岛","pol":"Pitcairn","urd":"جزائر پٹکیرن","kor":"핏케언 제도"}},"PL":{"currency":["PLN"],"callingCode":["48"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-pl","name":{"common":"Poland","ces":"Polsko","deu":"Polen","fra":"Pologne","hrv":"Poljska","ita":"Polonia","jpn":"ポーランド","nld":"Polen","por":"Polónia","rus":"Польша","slk":"Poľsko","spa":"Polonia","fin":"Puola","est":"Poola","zho":"波兰","pol":"Polska","urd":"پولینڈ","kor":"폴란드"}},"PT":{"currency":["EUR"],"callingCode":["351"],"region":"Europe","subregion":"Southern Europe","flag":"flag-pt","name":{"common":"Portugal","ces":"Portugalsko","deu":"Portugal","fra":"Portugal","hrv":"Portugal","ita":"Portogallo","jpn":"ポルトガル","nld":"Portugal","por":"Portugal","rus":"Португалия","slk":"Portugalsko","spa":"Portugal","fin":"Portugali","est":"Portugal","zho":"葡萄牙","pol":"Portugalia","urd":"پرتگال","kor":"포르투갈"}},"PR":{"currency":["USD"],"callingCode":["1787","1939"],"region":"Americas","subregion":"Caribbean","flag":"flag-pr","name":{"common":"Puerto Rico","ces":"Portoriko","deu":"Puerto Rico","fra":"Porto Rico","hrv":"Portoriko","ita":"Porto Rico","jpn":"プエルトリコ","nld":"Puerto Rico","por":"Porto Rico","rus":"Пуэрто-Рико","slk":"Portoriko","spa":"Puerto Rico","fin":"Puerto Rico","est":"Puerto Rico","zho":"波多黎各","pol":"Portoryko","urd":"پورٹو ریکو","kor":"푸에르토리코"}},"QA":{"currency":["QAR"],"callingCode":["974"],"region":"Asia","subregion":"Western Asia","flag":"flag-qa","name":{"common":"Qatar","ces":"Katar","deu":"Katar","fra":"Qatar","hrv":"Katar","ita":"Qatar","jpn":"カタール","nld":"Qatar","por":"Catar","rus":"Катар","slk":"Katar","spa":"Catar","fin":"Qatar","est":"Katar","zho":"卡塔尔","pol":"Katar","urd":"قطر","kor":"카타르"}},"CG":{"currency":["XAF"],"callingCode":["242"],"region":"Africa","subregion":"Middle Africa","flag":"flag-cg","name":{"common":"Republic of the Congo","ces":"Kongo","cym":"Gweriniaeth y Congo","deu":"Kongo","fra":"Congo","hrv":"Kongo","ita":"Congo","jpn":"コンゴ共和国","nld":"Congo","por":"Congo","rus":"Республика Конго","slk":"Kongo","spa":"Congo","fin":"Kongo-Brazzaville","est":"Kongo Vabariik","zho":"刚果","pol":"Kongo","urd":"جمہوریہ کانگو","kor":"콩고"}},"RO":{"currency":["RON"],"callingCode":["40"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-ro","name":{"common":"Romania","ces":"Rumunsko","deu":"Rumänien","fra":"Roumanie","hrv":"Rumunjska","ita":"Romania","jpn":"ルーマニア","nld":"Roemenië","por":"Roménia","rus":"Румыния","slk":"Rumunsko","spa":"Rumania","fin":"Romania","est":"Rumeenia","zho":"罗马尼亚","pol":"Rumunia","urd":"رومانیہ","kor":"루마니아"}},"RU":{"currency":["RUB"],"callingCode":["7"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-ru","name":{"common":"Russia","ces":"Rusko","deu":"Russland","fra":"Russie","hrv":"Rusija","ita":"Russia","jpn":"ロシア連邦","nld":"Rusland","por":"Rússia","rus":"Россия","slk":"Rusko","spa":"Rusia","fin":"Venäjä","est":"Venemaa","zho":"俄罗斯","pol":"Rosja","urd":"روس","kor":"러시아"}},"RW":{"currency":["RWF"],"callingCode":["250"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-rw","name":{"common":"Rwanda","ces":"Rwanda","deu":"Ruanda","fra":"Rwanda","hrv":"Ruanda","ita":"Ruanda","jpn":"ルワンダ","nld":"Rwanda","por":"Ruanda","rus":"Руанда","slk":"Rwanda","spa":"Ruanda","fin":"Ruanda","est":"Rwanda","zho":"卢旺达","pol":"Rwanda","urd":"روانڈا","kor":"르완다"}},"RE":{"currency":["EUR"],"callingCode":["262"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-re","name":{"common":"Réunion","ces":"Réunion","deu":"Réunion","fra":"Réunion","hrv":"Réunion","ita":"Riunione","jpn":"レユニオン","nld":"Réunion","por":"Reunião","rus":"Реюньон","slk":"Réunion","spa":"Reunión","fin":"Réunion","est":"Réunion","zho":"留尼旺岛","pol":"Reunion","urd":"رے یونیوں","kor":"레위니옹"}},"BL":{"currency":["EUR"],"callingCode":["590"],"region":"Americas","subregion":"Caribbean","flag":"flag-bl","name":{"common":"Saint Barthélemy","ces":"Svatý Bartoloměj","deu":"Saint-Barthélemy","fra":"Saint-Barthélemy","hrv":"Saint Barthélemy","ita":"Antille Francesi","jpn":"サン・バルテルミー","nld":"Saint Barthélemy","por":"São Bartolomeu","rus":"Сен-Бартелеми","slk":"Svätý Bartolomej","spa":"San Bartolomé","fin":"Saint-Barthélemy","est":"Saint-Barthélemy","zho":"圣巴泰勒米","pol":"Saint-Barthélemy","urd":"سینٹ بارتھیملے","kor":"생바르텔레미"}},"SH":{"currency":["SHP","GBP"],"callingCode":["290","247"],"region":"Africa","subregion":"Western Africa","flag":"flag-sh","name":{"common":"Saint Helena, Ascension and Tristan da Cunha","ces":"Svatá Helena, Ascension a Tristan da Cunha","deu":"St. Helena, Ascension und Tristan da Cunha","fra":"Sainte-Hélène, Ascension et Tristan da Cunha","hrv":"Sveta Helena","ita":"Sant'Elena, Ascensione e Tristan da Cunha","jpn":"セントヘレナ・アセンションおよびトリスタンダクーニャ","nld":"Sint-Helena, Ascension en Tristan da Cunha","por":"Santa Helena, Ascensão e Tristão da Cunha","rus":"Острова Святой Елены, Вознесения и Тристан-да-Кунья","slk":"Svätá Helena (zámorské územie)","spa":"Santa Elena, Ascensión y Tristán de Acuña","fin":"Saint Helena, Ascension ja Tristan da Cunha","est":"Saint Helena, Ascension ja Tristan da Cunha","zho":"圣赫勒拿、阿森松和特里斯坦-达库尼亚","pol":"Wyspa Świętej Heleny, Wyspa Wniebowstąpienia i Tristan da Cunha","urd":"سینٹ ہلینا، اسینشن و ترسٹان دا کونیا","kor":"세인트헬레나"}},"KN":{"currency":["XCD"],"callingCode":["1869"],"region":"Americas","subregion":"Caribbean","flag":"flag-kn","name":{"common":"Saint Kitts and Nevis","ces":"Svatý Kryštof a Nevis","deu":"St. Kitts und Nevis","fra":"Saint-Christophe-et-Niévès","hrv":"Sveti Kristof i Nevis","ita":"Saint Kitts e Nevis","jpn":"セントクリストファー・ネイビス","nld":"Saint Kitts en Nevis","por":"São Cristóvão e Nevis","rus":"Сент-Китс и Невис","slk":"Svätý Krištof a Nevis","spa":"San Cristóbal y Nieves","fin":"Saint Kitts ja Nevis","est":"Saint Kitts ja Nevis","zho":"圣基茨和尼维斯","pol":"Saint Kitts i Nevis","urd":"سینٹ کیٹز و ناویس","kor":"세인트키츠 네비스"}},"LC":{"currency":["XCD"],"callingCode":["1758"],"region":"Americas","subregion":"Caribbean","flag":"flag-lc","name":{"common":"Saint Lucia","ces":"Svatá Lucie","deu":"St. Lucia","fra":"Sainte-Lucie","hrv":"Sveta Lucija","ita":"Santa Lucia","jpn":"セントルシア","nld":"Saint Lucia","por":"Santa Lúcia","rus":"Сент-Люсия","slk":"Svätá Lucia","spa":"Santa Lucía","fin":"Saint Lucia","est":"Saint Lucia","zho":"圣卢西亚","pol":"Saint Lucia","urd":"سینٹ لوسیا","kor":"세인트루시아"}},"MF":{"currency":["EUR"],"callingCode":["590"],"region":"Americas","subregion":"Caribbean","flag":"flag-mf","name":{"common":"Saint Martin","ces":"Svatý Martin (Francie)","deu":"Saint-Martin","fra":"Saint-Martin","hrv":"Sveti Martin","ita":"Saint Martin","jpn":"サン・マルタン(フランス領)","nld":"Saint-Martin","por":"São Martinho","rus":"Сен-Мартен","slk":"Saint-Martin","spa":"Saint Martin","fin":"Saint-Martin","est":"Saint-Martin","zho":"圣马丁","pol":"Saint-Martin","urd":"سینٹ مارٹن","kor":"생마르탱"}},"PM":{"currency":["EUR"],"callingCode":["508"],"region":"Americas","subregion":"North America","flag":"flag-pm","name":{"common":"Saint Pierre and Miquelon","ces":"Saint-Pierre a Miquelon","deu":"St. Pierre und Miquelon","fra":"Saint-Pierre-et-Miquelon","hrv":"Sveti Petar i Mikelon","ita":"Saint-Pierre e Miquelon","jpn":"サンピエール島・ミクロン島","nld":"Saint Pierre en Miquelon","por":"Saint-Pierre e Miquelon","rus":"Сен-Пьер и Микелон","slk":"Saint Pierre a Miquelon","spa":"San Pedro y Miquelón","fin":"Saint-Pierre ja Miquelon","est":"Saint-Pierre ja Miquelon","zho":"圣皮埃尔和密克隆","pol":"Saint-Pierre i Miquelon","urd":"سینٹ پیئر و میکیلون","kor":"생피에르 미클롱"}},"VC":{"currency":["XCD"],"callingCode":["1784"],"region":"Americas","subregion":"Caribbean","flag":"flag-vc","name":{"common":"Saint Vincent and the Grenadines","ces":"Svatý Vincenc a Grenadiny","deu":"St. Vincent und die Grenadinen","fra":"Saint-Vincent-et-les-Grenadines","hrv":"Sveti Vincent i Grenadini","ita":"Saint Vincent e Grenadine","jpn":"セントビンセントおよびグレナディーン諸島","nld":"Saint Vincent en de Grenadines","por":"São Vincente e Granadinas","rus":"Сент-Винсент и Гренадины","slk":"Svätý Vincent a Grenadíny","spa":"San Vicente y Granadinas","fin":"Saint Vincent ja Grenadiinit","est":"Saint Vincent","zho":"圣文森特和格林纳丁斯","pol":"Saint Vincent i Grenadyny","urd":"سینٹ وینسینٹ و گریناڈائنز","kor":"세인트빈센트 그레나딘"}},"WS":{"currency":["WST"],"callingCode":["685"],"region":"Oceania","subregion":"Polynesia","flag":"flag-ws","name":{"common":"Samoa","ces":"Samoa","deu":"Samoa","fra":"Samoa","hrv":"Samoa","ita":"Samoa","jpn":"サモア","nld":"Samoa","por":"Samoa","rus":"Самоа","slk":"Samoa","spa":"Samoa","fin":"Samoa","est":"Samoa","zho":"萨摩亚","pol":"Samoa","urd":"سامووا","kor":"사모아"}},"SM":{"currency":["EUR"],"callingCode":["378"],"region":"Europe","subregion":"Southern Europe","flag":"flag-sm","name":{"common":"San Marino","ces":"San Marino","deu":"San Marino","fra":"Saint-Marin","hrv":"San Marino","ita":"San Marino","jpn":"サンマリノ","nld":"San Marino","por":"San Marino","rus":"Сан-Марино","slk":"San Maríno","spa":"San Marino","fin":"San Marino","est":"San Marino","zho":"圣马力诺","pol":"San Marino","urd":"سان مارینو","kor":"산마리노"}},"SA":{"currency":["SAR"],"callingCode":["966"],"region":"Asia","subregion":"Western Asia","flag":"flag-sa","name":{"common":"Saudi Arabia","ces":"Saúdská Arábie","deu":"Saudi-Arabien","fra":"Arabie Saoudite","hrv":"Saudijska Arabija","ita":"Arabia Saudita","jpn":"サウジアラビア","nld":"Saoedi-Arabië","por":"Arábia Saudita","rus":"Саудовская Аравия","slk":"Saudská Arábia","spa":"Arabia Saudí","fin":"Saudi-Arabia","est":"Saudi Araabia","zho":"沙特阿拉伯","pol":"Arabia Saudyjska","urd":"سعودی عرب","kor":"사우디아라비아"}},"SN":{"currency":["XOF"],"callingCode":["221"],"region":"Africa","subregion":"Western Africa","flag":"flag-sn","name":{"common":"Senegal","ces":"Senegal","deu":"Senegal","fra":"Sénégal","hrv":"Senegal","ita":"Senegal","jpn":"セネガル","nld":"Senegal","por":"Senegal","rus":"Сенегал","slk":"Senegal","spa":"Senegal","fin":"Senegal","est":"Senegal","zho":"塞内加尔","pol":"Senegal","urd":"سینیگال","kor":"세네갈"}},"RS":{"currency":["RSD"],"callingCode":["381"],"region":"Europe","subregion":"Southern Europe","flag":"flag-rs","name":{"common":"Serbia","ces":"Srbsko","deu":"Serbien","fra":"Serbie","hrv":"Srbija","ita":"Serbia","jpn":"セルビア","nld":"Servië","por":"Sérvia","rus":"Сербия","slk":"Srbsko","spa":"Serbia","fin":"Serbia","est":"Serbia","zho":"塞尔维亚","pol":"Serbia","urd":"سربیا","kor":"세르비아"}},"SC":{"currency":["SCR"],"callingCode":["248"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-sc","name":{"common":"Seychelles","ces":"Seychely","deu":"Seychellen","fra":"Seychelles","hrv":"Sejšeli","ita":"Seychelles","jpn":"セーシェル","nld":"Seychellen","por":"Seicheles","rus":"Сейшельские Острова","slk":"Seychely","spa":"Seychelles","fin":"Seychellit","est":"Seišellid","zho":"塞舌尔","pol":"Seszele","urd":"سیچیلیس","kor":"세이셸"}},"SL":{"currency":["SLL"],"callingCode":["232"],"region":"Africa","subregion":"Western Africa","flag":"flag-sl","name":{"common":"Sierra Leone","ces":"Sierra Leone","deu":"Sierra Leone","fra":"Sierra Leone","hrv":"Sijera Leone","ita":"Sierra Leone","jpn":"シエラレオネ","nld":"Sierra Leone","por":"Serra Leoa","rus":"Сьерра-Леоне","slk":"Sierra Leone","spa":"Sierra Leone","fin":"Sierra Leone","est":"Sierra Leone","zho":"塞拉利昂","pol":"Sierra Leone","urd":"سیرالیون","kor":"시에라리온"}},"SG":{"currency":["SGD"],"callingCode":["65"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-sg","name":{"common":"Singapore","ces":"Singapur","deu":"Singapur","fra":"Singapour","hrv":"Singapur","ita":"Singapore","jpn":"シンガポール","nld":"Singapore","por":"Singapura","rus":"Сингапур","slk":"Singapur","spa":"Singapur","fin":"Singapore","est":"Singapur","pol":"Singapur","urd":"سنگاپور","kor":"싱가포르"}},"SX":{"currency":["ANG"],"callingCode":["1721"],"region":"Americas","subregion":"Caribbean","flag":"flag-sx","name":{"common":"Sint Maarten","ces":"Svatý Martin (Nizozemsko)","deu":"Sint Maarten","fra":"Saint-Martin","hrv":"Sveti Martin","ita":"Sint Maarten","jpn":"シント・マールテン","nld":"Sint Maarten","por":"São Martinho","rus":"Синт-Мартен","slk":"Sint Maarten","spa":"Sint Maarten","fin":"Sint Maarten","est":"Sint Maarten","zho":"圣马丁岛","pol":"Sint Maarten","urd":"سنٹ مارٹن","kor":"신트마르턴"}},"SK":{"currency":["EUR"],"callingCode":["421"],"region":"Europe","subregion":"Central Europe","flag":"flag-sk","name":{"common":"Slovakia","ces":"Slovensko","deu":"Slowakei","fra":"Slovaquie","hrv":"Slovačka","ita":"Slovacchia","jpn":"スロバキア","nld":"Slowakije","por":"Eslováquia","rus":"Словакия","slk":"Slovensko","spa":"República Eslovaca","fin":"Slovakia","est":"Slovakkia","zho":"斯洛伐克","pol":"Słowacja","urd":"سلوواکیہ","kor":"슬로바키아"}},"SI":{"currency":["EUR"],"callingCode":["386"],"region":"Europe","subregion":"Southern Europe","flag":"flag-si","name":{"common":"Slovenia","ces":"Slovinsko","deu":"Slowenien","fra":"Slovénie","hrv":"Slovenija","ita":"Slovenia","jpn":"スロベニア","nld":"Slovenië","por":"Eslovénia","rus":"Словения","slk":"Slovinsko","spa":"Eslovenia","fin":"Slovenia","est":"Sloveenia","zho":"斯洛文尼亚","pol":"Słowenia","urd":"سلووینیا","kor":"슬로베니아"}},"SB":{"currency":["SBD"],"callingCode":["677"],"region":"Oceania","subregion":"Melanesia","flag":"flag-sb","name":{"common":"Solomon Islands","ces":"Šalamounovy ostrovy","deu":"Salomonen","fra":"Îles Salomon","hrv":"Solomonski Otoci","ita":"Isole Salomone","jpn":"ソロモン諸島","nld":"Salomonseilanden","por":"Ilhas Salomão","rus":"Соломоновы Острова","slk":"Salomonove ostrovy","spa":"Islas Salomón","fin":"Salomonsaaret","est":"Saalomoni Saared","zho":"所罗门群岛","pol":"Wyspy Salomona","urd":"جزائر سلیمان","kor":"솔로몬 제도"}},"SO":{"currency":["SOS"],"callingCode":["252"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-so","name":{"common":"Somalia","ces":"Somálsko","deu":"Somalia","fra":"Somalie","hrv":"Somalija","ita":"Somalia","jpn":"ソマリア","nld":"Somalië","por":"Somália","rus":"Сомали","slk":"Somálsko","spa":"Somalia","fin":"Somalia","est":"Somaalia","zho":"索马里","pol":"Somalia","urd":"صومالیہ","kor":"소말리아"}},"ZA":{"currency":["ZAR"],"callingCode":["27"],"region":"Africa","subregion":"Southern Africa","flag":"flag-za","name":{"common":"South Africa","ces":"Jihoafrická republika","deu":"Südafrika","fra":"Afrique du Sud","hrv":"Južnoafrička Republika","ita":"Sud Africa","jpn":"南アフリカ","nld":"Zuid-Afrika","por":"África do Sul","rus":"Южно-Африканская Республика","slk":"Juhoafrická republika","spa":"República de Sudáfrica","fin":"Etelä-Afrikka","est":"Lõuna-Aafrika Vabariik","zho":"南非","pol":"Południowa Afryka","urd":"جنوبی افریقا","kor":"남아프리카"}},"GS":{"currency":["GBP"],"callingCode":["500"],"region":"Antarctic","subregion":"","flag":"flag-gs","name":{"common":"South Georgia","ces":"Jižní Georgie a Jižní Sandwichovy ostrovy","deu":"Südgeorgien und die Südlichen Sandwichinseln","fra":"Géorgie du Sud-et-les Îles Sandwich du Sud","hrv":"Južna Georgija i otočje Južni Sandwich","ita":"Georgia del Sud e Isole Sandwich Meridionali","jpn":"サウスジョージア・サウスサンドウィッチ諸島","nld":"Zuid-Georgia en Zuidelijke Sandwicheilanden","por":"Ilhas Geórgia do Sul e Sandwich do Sul","rus":"Южная Георгия и Южные Сандвичевы острова","slk":"Južná Georgia a Južné Sandwichove ostrovy","spa":"Islas Georgias del Sur y Sandwich del Sur","fin":"Etelä-Georgia ja Eteläiset Sandwichsaaret","est":"Lõuna-Georgia ja Lõuna-Sandwichi saared","zho":"南乔治亚","pol":"Georgia Południowa i Sandwich Południowy","urd":"جنوبی جارجیا","kor":"조지아"}},"KR":{"currency":["KRW"],"callingCode":["82"],"region":"Asia","subregion":"Eastern Asia","flag":"flag-kr","name":{"common":"South Korea","ces":"Jižní Korea","deu":"Südkorea","fra":"Corée du Sud","hrv":"Južna Koreja","ita":"Corea del Sud","jpn":"韓国","nld":"Zuid-Korea","por":"Coreia do Sul","rus":"Южная Корея","slk":"Južná Kórea","spa":"Corea del Sur","fin":"Etelä-Korea","est":"Lõuna-Korea","zho":"韩国","pol":"Korea Południowa","urd":"جنوبی کوریا","kor":"한국"}},"SS":{"currency":["SSP"],"callingCode":["211"],"region":"Africa","subregion":"Middle Africa","flag":"flag-ss","name":{"common":"South Sudan","ces":"Jižní Súdán","deu":"Südsudan","fra":"Soudan du Sud","hrv":"Južni Sudan","ita":"Sudan del sud","jpn":"南スーダン","nld":"Zuid-Soedan","por":"Sudão do Sul","rus":"Южный Судан","slk":"Južný Sudán","spa":"Sudán del Sur","fin":"Etelä-Sudan","est":"Lõuna-Sudaan","zho":"南苏丹","pol":"Sudan","urd":"جنوبی سوڈان","kor":"남수단"}},"ES":{"currency":["EUR"],"callingCode":["34"],"region":"Europe","subregion":"Southern Europe","flag":"flag-es","name":{"common":"Spain","ces":"Španělsko","deu":"Spanien","fra":"Espagne","hrv":"Španjolska","ita":"Spagna","jpn":"スペイン","nld":"Spanje","por":"Espanha","rus":"Испания","slk":"Španielsko","spa":"España","fin":"Espanja","est":"Hispaania","zho":"西班牙","pol":"Hiszpania","urd":"ہسپانیہ","kor":"스페인"}},"LK":{"currency":["LKR"],"callingCode":["94"],"region":"Asia","subregion":"Southern Asia","flag":"flag-lk","name":{"common":"Sri Lanka","ces":"Srí Lanka","deu":"Sri Lanka","fra":"Sri Lanka","hrv":"Šri Lanka","ita":"Sri Lanka","jpn":"スリランカ","nld":"Sri Lanka","por":"Sri Lanka","rus":"Шри-Ланка","slk":"Srí Lanka","spa":"Sri Lanka","fin":"Sri Lanka","est":"Sri Lanka","zho":"斯里兰卡","pol":"Sri Lanka","urd":"سری لنکا","kor":"스리랑카"}},"SD":{"currency":["SDG"],"callingCode":["249"],"region":"Africa","subregion":"Northern Africa","flag":"flag-sd","name":{"common":"Sudan","ces":"Súdán","deu":"Sudan","fra":"Soudan","hrv":"Sudan","ita":"Sudan","jpn":"スーダン","nld":"Soedan","por":"Sudão","rus":"Судан","slk":"Sudán","spa":"Sudán","fin":"Sudan","est":"Sudaan","zho":"苏丹","pol":"Sudan","urd":"سودان","kor":"수단"}},"SR":{"currency":["SRD"],"callingCode":["597"],"region":"Americas","subregion":"South America","flag":"flag-sr","name":{"common":"Suriname","ces":"Surinam","deu":"Suriname","fra":"Surinam","hrv":"Surinam","ita":"Suriname","jpn":"スリナム","nld":"Suriname","por":"Suriname","rus":"Суринам","slk":"Surinam","spa":"Surinam","fin":"Suriname","est":"Suriname","zho":"苏里南","pol":"Surinam","urd":"سرینام","kor":"수리남"}},"SJ":{"currency":["NOK"],"callingCode":["4779"],"region":"Europe","subregion":"Northern Europe","flag":"flag-sj","name":{"common":"Svalbard and Jan Mayen","ces":"Špicberky a Jan Mayen","deu":"Spitzbergen und Jan Mayen","fra":"Svalbard et Jan Mayen","hrv":"Svalbard i Jan Mayen","ita":"Svalbard e Jan Mayen","jpn":"スヴァールバル諸島およびヤンマイエン島","nld":"Svalbard en Jan Mayen","por":"Ilhas Svalbard e Jan Mayen","rus":"Шпицберген и Ян-Майен","slk":"Svalbard a Jan Mayen","spa":"Islas Svalbard y Jan Mayen","fin":"Huippuvuoret","est":"Svalbard","zho":"斯瓦尔巴特","pol":"Svalbard i Jan Mayen","urd":"سوالبارڈ اور جان میئن","kor":"스발바르 얀마옌 제도"}},"SE":{"currency":["SEK"],"callingCode":["46"],"region":"Europe","subregion":"Northern Europe","flag":"flag-se","name":{"common":"Sweden","ces":"Švédsko","deu":"Schweden","fra":"Suède","hrv":"Švedska","ita":"Svezia","jpn":"スウェーデン","nld":"Zweden","por":"Suécia","rus":"Швеция","slk":"Švédsko","spa":"Suecia","fin":"Ruotsi","est":"Rootsi","zho":"瑞典","pol":"Szwecja","urd":"سویڈن","kor":"스웨덴"}},"CH":{"currency":["CHF"],"callingCode":["41"],"region":"Europe","subregion":"Western Europe","flag":"flag-ch","name":{"common":"Switzerland","ces":"Švýcarsko","deu":"Schweiz","fra":"Suisse","hrv":"Švicarska","ita":"Svizzera","jpn":"スイス","nld":"Zwitserland","por":"Suíça","rus":"Швейцария","slk":"Švajčiarsko","spa":"Suiza","fin":"Sveitsi","est":"Šveits","zho":"瑞士","pol":"Szwajcaria","urd":"سویٹذرلینڈ","kor":"스위스"}},"SY":{"currency":["SYP"],"callingCode":["963"],"region":"Asia","subregion":"Western Asia","flag":"flag-sy","name":{"common":"Syria","ces":"Sýrie","deu":"Syrien","fra":"Syrie","hrv":"Sirija","ita":"Siria","jpn":"シリア・アラブ共和国","nld":"Syrië","por":"Síria","rus":"Сирия","slk":"Sýria","spa":"Siria","fin":"Syyria","est":"Süüria","zho":"叙利亚","pol":"Syria","urd":"سوریہ","kor":"시리아"}},"ST":{"currency":["STD"],"callingCode":["239"],"region":"Africa","subregion":"Middle Africa","flag":"flag-st","name":{"common":"São Tomé and Príncipe","ces":"Svatý Tomáš a Princův ostrov","deu":"São Tomé und Príncipe","fra":"São Tomé et Príncipe","hrv":"Sveti Toma i Princip","ita":"São Tomé e Príncipe","jpn":"サントメ・プリンシペ","nld":"Sao Tomé en Principe","por":"São Tomé e Príncipe","spa":"Santo Tomé y Príncipe","rus":"Сан-Томе и Принсипи","slk":"Svätý Tomáš a Princov ostrov","fin":"São Téme ja Príncipe","est":"São Tomé ja Príncipe","zho":"圣多美和普林西比","pol":"Wyspy Świętego Tomasza i Książęca","urd":"ساؤ ٹومے و پرنسپے","kor":"상투메 프린시페"}},"TW":{"currency":["TWD"],"callingCode":["886"],"region":"Asia","subregion":"Eastern Asia","flag":"flag-tw","name":{"common":"Taiwan","ces":"Tchaj-wan","deu":"Taiwan","fra":"Taïwan","hrv":"Tajvan","ita":"Taiwan","jpn":"台湾","nld":"Taiwan","por":"Ilha Formosa","rus":"Тайвань","slk":"Taiwan","spa":"Taiwán","fin":"Taiwan","est":"Taiwan","pol":"Tajwan","urd":"تائیوان","kor":"대만"}},"TJ":{"currency":["TJS"],"callingCode":["992"],"region":"Asia","subregion":"Central Asia","flag":"flag-tj","name":{"common":"Tajikistan","ces":"Tádžikistán","deu":"Tadschikistan","fra":"Tadjikistan","hrv":"Tađikistan","ita":"Tagikistan","jpn":"タジキスタン","nld":"Tadzjikistan","por":"Tajiquistão","rus":"Таджикистан","slk":"Tadžikistan","spa":"Tayikistán","fin":"Tadžikistan","est":"Tadžikistan","zho":"塔吉克斯坦","pol":"Tadżykistan","urd":"تاجکستان","kor":"타지키스탄"}},"TZ":{"currency":["TZS"],"callingCode":["255"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-tz","name":{"common":"Tanzania","ces":"Tanzanie","deu":"Tansania","fra":"Tanzanie","hrv":"Tanzanija","ita":"Tanzania","jpn":"タンザニア","nld":"Tanzania","por":"Tanzânia","rus":"Танзания","slk":"Tanzánia","spa":"Tanzania","fin":"Tansania","est":"Tansaania","zho":"坦桑尼亚","pol":"Tanzania","urd":"تنزانیہ","kor":"탄자니아"}},"TH":{"currency":["THB"],"callingCode":["66"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-th","name":{"common":"Thailand","ces":"Thajsko","deu":"Thailand","fra":"Thaïlande","hrv":"Tajland","ita":"Tailandia","jpn":"タイ","nld":"Thailand","por":"Tailândia","rus":"Таиланд","slk":"Thajsko","spa":"Tailandia","fin":"Thaimaa","est":"Tai","zho":"泰国","pol":"Tajlandia","urd":"تھائی لینڈ","kor":"태국"}},"TL":{"currency":["USD"],"callingCode":["670"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-tl","name":{"common":"Timor-Leste","ces":"Východní Timor","deu":"Osttimor","fra":"Timor oriental","hrv":"Istočni Timor","ita":"Timor Est","jpn":"東ティモール","nld":"Oost-Timor","por":"Timor-Leste","rus":"Восточный Тимор","slk":"Východný Timor","spa":"Timor Oriental","fin":"Itä-Timor","est":"Ida-Timor","zho":"东帝汶","pol":"Timor Wschodni","urd":"مشرقی تیمور","kor":"동티모르"}},"TG":{"currency":["XOF"],"callingCode":["228"],"region":"Africa","subregion":"Western Africa","flag":"flag-tg","name":{"common":"Togo","ces":"Togo","deu":"Togo","fra":"Togo","hrv":"Togo","ita":"Togo","jpn":"トーゴ","nld":"Togo","por":"Togo","rus":"Того","slk":"Togo","spa":"Togo","fin":"Togo","est":"Togo","zho":"多哥","pol":"Togo","urd":"ٹوگو","kor":"토고"}},"TK":{"currency":["NZD"],"callingCode":["690"],"region":"Oceania","subregion":"Polynesia","flag":"flag-tk","name":{"common":"Tokelau","ces":"Tokelau","deu":"Tokelau","fra":"Tokelau","hrv":"Tokelau","ita":"Isole Tokelau","jpn":"トケラウ","nld":"Tokelau","por":"Tokelau","rus":"Токелау","slk":"Tokelau","spa":"Islas Tokelau","fin":"Tokelau","est":"Tokelau","zho":"托克劳","pol":"Tokelau","urd":"ٹوکیلاؤ","kor":"토켈라우"}},"TO":{"currency":["TOP"],"callingCode":["676"],"region":"Oceania","subregion":"Polynesia","flag":"flag-to","name":{"common":"Tonga","ces":"Tonga","deu":"Tonga","fra":"Tonga","hrv":"Tonga","ita":"Tonga","jpn":"トンガ","nld":"Tonga","por":"Tonga","rus":"Тонга","slk":"Tonga","spa":"Tonga","fin":"Tonga","est":"Tonga","zho":"汤加","pol":"Tonga","urd":"ٹونگا","kor":"통가"}},"TT":{"currency":["TTD"],"callingCode":["1868"],"region":"Americas","subregion":"Caribbean","flag":"flag-tt","name":{"common":"Trinidad and Tobago","ces":"Trinidad a Tobago","deu":"Trinidad und Tobago","fra":"Trinité-et-Tobago","hrv":"Trinidad i Tobago","ita":"Trinidad e Tobago","jpn":"トリニダード・トバゴ","nld":"Trinidad en Tobago","por":"Trinidade e Tobago","rus":"Тринидад и Тобаго","slk":"Trinidad a Tobago","spa":"Trinidad y Tobago","fin":"Trinidad ja Tobago","est":"Trinidad ja Tobago","zho":"特立尼达和多巴哥","pol":"Trynidad i Tobago","urd":"ٹرینیڈاڈ و ٹوباگو","kor":"트리니다드 토바고"}},"TN":{"currency":["TND"],"callingCode":["216"],"region":"Africa","subregion":"Northern Africa","flag":"flag-tn","name":{"common":"Tunisia","ces":"Tunisko","deu":"Tunesien","fra":"Tunisie","hrv":"Tunis","ita":"Tunisia","jpn":"チュニジア","nld":"Tunesië","por":"Tunísia","rus":"Тунис","slk":"Tunisko","spa":"Túnez","fin":"Tunisia","est":"Tuneesia","zho":"突尼斯","pol":"Tunezja","urd":"تونس","kor":"튀니지"}},"TR":{"currency":["TRY"],"callingCode":["90"],"region":"Asia","subregion":"Western Asia","flag":"flag-tr","name":{"common":"Turkey","ces":"Turecko","deu":"Türkei","fra":"Turquie","hrv":"Turska","ita":"Turchia","jpn":"トルコ","nld":"Turkije","por":"Turquia","rus":"Турция","slk":"Turecko","spa":"Turquía","fin":"Turkki","est":"Türgi","zho":"土耳其","pol":"Turcja","urd":"ترکی","kor":"터키"}},"TM":{"currency":["TMT"],"callingCode":["993"],"region":"Asia","subregion":"Central Asia","flag":"flag-tm","name":{"common":"Turkmenistan","ces":"Turkmenistán","deu":"Turkmenistan","fra":"Turkménistan","hrv":"Turkmenistan","ita":"Turkmenistan","jpn":"トルクメニスタン","nld":"Turkmenistan","por":"Turquemenistão","rus":"Туркмения","slk":"Turkménsko","spa":"Turkmenistán","fin":"Turkmenistan","est":"Türkmenistan","zho":"土库曼斯坦","pol":"Turkmenistan","urd":"ترکمانستان","kor":"투르크메니스탄"}},"TC":{"currency":["USD"],"callingCode":["1649"],"region":"Americas","subregion":"Caribbean","flag":"flag-tc","name":{"common":"Turks and Caicos Islands","ces":"Turks a Caicos","deu":"Turks-und Caicosinseln","fra":"Îles Turques-et-Caïques","hrv":"Otoci Turks i Caicos","ita":"Isole Turks e Caicos","jpn":"タークス・カイコス諸島","nld":"Turks-en Caicoseilanden","por":"Ilhas Turks e Caicos","rus":"Теркс и Кайкос","slk":"Turks a Caicos","spa":"Islas Turks y Caicos","fin":"Turks-ja Caicossaaret","est":"Turks ja Caicos","zho":"特克斯和凯科斯群岛","pol":"Turks i Caicos","urd":"جزائر کیکس و ترکیہ","kor":"터크스 케이커스 제도"}},"TV":{"currency":["AUD"],"callingCode":["688"],"region":"Oceania","subregion":"Polynesia","flag":"flag-tv","name":{"common":"Tuvalu","ces":"Tuvalu","deu":"Tuvalu","fra":"Tuvalu","hrv":"Tuvalu","ita":"Tuvalu","jpn":"ツバル","nld":"Tuvalu","por":"Tuvalu","rus":"Тувалу","slk":"Tuvalu","spa":"Tuvalu","fin":"Tuvalu","est":"Tuvalu","zho":"图瓦卢","pol":"Tuvalu","urd":"تووالو","kor":"투발루"}},"UG":{"currency":["UGX"],"callingCode":["256"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-ug","name":{"common":"Uganda","ces":"Uganda","deu":"Uganda","fra":"Ouganda","hrv":"Uganda","ita":"Uganda","jpn":"ウガンダ","nld":"Oeganda","por":"Uganda","rus":"Уганда","slk":"Uganda","spa":"Uganda","fin":"Uganda","est":"Uganda","zho":"乌干达","pol":"Uganda","urd":"یوگنڈا","kor":"우간다"}},"UA":{"currency":["UAH"],"callingCode":["380"],"region":"Europe","subregion":"Eastern Europe","flag":"flag-ua","name":{"common":"Ukraine","ces":"Ukrajina","deu":"Ukraine","fra":"Ukraine","hrv":"Ukrajina","ita":"Ucraina","jpn":"ウクライナ","nld":"Oekraïne","por":"Ucrânia","rus":"Украина","slk":"Ukrajina","spa":"Ucrania","fin":"Ukraina","est":"Ukraina","zho":"乌克兰","pol":"Ukraina","urd":"یوکرین","kor":"우크라이나"}},"AE":{"currency":["AED"],"callingCode":["971"],"region":"Asia","subregion":"Western Asia","flag":"flag-ae","name":{"common":"United Arab Emirates","ces":"Spojené arabské emiráty","deu":"Vereinigte Arabische Emirate","fra":"Émirats arabes unis","hrv":"Ujedinjeni Arapski Emirati","ita":"Emirati Arabi Uniti","jpn":"アラブ首長国連邦","nld":"Verenigde Arabische Emiraten","por":"Emirados Árabes Unidos","rus":"Объединённые Арабские Эмираты","slk":"Spojené arabské emiráty","spa":"Emiratos Árabes Unidos","fin":"Arabiemiraatit","est":"Araabia Ühendemiraadid","zho":"阿拉伯联合酋长国","pol":"Zjednoczone Emiraty Arabskie","urd":"متحدہ عرب امارات","kor":"아랍에미리트"}},"GB":{"currency":["GBP"],"callingCode":["44"],"region":"Europe","subregion":"Northern Europe","flag":"flag-gb","name":{"common":"United Kingdom","ces":"Spojené království","deu":"Vereinigtes Königreich","fra":"Royaume-Uni","hrv":"Ujedinjeno Kraljevstvo","ita":"Regno Unito","jpn":"イギリス","nld":"Verenigd Koninkrijk","por":"Reino Unido","rus":"Великобритания","slk":"Veľká Británia (Spojené kráľovstvo)","spa":"Reino Unido","fin":"Yhdistynyt kuningaskunta","est":"Suurbritannia","zho":"英国","pol":"Zjednoczone Krłlestwo","urd":"مملکتِ متحدہ","kor":"영국"}},"US":{"currency":["USD"],"callingCode":["1"],"region":"Americas","subregion":"North America","flag":"flag-us","name":{"common":"United States","ces":"Spojené státy","deu":"Vereinigte Staaten","fra":"États-Unis","hrv":"Sjedinjene Američke Države","ita":"Stati Uniti d'America","jpn":"アメリカ合衆国","nld":"Verenigde Staten","por":"Estados Unidos","rus":"Соединённые Штаты Америки","slk":"Spojené štáty americké","spa":"Estados Unidos","fin":"Yhdysvallat","est":"Ameerika Ühendriigid","zho":"美国","pol":"Stany Zjednoczone","urd":"ریاستہائے متحدہ","kor":"미국"}},"UM":{"currency":["USD"],"callingCode":[],"region":"Americas","subregion":"North America","flag":"flag-um","name":{"common":"United States Minor Outlying Islands","ces":"Menší odlehlé ostrovy USA","deu":"Kleinere Inselbesitzungen der Vereinigten Staaten","fra":"Îles mineures éloignées des États-Unis","hrv":"Mali udaljeni otoci SAD-a","ita":"Isole minori esterne degli Stati Uniti d'America","jpn":"合衆国領有小離島","nld":"Kleine afgelegen eilanden van de Verenigde Staten","por":"Ilhas Menores Distantes dos Estados Unidos","rus":"Внешние малые острова США","slk":"Menšie odľahlé ostrovy USA","spa":"Islas Ultramarinas Menores de Estados Unidos","fin":"Yhdysvaltain asumattomat saaret","est":"Ühendriikide hajasaared","zho":"美国本土外小岛屿","pol":"Dalekie Wyspy Mniejsze Stanów Zjednoczonych","urd":"امریکی چھوٹے بیرونی جزائر","kor":"미국령 군소 제도"}},"VI":{"currency":["USD"],"callingCode":["1340"],"region":"Americas","subregion":"Caribbean","flag":"flag-vi","name":{"common":"United States Virgin Islands","ces":"Americké Panenské ostrovy","deu":"Amerikanische Jungferninseln","fra":"Îles Vierges des États-Unis","hrv":"Američki Djevičanski Otoci","ita":"Isole Vergini americane","jpn":"アメリカ領ヴァージン諸島","nld":"Amerikaanse Maagdeneilanden","por":"Ilhas Virgens dos Estados Unidos","rus":"Виргинские Острова","slk":"Americké Panenské ostrovy","spa":"Islas Vírgenes de los Estados Unidos","fin":"Neitsytsaaret","est":"Neitsisaared, USA","zho":"美属维尔京群岛","pol":"Wyspy Dziewicze Stanów Zjednoczonych","urd":"امریکی جزائر ورجن","kor":"미국령 버진아일랜드"}},"UY":{"currency":["UYU"],"callingCode":["598"],"region":"Americas","subregion":"South America","flag":"flag-uy","name":{"common":"Uruguay","ces":"Uruguay","deu":"Uruguay","fra":"Uruguay","hrv":"Urugvaj","ita":"Uruguay","jpn":"ウルグアイ","nld":"Uruguay","por":"Uruguai","rus":"Уругвай","slk":"Uruguaj","spa":"Uruguay","fin":"Uruguay","est":"Uruguay","zho":"乌拉圭","pol":"Urugwaj","urd":"یوراگوئے","kor":"우루과이"}},"UZ":{"currency":["UZS"],"callingCode":["998"],"region":"Asia","subregion":"Central Asia","flag":"flag-uz","name":{"common":"Uzbekistan","ces":"Uzbekistán","deu":"Usbekistan","fra":"Ouzbékistan","hrv":"Uzbekistan","ita":"Uzbekistan","jpn":"ウズベキスタン","nld":"Oezbekistan","por":"Uzbequistão","rus":"Узбекистан","slk":"Uzbekistan","spa":"Uzbekistán","fin":"Uzbekistan","est":"Usbekistan","zho":"乌兹别克斯坦","pol":"Uzbekistan","urd":"ازبکستان","kor":"우즈베키스탄"}},"VU":{"currency":["VUV"],"callingCode":["678"],"region":"Oceania","subregion":"Melanesia","flag":"flag-vu","name":{"common":"Vanuatu","ces":"Vanuatu","deu":"Vanuatu","fra":"Vanuatu","hrv":"Vanuatu","ita":"Vanuatu","jpn":"バヌアツ","nld":"Vanuatu","por":"Vanuatu","rus":"Вануату","slk":"Vanuatu","spa":"Vanuatu","fin":"Vanuatu","est":"Vanuatu","zho":"瓦努阿图","pol":"Vanuatu","urd":"وانواتو","kor":"바누아투"}},"VA":{"currency":["EUR"],"callingCode":["3906698","379"],"region":"Europe","subregion":"Southern Europe","flag":"flag-va","name":{"common":"Vatican City","ces":"Vatikán","deu":"Vatikanstadt","fra":"Cité du Vatican","hrv":"Vatikan","ita":"Città del Vaticano","jpn":"バチカン市国","nld":"Vaticaanstad","por":"Cidade do Vaticano","rus":"Ватикан","slk":"Vatikán","spa":"Ciudad del Vaticano","fin":"Vatikaani","est":"Vatikan","zho":"梵蒂冈","pol":"Watykan","urd":"ویٹیکن سٹی","kor":"바티칸"}},"VE":{"currency":["VEF"],"callingCode":["58"],"region":"Americas","subregion":"South America","flag":"flag-ve","name":{"common":"Venezuela","ces":"Venezuela","deu":"Venezuela","fra":"Venezuela","hrv":"Venezuela","ita":"Venezuela","jpn":"ベネズエラ・ボリバル共和国","nld":"Venezuela","por":"Venezuela","rus":"Венесуэла","slk":"Venezuela","spa":"Venezuela","fin":"Venezuela","est":"Venezuela","zho":"委内瑞拉","pol":"Wenezuela","urd":"وینیزویلا","kor":"베네수엘라"}},"VN":{"currency":["VND"],"callingCode":["84"],"region":"Asia","subregion":"South-Eastern Asia","flag":"flag-vn","name":{"common":"Vietnam","ces":"Vietnam","deu":"Vietnam","fra":"Viêt Nam","hrv":"Vijetnam","ita":"Vietnam","jpn":"ベトナム","nld":"Vietnam","por":"Vietname","rus":"Вьетнам","slk":"Vietnam","spa":"Vietnam","fin":"Vietnam","est":"Vietnam","zho":"越南","pol":"Wietnam","urd":"ویتنام","kor":"베트남"}},"WF":{"currency":["XPF"],"callingCode":["681"],"region":"Oceania","subregion":"Polynesia","flag":"flag-wf","name":{"common":"Wallis and Futuna","ces":"Wallis a Futuna","deu":"Wallis und Futuna","fra":"Wallis-et-Futuna","hrv":"Wallis i Fortuna","ita":"Wallis e Futuna","jpn":"ウォリス・フツナ","nld":"Wallis en Futuna","por":"Wallis e Futuna","rus":"Уоллис и Футуна","slk":"Wallis a Futuna","spa":"Wallis y Futuna","fin":"Wallis ja Futuna","est":"Wallis ja Futuna","zho":"瓦利斯和富图纳群岛","pol":"Wallis i Futuna","urd":"والس و فتونہ","kor":""}},"EH":{"currency":["MAD","DZD","MRO"],"callingCode":["212"],"region":"Africa","subregion":"Northern Africa","flag":"flag-eh","name":{"common":"Western Sahara","ces":"Západní Sahara","deu":"Westsahara","fra":"Sahara Occidental","hrv":"Zapadna Sahara","ita":"Sahara Occidentale","jpn":"西サハラ","nld":"Westelijke Sahara","por":"Saara Ocidental","rus":"Западная Сахара","slk":"Západná Sahara","spa":"Sahara Occidental","fin":"Länsi-Sahara","est":"Lääne-Sahara","zho":"西撒哈拉","pol":"Sahara Zachodnia","urd":"مغربی صحارا","kor":"서사하라"}},"YE":{"currency":["YER"],"callingCode":["967"],"region":"Asia","subregion":"Western Asia","flag":"flag-ye","name":{"common":"Yemen","ces":"Jemen","deu":"Jemen","fra":"Yémen","hrv":"Jemen","ita":"Yemen","jpn":"イエメン","nld":"Jemen","por":"Iémen","rus":"Йемен","slk":"Jemen","spa":"Yemen","fin":"Jemen","est":"Jeemen","zho":"也门","pol":"Jemen","urd":"یمن","kor":"예멘"}},"ZM":{"currency":["ZMW"],"callingCode":["260"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-zm","name":{"common":"Zambia","ces":"Zambie","deu":"Sambia","fra":"Zambie","hrv":"Zambija","ita":"Zambia","jpn":"ザンビア","nld":"Zambia","por":"Zâmbia","rus":"Замбия","slk":"Zambia","spa":"Zambia","fin":"Sambia","est":"Sambia","zho":"赞比亚","pol":"Zambia","urd":"زیمبیا","kor":"잠비아"}},"ZW":{"currency":["ZWL"],"callingCode":["263"],"region":"Africa","subregion":"Eastern Africa","flag":"flag-zw","name":{"common":"Zimbabwe","ces":"Zimbabwe","deu":"Simbabwe","fra":"Zimbabwe","hrv":"Zimbabve","ita":"Zimbabwe","jpn":"ジンバブエ","nld":"Zimbabwe","por":"Zimbabwe","rus":"Зимбабве","slk":"Zimbabwe","spa":"Zimbabue","fin":"Zimbabwe","est":"Zimbabwe","zho":"津巴布韦","pol":"Zimbabwe","urd":"زمبابوے","kor":"짐바브웨"}},"AX":{"currency":["EUR"],"callingCode":["358"],"region":"Europe","subregion":"Northern Europe","flag":"flag-ax","name":{"common":"Åland Islands","ces":"Ålandy","deu":"Åland","fra":"Ahvenanmaa","hrv":"Ålandski otoci","ita":"Isole Aland","jpn":"オーランド諸島","nld":"Ålandeilanden","por":"Alândia","rus":"Аландские острова","slk":"Alandy","spa":"Alandia","fin":"Ahvenanmaa","est":"Ahvenamaa","zho":"奥兰群岛","pol":"Wyspy Alandzkie","urd":"جزائر اولند","kor":"올란드 제도"}}} \ No newline at end of file diff --git a/app/src/main/res/raw/node_modules_reactnativephoneinput_lib_resources_countries.json b/app/src/main/res/raw/node_modules_reactnativephoneinput_lib_resources_countries.json deleted file mode 100644 index 0cb35ead..00000000 --- a/app/src/main/res/raw/node_modules_reactnativephoneinput_lib_resources_countries.json +++ /dev/null @@ -1,1746 +0,0 @@ -[ - { - "name": "Afghanistan (‫افغانستان‬‎)", - "iso2": "af", - "dialCode": "93", - "priority": 0, - "areaCodes": null - }, - { - "name": "Albania (Shqipëri)", - "iso2": "al", - "dialCode": "355", - "priority": 0, - "areaCodes": null - }, - { - "name": "Algeria (‫الجزائر‬‎)", - "iso2": "dz", - "dialCode": "213", - "priority": 0, - "areaCodes": null - }, - { - "name": "American Samoa", - "iso2": "as", - "dialCode": "1684", - "priority": 0, - "areaCodes": null - }, - { - "name": "Andorra", - "iso2": "ad", - "dialCode": "376", - "priority": 0, - "areaCodes": null - }, - { - "name": "Angola", - "iso2": "ao", - "dialCode": "244", - "priority": 0, - "areaCodes": null - }, - { - "name": "Anguilla", - "iso2": "ai", - "dialCode": "1264", - "priority": 0, - "areaCodes": null - }, - { - "name": "Antigua and Barbuda", - "iso2": "ag", - "dialCode": "1268", - "priority": 0, - "areaCodes": null - }, - { - "name": "Argentina", - "iso2": "ar", - "dialCode": "54", - "priority": 0, - "areaCodes": null - }, - { - "name": "Armenia (Հայաստան)", - "iso2": "am", - "dialCode": "374", - "priority": 0, - "areaCodes": null - }, - { - "name": "Aruba", - "iso2": "aw", - "dialCode": "297", - "priority": 0, - "areaCodes": null - }, - { - "name": "Australia", - "iso2": "au", - "dialCode": "61", - "priority": 0, - "areaCodes": null - }, - { - "name": "Austria (Österreich)", - "iso2": "at", - "dialCode": "43", - "priority": 0, - "areaCodes": null - }, - { - "name": "Azerbaijan (Azərbaycan)", - "iso2": "az", - "dialCode": "994", - "priority": 0, - "areaCodes": null - }, - { - "name": "Bahamas", - "iso2": "bs", - "dialCode": "1242", - "priority": 0, - "areaCodes": null - }, - { - "name": "Bahrain (‫البحرين‬‎)", - "iso2": "bh", - "dialCode": "973", - "priority": 0, - "areaCodes": null - }, - { - "name": "Bangladesh (বাংলাদেশ)", - "iso2": "bd", - "dialCode": "880", - "priority": 0, - "areaCodes": null - }, - { - "name": "Barbados", - "iso2": "bb", - "dialCode": "1246", - "priority": 0, - "areaCodes": null - }, - { - "name": "Belarus (Беларусь)", - "iso2": "by", - "dialCode": "375", - "priority": 0, - "areaCodes": null - }, - { - "name": "Belgium (België)", - "iso2": "be", - "dialCode": "32", - "priority": 0, - "areaCodes": null - }, - { - "name": "Belize", - "iso2": "bz", - "dialCode": "501", - "priority": 0, - "areaCodes": null - }, - { - "name": "Benin (Bénin)", - "iso2": "bj", - "dialCode": "229", - "priority": 0, - "areaCodes": null - }, - { - "name": "Bermuda", - "iso2": "bm", - "dialCode": "1441", - "priority": 0, - "areaCodes": null - }, - { - "name": "Bhutan (འབྲུག)", - "iso2": "bt", - "dialCode": "975", - "priority": 0, - "areaCodes": null - }, - { - "name": "Bolivia", - "iso2": "bo", - "dialCode": "591", - "priority": 0, - "areaCodes": null - }, - { - "name": "Bosnia and Herzegovina (Босна и Херцеговина)", - "iso2": "ba", - "dialCode": "387", - "priority": 0, - "areaCodes": null - }, - { - "name": "Botswana", - "iso2": "bw", - "dialCode": "267", - "priority": 0, - "areaCodes": null - }, - { - "name": "Brazil (Brasil)", - "iso2": "br", - "dialCode": "55", - "priority": 0, - "areaCodes": null - }, - { - "name": "British Indian Ocean Territory", - "iso2": "io", - "dialCode": "246", - "priority": 0, - "areaCodes": null - }, - { - "name": "British Virgin Islands", - "iso2": "vg", - "dialCode": "1284", - "priority": 0, - "areaCodes": null - }, - { - "name": "Brunei", - "iso2": "bn", - "dialCode": "673", - "priority": 0, - "areaCodes": null - }, - { - "name": "Bulgaria (България)", - "iso2": "bg", - "dialCode": "359", - "priority": 0, - "areaCodes": null - }, - { - "name": "Burkina Faso", - "iso2": "bf", - "dialCode": "226", - "priority": 0, - "areaCodes": null - }, - { - "name": "Burundi (Uburundi)", - "iso2": "bi", - "dialCode": "257", - "priority": 0, - "areaCodes": null - }, - { - "name": "Cambodia (កម្ពុជា)", - "iso2": "kh", - "dialCode": "855", - "priority": 0, - "areaCodes": null - }, - { - "name": "Cameroon (Cameroun)", - "iso2": "cm", - "dialCode": "237", - "priority": 0, - "areaCodes": null - }, - { - "name": "Canada", - "iso2": "ca", - "dialCode": "1", - "priority": 1, - "areaCodes": [ - "204", - "226", - "236", - "249", - "250", - "289", - "306", - "343", - "365", - "387", - "403", - "416", - "418", - "431", - "437", - "438", - "450", - "506", - "514", - "519", - "548", - "579", - "581", - "587", - "604", - "613", - "639", - "647", - "672", - "705", - "709", - "742", - "778", - "780", - "782", - "807", - "819", - "825", - "867", - "873", - "902", - "905" - ] - }, - { - "name": "Cape Verde (Kabu Verdi)", - "iso2": "cv", - "dialCode": "238", - "priority": 0, - "areaCodes": null - }, - { - "name": "Caribbean Netherlands", - "iso2": "bq", - "dialCode": "599", - "priority": 1, - "areaCodes": null - }, - { - "name": "Cayman Islands", - "iso2": "ky", - "dialCode": "1345", - "priority": 0, - "areaCodes": null - }, - { - "name": "Central African Republic (République centrafricaine)", - "iso2": "cf", - "dialCode": "236", - "priority": 0, - "areaCodes": null - }, - { - "name": "Chad (Tchad)", - "iso2": "td", - "dialCode": "235", - "priority": 0, - "areaCodes": null - }, - { - "name": "Chile", - "iso2": "cl", - "dialCode": "56", - "priority": 0, - "areaCodes": null - }, - { - "name": "China (中国)", - "iso2": "cn", - "dialCode": "86", - "priority": 0, - "areaCodes": null - }, - { - "name": "Christmas Island", - "iso2": "cx", - "dialCode": "61", - "priority": 2, - "areaCodes": null - }, - { - "name": "Cocos (Keeling) Islands", - "iso2": "cc", - "dialCode": "61", - "priority": 1, - "areaCodes": null - }, - { - "name": "Colombia", - "iso2": "co", - "dialCode": "57", - "priority": 0, - "areaCodes": null - }, - { - "name": "Comoros (‫جزر القمر‬‎)", - "iso2": "km", - "dialCode": "269", - "priority": 0, - "areaCodes": null - }, - { - "name": "Congo (DRC) (Jamhuri ya Kidemokrasia ya Kongo)", - "iso2": "cd", - "dialCode": "243", - "priority": 0, - "areaCodes": null - }, - { - "name": "Congo (Republic) (Congo-Brazzaville)", - "iso2": "cg", - "dialCode": "242", - "priority": 0, - "areaCodes": null - }, - { - "name": "Cook Islands", - "iso2": "ck", - "dialCode": "682", - "priority": 0, - "areaCodes": null - }, - { - "name": "Costa Rica", - "iso2": "cr", - "dialCode": "506", - "priority": 0, - "areaCodes": null - }, - { - "name": "Côte d’Ivoire", - "iso2": "ci", - "dialCode": "225", - "priority": 0, - "areaCodes": null - }, - { - "name": "Croatia (Hrvatska)", - "iso2": "hr", - "dialCode": "385", - "priority": 0, - "areaCodes": null - }, - { - "name": "Cuba", - "iso2": "cu", - "dialCode": "53", - "priority": 0, - "areaCodes": null - }, - { - "name": "Curaçao", - "iso2": "cw", - "dialCode": "599", - "priority": 0, - "areaCodes": null - }, - { - "name": "Cyprus (Κύπρος)", - "iso2": "cy", - "dialCode": "357", - "priority": 0, - "areaCodes": null - }, - { - "name": "Czech Republic (Česká republika)", - "iso2": "cz", - "dialCode": "420", - "priority": 0, - "areaCodes": null - }, - { - "name": "Denmark (Danmark)", - "iso2": "dk", - "dialCode": "45", - "priority": 0, - "areaCodes": null - }, - { - "name": "Djibouti", - "iso2": "dj", - "dialCode": "253", - "priority": 0, - "areaCodes": null - }, - { - "name": "Dominica", - "iso2": "dm", - "dialCode": "1767", - "priority": 0, - "areaCodes": null - }, - { - "name": "Dominican Republic (República Dominicana)", - "iso2": "do", - "dialCode": "1", - "priority": 2, - "areaCodes": [ - "809", - "829", - "849" - ] - }, - { - "name": "Ecuador", - "iso2": "ec", - "dialCode": "593", - "priority": 0, - "areaCodes": null - }, - { - "name": "Egypt (‫مصر‬‎)", - "iso2": "eg", - "dialCode": "20", - "priority": 0, - "areaCodes": null - }, - { - "name": "El Salvador", - "iso2": "sv", - "dialCode": "503", - "priority": 0, - "areaCodes": null - }, - { - "name": "Equatorial Guinea (Guinea Ecuatorial)", - "iso2": "gq", - "dialCode": "240", - "priority": 0, - "areaCodes": null - }, - { - "name": "Eritrea", - "iso2": "er", - "dialCode": "291", - "priority": 0, - "areaCodes": null - }, - { - "name": "Estonia (Eesti)", - "iso2": "ee", - "dialCode": "372", - "priority": 0, - "areaCodes": null - }, - { - "name": "Ethiopia", - "iso2": "et", - "dialCode": "251", - "priority": 0, - "areaCodes": null - }, - { - "name": "Falkland Islands (Islas Malvinas)", - "iso2": "fk", - "dialCode": "500", - "priority": 0, - "areaCodes": null - }, - { - "name": "Faroe Islands (Føroyar)", - "iso2": "fo", - "dialCode": "298", - "priority": 0, - "areaCodes": null - }, - { - "name": "Fiji", - "iso2": "fj", - "dialCode": "679", - "priority": 0, - "areaCodes": null - }, - { - "name": "Finland (Suomi)", - "iso2": "fi", - "dialCode": "358", - "priority": 0, - "areaCodes": null - }, - { - "name": "France", - "iso2": "fr", - "dialCode": "33", - "priority": 0, - "areaCodes": null - }, - { - "name": "French Guiana (Guyane française)", - "iso2": "gf", - "dialCode": "594", - "priority": 0, - "areaCodes": null - }, - { - "name": "French Polynesia (Polynésie française)", - "iso2": "pf", - "dialCode": "689", - "priority": 0, - "areaCodes": null - }, - { - "name": "Gabon", - "iso2": "ga", - "dialCode": "241", - "priority": 0, - "areaCodes": null - }, - { - "name": "Gambia", - "iso2": "gm", - "dialCode": "220", - "priority": 0, - "areaCodes": null - }, - { - "name": "Georgia (საქართველო)", - "iso2": "ge", - "dialCode": "995", - "priority": 0, - "areaCodes": null - }, - { - "name": "Germany (Deutschland)", - "iso2": "de", - "dialCode": "49", - "priority": 0, - "areaCodes": null - }, - { - "name": "Ghana (Gaana)", - "iso2": "gh", - "dialCode": "233", - "priority": 0, - "areaCodes": null - }, - { - "name": "Gibraltar", - "iso2": "gi", - "dialCode": "350", - "priority": 0, - "areaCodes": null - }, - { - "name": "Greece (Ελλάδα)", - "iso2": "gr", - "dialCode": "30", - "priority": 0, - "areaCodes": null - }, - { - "name": "Greenland (Kalaallit Nunaat)", - "iso2": "gl", - "dialCode": "299", - "priority": 0, - "areaCodes": null - }, - { - "name": "Grenada", - "iso2": "gd", - "dialCode": "1473", - "priority": 0, - "areaCodes": null - }, - { - "name": "Guadeloupe", - "iso2": "gp", - "dialCode": "590", - "priority": 0, - "areaCodes": null - }, - { - "name": "Guam", - "iso2": "gu", - "dialCode": "1671", - "priority": 0, - "areaCodes": null - }, - { - "name": "Guatemala", - "iso2": "gt", - "dialCode": "502", - "priority": 0, - "areaCodes": null - }, - { - "name": "Guernsey", - "iso2": "gg", - "dialCode": "44", - "priority": 1, - "areaCodes": null - }, - { - "name": "Guinea (Guinée)", - "iso2": "gn", - "dialCode": "224", - "priority": 0, - "areaCodes": null - }, - { - "name": "Guinea-Bissau (Guiné Bissau)", - "iso2": "gw", - "dialCode": "245", - "priority": 0, - "areaCodes": null - }, - { - "name": "Guyana", - "iso2": "gy", - "dialCode": "592", - "priority": 0, - "areaCodes": null - }, - { - "name": "Haiti", - "iso2": "ht", - "dialCode": "509", - "priority": 0, - "areaCodes": null - }, - { - "name": "Honduras", - "iso2": "hn", - "dialCode": "504", - "priority": 0, - "areaCodes": null - }, - { - "name": "Hong Kong (香港)", - "iso2": "hk", - "dialCode": "852", - "priority": 0, - "areaCodes": null - }, - { - "name": "Hungary (Magyarország)", - "iso2": "hu", - "dialCode": "36", - "priority": 0, - "areaCodes": null - }, - { - "name": "Iceland (Ísland)", - "iso2": "is", - "dialCode": "354", - "priority": 0, - "areaCodes": null - }, - { - "name": "India (भारत)", - "iso2": "in", - "dialCode": "91", - "priority": 0, - "areaCodes": null - }, - { - "name": "Indonesia", - "iso2": "id", - "dialCode": "62", - "priority": 0, - "areaCodes": null - }, - { - "name": "Iran (‫ایران‬‎)", - "iso2": "ir", - "dialCode": "98", - "priority": 0, - "areaCodes": null - }, - { - "name": "Iraq (‫العراق‬‎)", - "iso2": "iq", - "dialCode": "964", - "priority": 0, - "areaCodes": null - }, - { - "name": "Ireland", - "iso2": "ie", - "dialCode": "353", - "priority": 0, - "areaCodes": null - }, - { - "name": "Isle of Man", - "iso2": "im", - "dialCode": "44", - "priority": 2, - "areaCodes": null - }, - { - "name": "Israel (‫ישראל‬‎)", - "iso2": "il", - "dialCode": "972", - "priority": 0, - "areaCodes": null - }, - { - "name": "Italy (Italia)", - "iso2": "it", - "dialCode": "39", - "priority": 0, - "areaCodes": null - }, - { - "name": "Jamaica", - "iso2": "jm", - "dialCode": "1876", - "priority": 0, - "areaCodes": null - }, - { - "name": "Japan (日本)", - "iso2": "jp", - "dialCode": "81", - "priority": 0, - "areaCodes": null - }, - { - "name": "Jersey", - "iso2": "je", - "dialCode": "44", - "priority": 3, - "areaCodes": null - }, - { - "name": "Jordan (‫الأردن‬‎)", - "iso2": "jo", - "dialCode": "962", - "priority": 0, - "areaCodes": null - }, - { - "name": "Kazakhstan (Казахстан)", - "iso2": "kz", - "dialCode": "7", - "priority": 1, - "areaCodes": null - }, - { - "name": "Kenya", - "iso2": "ke", - "dialCode": "254", - "priority": 0, - "areaCodes": null - }, - { - "name": "Kiribati", - "iso2": "ki", - "dialCode": "686", - "priority": 0, - "areaCodes": null - }, - { - "name": "Kuwait (‫الكويت‬‎)", - "iso2": "kw", - "dialCode": "965", - "priority": 0, - "areaCodes": null - }, - { - "name": "Kyrgyzstan (Кыргызстан)", - "iso2": "kg", - "dialCode": "996", - "priority": 0, - "areaCodes": null - }, - { - "name": "Laos (ລາວ)", - "iso2": "la", - "dialCode": "856", - "priority": 0, - "areaCodes": null - }, - { - "name": "Latvia (Latvija)", - "iso2": "lv", - "dialCode": "371", - "priority": 0, - "areaCodes": null - }, - { - "name": "Lebanon (‫لبنان‬‎)", - "iso2": "lb", - "dialCode": "961", - "priority": 0, - "areaCodes": null - }, - { - "name": "Lesotho", - "iso2": "ls", - "dialCode": "266", - "priority": 0, - "areaCodes": null - }, - { - "name": "Liberia", - "iso2": "lr", - "dialCode": "231", - "priority": 0, - "areaCodes": null - }, - { - "name": "Libya (‫ليبيا‬‎)", - "iso2": "ly", - "dialCode": "218", - "priority": 0, - "areaCodes": null - }, - { - "name": "Liechtenstein", - "iso2": "li", - "dialCode": "423", - "priority": 0, - "areaCodes": null - }, - { - "name": "Lithuania (Lietuva)", - "iso2": "lt", - "dialCode": "370", - "priority": 0, - "areaCodes": null - }, - { - "name": "Luxembourg", - "iso2": "lu", - "dialCode": "352", - "priority": 0, - "areaCodes": null - }, - { - "name": "Macau (澳門)", - "iso2": "mo", - "dialCode": "853", - "priority": 0, - "areaCodes": null - }, - { - "name": "Macedonia (FYROM) (Македонија)", - "iso2": "mk", - "dialCode": "389", - "priority": 0, - "areaCodes": null - }, - { - "name": "Madagascar (Madagasikara)", - "iso2": "mg", - "dialCode": "261", - "priority": 0, - "areaCodes": null - }, - { - "name": "Malawi", - "iso2": "mw", - "dialCode": "265", - "priority": 0, - "areaCodes": null - }, - { - "name": "Malaysia", - "iso2": "my", - "dialCode": "60", - "priority": 0, - "areaCodes": null - }, - { - "name": "Maldives", - "iso2": "mv", - "dialCode": "960", - "priority": 0, - "areaCodes": null - }, - { - "name": "Mali", - "iso2": "ml", - "dialCode": "223", - "priority": 0, - "areaCodes": null - }, - { - "name": "Malta", - "iso2": "mt", - "dialCode": "356", - "priority": 0, - "areaCodes": null - }, - { - "name": "Marshall Islands", - "iso2": "mh", - "dialCode": "692", - "priority": 0, - "areaCodes": null - }, - { - "name": "Martinique", - "iso2": "mq", - "dialCode": "596", - "priority": 0, - "areaCodes": null - }, - { - "name": "Mauritania (‫موريتانيا‬‎)", - "iso2": "mr", - "dialCode": "222", - "priority": 0, - "areaCodes": null - }, - { - "name": "Mauritius (Moris)", - "iso2": "mu", - "dialCode": "230", - "priority": 0, - "areaCodes": null - }, - { - "name": "Mayotte", - "iso2": "yt", - "dialCode": "262", - "priority": 1, - "areaCodes": null - }, - { - "name": "Mexico (México)", - "iso2": "mx", - "dialCode": "52", - "priority": 0, - "areaCodes": null - }, - { - "name": "Micronesia", - "iso2": "fm", - "dialCode": "691", - "priority": 0, - "areaCodes": null - }, - { - "name": "Moldova (Republica Moldova)", - "iso2": "md", - "dialCode": "373", - "priority": 0, - "areaCodes": null - }, - { - "name": "Monaco", - "iso2": "mc", - "dialCode": "377", - "priority": 0, - "areaCodes": null - }, - { - "name": "Mongolia (Монгол)", - "iso2": "mn", - "dialCode": "976", - "priority": 0, - "areaCodes": null - }, - { - "name": "Montenegro (Crna Gora)", - "iso2": "me", - "dialCode": "382", - "priority": 0, - "areaCodes": null - }, - { - "name": "Montserrat", - "iso2": "ms", - "dialCode": "1664", - "priority": 0, - "areaCodes": null - }, - { - "name": "Morocco (‫المغرب‬‎)", - "iso2": "ma", - "dialCode": "212", - "priority": 0, - "areaCodes": null - }, - { - "name": "Mozambique (Moçambique)", - "iso2": "mz", - "dialCode": "258", - "priority": 0, - "areaCodes": null - }, - { - "name": "Myanmar (Burma)", - "iso2": "mm", - "dialCode": "95", - "priority": 0, - "areaCodes": null - }, - { - "name": "Namibia (Namibië)", - "iso2": "na", - "dialCode": "264", - "priority": 0, - "areaCodes": null - }, - { - "name": "Nauru", - "iso2": "nr", - "dialCode": "674", - "priority": 0, - "areaCodes": null - }, - { - "name": "Nepal (नेपाल)", - "iso2": "np", - "dialCode": "977", - "priority": 0, - "areaCodes": null - }, - { - "name": "Netherlands (Nederland)", - "iso2": "nl", - "dialCode": "31", - "priority": 0, - "areaCodes": null - }, - { - "name": "New Caledonia (Nouvelle-Calédonie)", - "iso2": "nc", - "dialCode": "687", - "priority": 0, - "areaCodes": null - }, - { - "name": "New Zealand", - "iso2": "nz", - "dialCode": "64", - "priority": 0, - "areaCodes": null - }, - { - "name": "Nicaragua", - "iso2": "ni", - "dialCode": "505", - "priority": 0, - "areaCodes": null - }, - { - "name": "Niger (Nijar)", - "iso2": "ne", - "dialCode": "227", - "priority": 0, - "areaCodes": null - }, - { - "name": "Nigeria", - "iso2": "ng", - "dialCode": "234", - "priority": 0, - "areaCodes": null - }, - { - "name": "Niue", - "iso2": "nu", - "dialCode": "683", - "priority": 0, - "areaCodes": null - }, - { - "name": "Norfolk Island", - "iso2": "nf", - "dialCode": "672", - "priority": 0, - "areaCodes": null - }, - { - "name": "North Korea (조선 민주주의 인민 공화국)", - "iso2": "kp", - "dialCode": "850", - "priority": 0, - "areaCodes": null - }, - { - "name": "Northern Mariana Islands", - "iso2": "mp", - "dialCode": "1670", - "priority": 0, - "areaCodes": null - }, - { - "name": "Norway (Norge)", - "iso2": "no", - "dialCode": "47", - "priority": 0, - "areaCodes": null - }, - { - "name": "Oman (‫عُمان‬‎)", - "iso2": "om", - "dialCode": "968", - "priority": 0, - "areaCodes": null - }, - { - "name": "Pakistan (‫پاکستان‬‎)", - "iso2": "pk", - "dialCode": "92", - "priority": 0, - "areaCodes": null - }, - { - "name": "Palau", - "iso2": "pw", - "dialCode": "680", - "priority": 0, - "areaCodes": null - }, - { - "name": "Palestine (‫فلسطين‬‎)", - "iso2": "ps", - "dialCode": "970", - "priority": 0, - "areaCodes": null - }, - { - "name": "Panama (Panamá)", - "iso2": "pa", - "dialCode": "507", - "priority": 0, - "areaCodes": null - }, - { - "name": "Papua New Guinea", - "iso2": "pg", - "dialCode": "675", - "priority": 0, - "areaCodes": null - }, - { - "name": "Paraguay", - "iso2": "py", - "dialCode": "595", - "priority": 0, - "areaCodes": null - }, - { - "name": "Peru (Perú)", - "iso2": "pe", - "dialCode": "51", - "priority": 0, - "areaCodes": null - }, - { - "name": "Philippines", - "iso2": "ph", - "dialCode": "63", - "priority": 0, - "areaCodes": null - }, - { - "name": "Poland (Polska)", - "iso2": "pl", - "dialCode": "48", - "priority": 0, - "areaCodes": null - }, - { - "name": "Portugal", - "iso2": "pt", - "dialCode": "351", - "priority": 0, - "areaCodes": null - }, - { - "name": "Puerto Rico", - "iso2": "pr", - "dialCode": "1", - "priority": 3, - "areaCodes": [ - "787", - "939" - ] - }, - { - "name": "Qatar (‫قطر‬‎)", - "iso2": "qa", - "dialCode": "974", - "priority": 0, - "areaCodes": null - }, - { - "name": "Réunion (La Réunion)", - "iso2": "re", - "dialCode": "262", - "priority": 0, - "areaCodes": null - }, - { - "name": "Romania (România)", - "iso2": "ro", - "dialCode": "40", - "priority": 0, - "areaCodes": null - }, - { - "name": "Russia (Россия)", - "iso2": "ru", - "dialCode": "7", - "priority": 0, - "areaCodes": null - }, - { - "name": "Rwanda", - "iso2": "rw", - "dialCode": "250", - "priority": 0, - "areaCodes": null - }, - { - "name": "Saint Barthélemy (Saint-Barthélemy)", - "iso2": "bl", - "dialCode": "590", - "priority": 1, - "areaCodes": null - }, - { - "name": "Saint Helena", - "iso2": "sh", - "dialCode": "290", - "priority": 0, - "areaCodes": null - }, - { - "name": "Saint Kitts and Nevis", - "iso2": "kn", - "dialCode": "1869", - "priority": 0, - "areaCodes": null - }, - { - "name": "Saint Lucia", - "iso2": "lc", - "dialCode": "1758", - "priority": 0, - "areaCodes": null - }, - { - "name": "Saint Martin (Saint-Martin (partie française))", - "iso2": "mf", - "dialCode": "590", - "priority": 2, - "areaCodes": null - }, - { - "name": "Saint Pierre and Miquelon (Saint-Pierre-et-Miquelon)", - "iso2": "pm", - "dialCode": "508", - "priority": 0, - "areaCodes": null - }, - { - "name": "Saint Vincent and the Grenadines", - "iso2": "vc", - "dialCode": "1784", - "priority": 0, - "areaCodes": null - }, - { - "name": "Samoa", - "iso2": "ws", - "dialCode": "685", - "priority": 0, - "areaCodes": null - }, - { - "name": "San Marino", - "iso2": "sm", - "dialCode": "378", - "priority": 0, - "areaCodes": null - }, - { - "name": "São Tomé and Príncipe (São Tomé e Príncipe)", - "iso2": "st", - "dialCode": "239", - "priority": 0, - "areaCodes": null - }, - { - "name": "Saudi Arabia (‫المملكة العربية السعودية‬‎)", - "iso2": "sa", - "dialCode": "966", - "priority": 0, - "areaCodes": null - }, - { - "name": "Senegal (Sénégal)", - "iso2": "sn", - "dialCode": "221", - "priority": 0, - "areaCodes": null - }, - { - "name": "Serbia (Србија)", - "iso2": "rs", - "dialCode": "381", - "priority": 0, - "areaCodes": null - }, - { - "name": "Seychelles", - "iso2": "sc", - "dialCode": "248", - "priority": 0, - "areaCodes": null - }, - { - "name": "Sierra Leone", - "iso2": "sl", - "dialCode": "232", - "priority": 0, - "areaCodes": null - }, - { - "name": "Singapore", - "iso2": "sg", - "dialCode": "65", - "priority": 0, - "areaCodes": null - }, - { - "name": "Sint Maarten", - "iso2": "sx", - "dialCode": "1721", - "priority": 0, - "areaCodes": null - }, - { - "name": "Slovakia (Slovensko)", - "iso2": "sk", - "dialCode": "421", - "priority": 0, - "areaCodes": null - }, - { - "name": "Slovenia (Slovenija)", - "iso2": "si", - "dialCode": "386", - "priority": 0, - "areaCodes": null - }, - { - "name": "Solomon Islands", - "iso2": "sb", - "dialCode": "677", - "priority": 0, - "areaCodes": null - }, - { - "name": "Somalia (Soomaaliya)", - "iso2": "so", - "dialCode": "252", - "priority": 0, - "areaCodes": null - }, - { - "name": "South Africa", - "iso2": "za", - "dialCode": "27", - "priority": 0, - "areaCodes": null - }, - { - "name": "South Korea (대한민국)", - "iso2": "kr", - "dialCode": "82", - "priority": 0, - "areaCodes": null - }, - { - "name": "South Sudan (‫جنوب السودان‬‎)", - "iso2": "ss", - "dialCode": "211", - "priority": 0, - "areaCodes": null - }, - { - "name": "Spain (España)", - "iso2": "es", - "dialCode": "34", - "priority": 0, - "areaCodes": null - }, - { - "name": "Sri Lanka (ශ්‍රී ලංකාව)", - "iso2": "lk", - "dialCode": "94", - "priority": 0, - "areaCodes": null - }, - { - "name": "Sudan (‫السودان‬‎)", - "iso2": "sd", - "dialCode": "249", - "priority": 0, - "areaCodes": null - }, - { - "name": "Suriname", - "iso2": "sr", - "dialCode": "597", - "priority": 0, - "areaCodes": null - }, - { - "name": "Svalbard and Jan Mayen", - "iso2": "sj", - "dialCode": "47", - "priority": 1, - "areaCodes": null - }, - { - "name": "Swaziland", - "iso2": "sz", - "dialCode": "268", - "priority": 0, - "areaCodes": null - }, - { - "name": "Sweden (Sverige)", - "iso2": "se", - "dialCode": "46", - "priority": 0, - "areaCodes": null - }, - { - "name": "Switzerland (Schweiz)", - "iso2": "ch", - "dialCode": "41", - "priority": 0, - "areaCodes": null - }, - { - "name": "Syria (‫سوريا‬‎)", - "iso2": "sy", - "dialCode": "963", - "priority": 0, - "areaCodes": null - }, - { - "name": "Taiwan (台灣)", - "iso2": "tw", - "dialCode": "886", - "priority": 0, - "areaCodes": null - }, - { - "name": "Tajikistan", - "iso2": "tj", - "dialCode": "992", - "priority": 0, - "areaCodes": null - }, - { - "name": "Tanzania", - "iso2": "tz", - "dialCode": "255", - "priority": 0, - "areaCodes": null - }, - { - "name": "Thailand (ไทย)", - "iso2": "th", - "dialCode": "66", - "priority": 0, - "areaCodes": null - }, - { - "name": "Timor-Leste", - "iso2": "tl", - "dialCode": "670", - "priority": 0, - "areaCodes": null - }, - { - "name": "Togo", - "iso2": "tg", - "dialCode": "228", - "priority": 0, - "areaCodes": null - }, - { - "name": "Tokelau", - "iso2": "tk", - "dialCode": "690", - "priority": 0, - "areaCodes": null - }, - { - "name": "Tonga", - "iso2": "to", - "dialCode": "676", - "priority": 0, - "areaCodes": null - }, - { - "name": "Trinidad and Tobago", - "iso2": "tt", - "dialCode": "1868", - "priority": 0, - "areaCodes": null - }, - { - "name": "Tunisia (‫تونس‬‎)", - "iso2": "tn", - "dialCode": "216", - "priority": 0, - "areaCodes": null - }, - { - "name": "Turkey (Türkiye)", - "iso2": "tr", - "dialCode": "90", - "priority": 0, - "areaCodes": null - }, - { - "name": "Turkmenistan", - "iso2": "tm", - "dialCode": "993", - "priority": 0, - "areaCodes": null - }, - { - "name": "Turks and Caicos Islands", - "iso2": "tc", - "dialCode": "1649", - "priority": 0, - "areaCodes": null - }, - { - "name": "Tuvalu", - "iso2": "tv", - "dialCode": "688", - "priority": 0, - "areaCodes": null - }, - { - "name": "U.S. Virgin Islands", - "iso2": "vi", - "dialCode": "1340", - "priority": 0, - "areaCodes": null - }, - { - "name": "Uganda", - "iso2": "ug", - "dialCode": "256", - "priority": 0, - "areaCodes": null - }, - { - "name": "Ukraine (Україна)", - "iso2": "ua", - "dialCode": "380", - "priority": 0, - "areaCodes": null - }, - { - "name": "United Arab Emirates (‫الإمارات العربية المتحدة‬‎)", - "iso2": "ae", - "dialCode": "971", - "priority": 0, - "areaCodes": null - }, - { - "name": "United Kingdom", - "iso2": "gb", - "dialCode": "44", - "priority": 0, - "areaCodes": null - }, - { - "name": "United States", - "iso2": "us", - "dialCode": "1", - "priority": 0, - "areaCodes": null - }, - { - "name": "Uruguay", - "iso2": "uy", - "dialCode": "598", - "priority": 0, - "areaCodes": null - }, - { - "name": "Uzbekistan (Oʻzbekiston)", - "iso2": "uz", - "dialCode": "998", - "priority": 0, - "areaCodes": null - }, - { - "name": "Vanuatu", - "iso2": "vu", - "dialCode": "678", - "priority": 0, - "areaCodes": null - }, - { - "name": "Vatican City (Città del Vaticano)", - "iso2": "va", - "dialCode": "39", - "priority": 1, - "areaCodes": null - }, - { - "name": "Venezuela", - "iso2": "ve", - "dialCode": "58", - "priority": 0, - "areaCodes": null - }, - { - "name": "Vietnam (Việt Nam)", - "iso2": "vn", - "dialCode": "84", - "priority": 0, - "areaCodes": null - }, - { - "name": "Wallis and Futuna", - "iso2": "wf", - "dialCode": "681", - "priority": 0, - "areaCodes": null - }, - { - "name": "Western Sahara (‫الصحراء الغربية‬‎)", - "iso2": "eh", - "dialCode": "212", - "priority": 1, - "areaCodes": null - }, - { - "name": "Yemen (‫اليمن‬‎)", - "iso2": "ye", - "dialCode": "967", - "priority": 0, - "areaCodes": null - }, - { - "name": "Zambia", - "iso2": "zm", - "dialCode": "260", - "priority": 0, - "areaCodes": null - }, - { - "name": "Zimbabwe", - "iso2": "zw", - "dialCode": "263", - "priority": 0, - "areaCodes": null - }, - { - "name": "Åland Islands", - "iso2": "ax", - "dialCode": "358", - "priority": 1, - "areaCodes": null - } -] \ No newline at end of file diff --git a/app/src/main/res/raw/node_modules_reactnativephoneinput_lib_resources_numbertype.json b/app/src/main/res/raw/node_modules_reactnativephoneinput_lib_resources_numbertype.json deleted file mode 100644 index 533f3f1f..00000000 --- a/app/src/main/res/raw/node_modules_reactnativephoneinput_lib_resources_numbertype.json +++ /dev/null @@ -1,14 +0,0 @@ -{ - "FIXED_LINE": 0, - "MOBILE": 1, - "FIXED_LINE_OR_MOBILE": 2, - "TOLL_FREE": 3, - "PREMIUM_RATE": 4, - "SHARED_COST": 5, - "VOIP": 6, - "PERSONAL_NUMBER": 7, - "PAGER": 8, - "UAN": 9, - "VOICEMAIL": 10, - "UNKNOWN": -1 -} diff --git a/app/src/main/res/raw/node_modules_reactnativevectoricons_glyphmaps_feather.json b/app/src/main/res/raw/node_modules_reactnativevectoricons_glyphmaps_feather.json deleted file mode 100644 index c17b44b8..00000000 --- a/app/src/main/res/raw/node_modules_reactnativevectoricons_glyphmaps_feather.json +++ /dev/null @@ -1,281 +0,0 @@ -{ - "activity": 61696, - "airplay": 61697, - "alert-circle": 61698, - "alert-octagon": 61699, - "alert-triangle": 61700, - "align-center": 61701, - "align-justify": 61702, - "align-left": 61703, - "align-right": 61704, - "anchor": 61705, - "aperture": 61706, - "archive": 61707, - "arrow-down": 61708, - "arrow-down-circle": 61709, - "arrow-down-left": 61710, - "arrow-down-right": 61711, - "arrow-left": 61712, - "arrow-left-circle": 61713, - "arrow-right": 61714, - "arrow-right-circle": 61715, - "arrow-up": 61716, - "arrow-up-circle": 61717, - "arrow-up-left": 61718, - "arrow-up-right": 61719, - "at-sign": 61720, - "award": 61721, - "bar-chart": 61722, - "bar-chart-2": 61723, - "battery": 61724, - "battery-charging": 61725, - "bell": 61726, - "bell-off": 61727, - "bluetooth": 61728, - "bold": 61729, - "book": 61730, - "book-open": 61731, - "bookmark": 61732, - "box": 61733, - "briefcase": 61734, - "calendar": 61735, - "camera": 61736, - "camera-off": 61737, - "cast": 61738, - "check": 61739, - "check-circle": 61740, - "check-square": 61741, - "chevron-down": 61742, - "chevron-left": 61743, - "chevron-right": 61744, - "chevron-up": 61745, - "chevrons-down": 61746, - "chevrons-left": 61747, - "chevrons-right": 61748, - "chevrons-up": 61749, - "chrome": 61750, - "circle": 61751, - "clipboard": 61752, - "clock": 61753, - "cloud": 61754, - "cloud-drizzle": 61755, - "cloud-lightning": 61756, - "cloud-off": 61757, - "cloud-rain": 61758, - "cloud-snow": 61759, - "code": 61760, - "codepen": 61761, - "codesandbox": 61972, - "coffee": 61762, - "columns": 61973, - "command": 61763, - "compass": 61764, - "copy": 61765, - "corner-down-left": 61766, - "corner-down-right": 61767, - "corner-left-down": 61768, - "corner-left-up": 61769, - "corner-right-down": 61770, - "corner-right-up": 61771, - "corner-up-left": 61772, - "corner-up-right": 61773, - "cpu": 61774, - "credit-card": 61775, - "crop": 61776, - "crosshair": 61777, - "database": 61778, - "delete": 61779, - "disc": 61780, - "dollar-sign": 61781, - "download": 61782, - "download-cloud": 61783, - "droplet": 61784, - "edit": 61785, - "edit-2": 61786, - "edit-3": 61787, - "external-link": 61788, - "eye": 61789, - "eye-off": 61790, - "facebook": 61791, - "fast-forward": 61792, - "feather": 61793, - "figma": 61970, - "file": 61794, - "file-minus": 61795, - "file-plus": 61796, - "file-text": 61797, - "film": 61798, - "filter": 61799, - "flag": 61800, - "folder": 61801, - "folder-minus": 61802, - "folder-plus": 61803, - "frown": 61804, - "gift": 61805, - "git-branch": 61806, - "git-commit": 61807, - "git-merge": 61808, - "git-pull-request": 61809, - "github": 61810, - "gitlab": 61811, - "globe": 61812, - "grid": 61813, - "hard-drive": 61814, - "hash": 61815, - "headphones": 61816, - "heart": 61817, - "help-circle": 61818, - "hexagon": 61974, - "home": 61819, - "image": 61820, - "inbox": 61821, - "info": 61822, - "instagram": 61823, - "italic": 61824, - "key": 61967, - "layers": 61825, - "layout": 61826, - "life-buoy": 61827, - "link": 61828, - "link-2": 61829, - "linkedin": 61830, - "list": 61831, - "loader": 61832, - "lock": 61833, - "log-in": 61834, - "log-out": 61835, - "mail": 61836, - "map": 61837, - "map-pin": 61838, - "maximize": 61839, - "maximize-2": 61840, - "meh": 61841, - "menu": 61842, - "message-circle": 61843, - "message-square": 61844, - "mic": 61845, - "mic-off": 61846, - "minimize": 61847, - "minimize-2": 61848, - "minus": 61849, - "minus-circle": 61850, - "minus-square": 61851, - "monitor": 61852, - "moon": 61853, - "more-horizontal": 61854, - "more-vertical": 61855, - "mouse-pointer": 61968, - "move": 61856, - "music": 61857, - "navigation": 61858, - "navigation-2": 61859, - "octagon": 61860, - "package": 61861, - "paperclip": 61862, - "pause": 61863, - "pause-circle": 61864, - "pen-tool": 61969, - "percent": 61865, - "phone": 61866, - "phone-call": 61867, - "phone-forwarded": 61868, - "phone-incoming": 61869, - "phone-missed": 61870, - "phone-off": 61871, - "phone-outgoing": 61872, - "pie-chart": 61873, - "play": 61874, - "play-circle": 61875, - "plus": 61876, - "plus-circle": 61877, - "plus-square": 61878, - "pocket": 61879, - "power": 61880, - "printer": 61881, - "radio": 61882, - "refresh-ccw": 61883, - "refresh-cw": 61884, - "repeat": 61885, - "rewind": 61886, - "rotate-ccw": 61887, - "rotate-cw": 61888, - "rss": 61889, - "save": 61890, - "scissors": 61891, - "search": 61892, - "send": 61893, - "server": 61894, - "settings": 61895, - "share": 61896, - "share-2": 61897, - "shield": 61898, - "shield-off": 61899, - "shopping-bag": 61900, - "shopping-cart": 61901, - "shuffle": 61902, - "sidebar": 61903, - "skip-back": 61904, - "skip-forward": 61905, - "slack": 61906, - "slash": 61907, - "sliders": 61908, - "smartphone": 61909, - "smile": 61910, - "speaker": 61911, - "square": 61912, - "star": 61913, - "stop-circle": 61914, - "sun": 61915, - "sunrise": 61916, - "sunset": 61917, - "tablet": 61975, - "tag": 61919, - "target": 61920, - "terminal": 61921, - "thermometer": 61922, - "thumbs-down": 61923, - "thumbs-up": 61924, - "toggle-left": 61925, - "toggle-right": 61926, - "trash": 61927, - "trash-2": 61928, - "trello": 61929, - "trending-down": 61930, - "trending-up": 61931, - "triangle": 61932, - "truck": 61933, - "tv": 61934, - "twitter": 61935, - "type": 61936, - "umbrella": 61937, - "underline": 61938, - "unlock": 61939, - "upload": 61940, - "upload-cloud": 61941, - "user": 61942, - "user-check": 61943, - "user-minus": 61944, - "user-plus": 61945, - "user-x": 61946, - "users": 61947, - "video": 61948, - "video-off": 61949, - "voicemail": 61950, - "volume": 61951, - "volume-1": 61952, - "volume-2": 61953, - "volume-x": 61954, - "watch": 61955, - "wifi": 61956, - "wifi-off": 61957, - "wind": 61958, - "x": 61959, - "x-circle": 61960, - "x-octagon": 61971, - "x-square": 61961, - "youtube": 61962, - "zap": 61963, - "zap-off": 61964, - "zoom-in": 61965, - "zoom-out": 61966 -} \ No newline at end of file diff --git a/app/src/main/res/raw/node_modules_reactnativevectoricons_glyphmaps_fontawesome5free.json b/app/src/main/res/raw/node_modules_reactnativevectoricons_glyphmaps_fontawesome5free.json deleted file mode 100644 index a93ad409..00000000 --- a/app/src/main/res/raw/node_modules_reactnativevectoricons_glyphmaps_fontawesome5free.json +++ /dev/null @@ -1,1352 +0,0 @@ -{ - "500px": 62062, - "accessible-icon": 62312, - "accusoft": 62313, - "acquisitions-incorporated": 63151, - "ad": 63041, - "address-book": 62137, - "address-card": 62139, - "adjust": 61506, - "adn": 61808, - "adobe": 63352, - "adversal": 62314, - "affiliatetheme": 62315, - "air-freshener": 62928, - "algolia": 62316, - "align-center": 61495, - "align-justify": 61497, - "align-left": 61494, - "align-right": 61496, - "alipay": 63042, - "allergies": 62561, - "amazon": 62064, - "amazon-pay": 62508, - "ambulance": 61689, - "american-sign-language-interpreting": 62115, - "amilia": 62317, - "anchor": 61757, - "android": 61819, - "angellist": 61961, - "angle-double-down": 61699, - "angle-double-left": 61696, - "angle-double-right": 61697, - "angle-double-up": 61698, - "angle-down": 61703, - "angle-left": 61700, - "angle-right": 61701, - "angle-up": 61702, - "angry": 62806, - "angrycreative": 62318, - "angular": 62496, - "ankh": 63044, - "app-store": 62319, - "app-store-ios": 62320, - "apper": 62321, - "apple": 61817, - "apple-alt": 62929, - "apple-pay": 62485, - "archive": 61831, - "archway": 62807, - "arrow-alt-circle-down": 62296, - "arrow-alt-circle-left": 62297, - "arrow-alt-circle-right": 62298, - "arrow-alt-circle-up": 62299, - "arrow-circle-down": 61611, - "arrow-circle-left": 61608, - "arrow-circle-right": 61609, - "arrow-circle-up": 61610, - "arrow-down": 61539, - "arrow-left": 61536, - "arrow-right": 61537, - "arrow-up": 61538, - "arrows-alt": 61618, - "arrows-alt-h": 62263, - "arrows-alt-v": 62264, - "artstation": 63354, - "assistive-listening-systems": 62114, - "asterisk": 61545, - "asymmetrik": 62322, - "at": 61946, - "atlas": 62808, - "atlassian": 63355, - "atom": 62930, - "audible": 62323, - "audio-description": 62110, - "autoprefixer": 62492, - "avianex": 62324, - "aviato": 62497, - "award": 62809, - "aws": 62325, - "baby": 63356, - "baby-carriage": 63357, - "backspace": 62810, - "backward": 61514, - "bacon": 63461, - "balance-scale": 62030, - "ban": 61534, - "band-aid": 62562, - "bandcamp": 62165, - "barcode": 61482, - "bars": 61641, - "baseball-ball": 62515, - "basketball-ball": 62516, - "bath": 62157, - "battery-empty": 62020, - "battery-full": 62016, - "battery-half": 62018, - "battery-quarter": 62019, - "battery-three-quarters": 62017, - "bed": 62006, - "beer": 61692, - "behance": 61876, - "behance-square": 61877, - "bell": 61683, - "bell-slash": 61942, - "bezier-curve": 62811, - "bible": 63047, - "bicycle": 61958, - "bimobject": 62328, - "binoculars": 61925, - "biohazard": 63360, - "birthday-cake": 61949, - "bitbucket": 61809, - "bitcoin": 62329, - "bity": 62330, - "black-tie": 62078, - "blackberry": 62331, - "blender": 62743, - "blender-phone": 63158, - "blind": 62109, - "blog": 63361, - "blogger": 62332, - "blogger-b": 62333, - "bluetooth": 62099, - "bluetooth-b": 62100, - "bold": 61490, - "bolt": 61671, - "bomb": 61922, - "bone": 62935, - "bong": 62812, - "book": 61485, - "book-dead": 63159, - "book-medical": 63462, - "book-open": 62744, - "book-reader": 62938, - "bookmark": 61486, - "bowling-ball": 62518, - "box": 62566, - "box-open": 62622, - "boxes": 62568, - "braille": 62113, - "brain": 62940, - "bread-slice": 63468, - "briefcase": 61617, - "briefcase-medical": 62569, - "broadcast-tower": 62745, - "broom": 62746, - "brush": 62813, - "btc": 61786, - "bug": 61832, - "building": 61869, - "bullhorn": 61601, - "bullseye": 61760, - "burn": 62570, - "buromobelexperte": 62335, - "bus": 61959, - "bus-alt": 62814, - "business-time": 63050, - "buysellads": 61965, - "calculator": 61932, - "calendar": 61747, - "calendar-alt": 61555, - "calendar-check": 62068, - "calendar-day": 63363, - "calendar-minus": 62066, - "calendar-plus": 62065, - "calendar-times": 62067, - "calendar-week": 63364, - "camera": 61488, - "camera-retro": 61571, - "campground": 63163, - "canadian-maple-leaf": 63365, - "candy-cane": 63366, - "cannabis": 62815, - "capsules": 62571, - "car": 61881, - "car-alt": 62942, - "car-battery": 62943, - "car-crash": 62945, - "car-side": 62948, - "caret-down": 61655, - "caret-left": 61657, - "caret-right": 61658, - "caret-square-down": 61776, - "caret-square-left": 61841, - "caret-square-right": 61778, - "caret-square-up": 61777, - "caret-up": 61656, - "carrot": 63367, - "cart-arrow-down": 61976, - "cart-plus": 61975, - "cash-register": 63368, - "cat": 63166, - "cc-amazon-pay": 62509, - "cc-amex": 61939, - "cc-apple-pay": 62486, - "cc-diners-club": 62028, - "cc-discover": 61938, - "cc-jcb": 62027, - "cc-mastercard": 61937, - "cc-paypal": 61940, - "cc-stripe": 61941, - "cc-visa": 61936, - "centercode": 62336, - "centos": 63369, - "certificate": 61603, - "chair": 63168, - "chalkboard": 62747, - "chalkboard-teacher": 62748, - "charging-station": 62951, - "chart-area": 61950, - "chart-bar": 61568, - "chart-line": 61953, - "chart-pie": 61952, - "check": 61452, - "check-circle": 61528, - "check-double": 62816, - "check-square": 61770, - "cheese": 63471, - "chess": 62521, - "chess-bishop": 62522, - "chess-board": 62524, - "chess-king": 62527, - "chess-knight": 62529, - "chess-pawn": 62531, - "chess-queen": 62533, - "chess-rook": 62535, - "chevron-circle-down": 61754, - "chevron-circle-left": 61751, - "chevron-circle-right": 61752, - "chevron-circle-up": 61753, - "chevron-down": 61560, - "chevron-left": 61523, - "chevron-right": 61524, - "chevron-up": 61559, - "child": 61870, - "chrome": 62056, - "church": 62749, - "circle": 61713, - "circle-notch": 61902, - "city": 63055, - "clinic-medical": 63474, - "clipboard": 62248, - "clipboard-check": 62572, - "clipboard-list": 62573, - "clock": 61463, - "clone": 62029, - "closed-captioning": 61962, - "cloud": 61634, - "cloud-download-alt": 62337, - "cloud-meatball": 63291, - "cloud-moon": 63171, - "cloud-moon-rain": 63292, - "cloud-rain": 63293, - "cloud-showers-heavy": 63296, - "cloud-sun": 63172, - "cloud-sun-rain": 63299, - "cloud-upload-alt": 62338, - "cloudscale": 62339, - "cloudsmith": 62340, - "cloudversify": 62341, - "cocktail": 62817, - "code": 61729, - "code-branch": 61734, - "codepen": 61899, - "codiepie": 62084, - "coffee": 61684, - "cog": 61459, - "cogs": 61573, - "coins": 62750, - "columns": 61659, - "comment": 61557, - "comment-alt": 62074, - "comment-dollar": 63057, - "comment-dots": 62637, - "comment-medical": 63477, - "comment-slash": 62643, - "comments": 61574, - "comments-dollar": 63059, - "compact-disc": 62751, - "compass": 61774, - "compress": 61542, - "compress-arrows-alt": 63372, - "concierge-bell": 62818, - "confluence": 63373, - "connectdevelop": 61966, - "contao": 62061, - "cookie": 62819, - "cookie-bite": 62820, - "copy": 61637, - "copyright": 61945, - "couch": 62648, - "cpanel": 62344, - "creative-commons": 62046, - "creative-commons-by": 62695, - "creative-commons-nc": 62696, - "creative-commons-nc-eu": 62697, - "creative-commons-nc-jp": 62698, - "creative-commons-nd": 62699, - "creative-commons-pd": 62700, - "creative-commons-pd-alt": 62701, - "creative-commons-remix": 62702, - "creative-commons-sa": 62703, - "creative-commons-sampling": 62704, - "creative-commons-sampling-plus": 62705, - "creative-commons-share": 62706, - "creative-commons-zero": 62707, - "credit-card": 61597, - "critical-role": 63177, - "crop": 61733, - "crop-alt": 62821, - "cross": 63060, - "crosshairs": 61531, - "crow": 62752, - "crown": 62753, - "crutch": 63479, - "css3": 61756, - "css3-alt": 62347, - "cube": 61874, - "cubes": 61875, - "cut": 61636, - "cuttlefish": 62348, - "d-and-d": 62349, - "d-and-d-beyond": 63178, - "dashcube": 61968, - "database": 61888, - "deaf": 62116, - "delicious": 61861, - "democrat": 63303, - "deploydog": 62350, - "deskpro": 62351, - "desktop": 61704, - "dev": 63180, - "deviantart": 61885, - "dharmachakra": 63061, - "dhl": 63376, - "diagnoses": 62576, - "diaspora": 63377, - "dice": 62754, - "dice-d20": 63183, - "dice-d6": 63185, - "dice-five": 62755, - "dice-four": 62756, - "dice-one": 62757, - "dice-six": 62758, - "dice-three": 62759, - "dice-two": 62760, - "digg": 61862, - "digital-ocean": 62353, - "digital-tachograph": 62822, - "directions": 62955, - "discord": 62354, - "discourse": 62355, - "divide": 62761, - "dizzy": 62823, - "dna": 62577, - "dochub": 62356, - "docker": 62357, - "dog": 63187, - "dollar-sign": 61781, - "dolly": 62578, - "dolly-flatbed": 62580, - "donate": 62649, - "door-closed": 62762, - "door-open": 62763, - "dot-circle": 61842, - "dove": 62650, - "download": 61465, - "draft2digital": 62358, - "drafting-compass": 62824, - "dragon": 63189, - "draw-polygon": 62958, - "dribbble": 61821, - "dribbble-square": 62359, - "dropbox": 61803, - "drum": 62825, - "drum-steelpan": 62826, - "drumstick-bite": 63191, - "drupal": 61865, - "dumbbell": 62539, - "dumpster": 63379, - "dumpster-fire": 63380, - "dungeon": 63193, - "dyalog": 62361, - "earlybirds": 62362, - "ebay": 62708, - "edge": 62082, - "edit": 61508, - "egg": 63483, - "eject": 61522, - "elementor": 62512, - "ellipsis-h": 61761, - "ellipsis-v": 61762, - "ello": 62961, - "ember": 62499, - "empire": 61905, - "envelope": 61664, - "envelope-open": 62134, - "envelope-open-text": 63064, - "envelope-square": 61849, - "envira": 62105, - "equals": 62764, - "eraser": 61741, - "erlang": 62365, - "ethereum": 62510, - "ethernet": 63382, - "etsy": 62167, - "euro-sign": 61779, - "exchange-alt": 62306, - "exclamation": 61738, - "exclamation-circle": 61546, - "exclamation-triangle": 61553, - "expand": 61541, - "expand-arrows-alt": 62238, - "expeditedssl": 62014, - "external-link-alt": 62301, - "external-link-square-alt": 62304, - "eye": 61550, - "eye-dropper": 61947, - "eye-slash": 61552, - "facebook": 61594, - "facebook-f": 62366, - "facebook-messenger": 62367, - "facebook-square": 61570, - "fantasy-flight-games": 63196, - "fast-backward": 61513, - "fast-forward": 61520, - "fax": 61868, - "feather": 62765, - "feather-alt": 62827, - "fedex": 63383, - "fedora": 63384, - "female": 61826, - "fighter-jet": 61691, - "figma": 63385, - "file": 61787, - "file-alt": 61788, - "file-archive": 61894, - "file-audio": 61895, - "file-code": 61897, - "file-contract": 62828, - "file-csv": 63197, - "file-download": 62829, - "file-excel": 61891, - "file-export": 62830, - "file-image": 61893, - "file-import": 62831, - "file-invoice": 62832, - "file-invoice-dollar": 62833, - "file-medical": 62583, - "file-medical-alt": 62584, - "file-pdf": 61889, - "file-powerpoint": 61892, - "file-prescription": 62834, - "file-signature": 62835, - "file-upload": 62836, - "file-video": 61896, - "file-word": 61890, - "fill": 62837, - "fill-drip": 62838, - "film": 61448, - "filter": 61616, - "fingerprint": 62839, - "fire": 61549, - "fire-alt": 63460, - "fire-extinguisher": 61748, - "firefox": 62057, - "first-aid": 62585, - "first-order": 62128, - "first-order-alt": 62730, - "firstdraft": 62369, - "fish": 62840, - "fist-raised": 63198, - "flag": 61476, - "flag-checkered": 61726, - "flag-usa": 63309, - "flask": 61635, - "flickr": 61806, - "flipboard": 62541, - "flushed": 62841, - "fly": 62487, - "folder": 61563, - "folder-minus": 63069, - "folder-open": 61564, - "folder-plus": 63070, - "font": 61489, - "font-awesome": 62132, - "font-awesome-alt": 62300, - "font-awesome-flag": 62501, - "font-awesome-logo-full": 62694, - "fonticons": 62080, - "fonticons-fi": 62370, - "football-ball": 62542, - "fort-awesome": 62086, - "fort-awesome-alt": 62371, - "forumbee": 61969, - "forward": 61518, - "foursquare": 61824, - "free-code-camp": 62149, - "freebsd": 62372, - "frog": 62766, - "frown": 61721, - "frown-open": 62842, - "fulcrum": 62731, - "funnel-dollar": 63074, - "futbol": 61923, - "galactic-republic": 62732, - "galactic-senate": 62733, - "gamepad": 61723, - "gas-pump": 62767, - "gavel": 61667, - "gem": 62373, - "genderless": 61997, - "get-pocket": 62053, - "gg": 62048, - "gg-circle": 62049, - "ghost": 63202, - "gift": 61547, - "gifts": 63388, - "git": 61907, - "git-square": 61906, - "github": 61595, - "github-alt": 61715, - "github-square": 61586, - "gitkraken": 62374, - "gitlab": 62102, - "gitter": 62502, - "glass-cheers": 63391, - "glass-martini": 61440, - "glass-martini-alt": 62843, - "glass-whiskey": 63392, - "glasses": 62768, - "glide": 62117, - "glide-g": 62118, - "globe": 61612, - "globe-africa": 62844, - "globe-americas": 62845, - "globe-asia": 62846, - "globe-europe": 63394, - "gofore": 62375, - "golf-ball": 62544, - "goodreads": 62376, - "goodreads-g": 62377, - "google": 61856, - "google-drive": 62378, - "google-play": 62379, - "google-plus": 62131, - "google-plus-g": 61653, - "google-plus-square": 61652, - "google-wallet": 61934, - "gopuram": 63076, - "graduation-cap": 61853, - "gratipay": 61828, - "grav": 62166, - "greater-than": 62769, - "greater-than-equal": 62770, - "grimace": 62847, - "grin": 62848, - "grin-alt": 62849, - "grin-beam": 62850, - "grin-beam-sweat": 62851, - "grin-hearts": 62852, - "grin-squint": 62853, - "grin-squint-tears": 62854, - "grin-stars": 62855, - "grin-tears": 62856, - "grin-tongue": 62857, - "grin-tongue-squint": 62858, - "grin-tongue-wink": 62859, - "grin-wink": 62860, - "grip-horizontal": 62861, - "grip-lines": 63396, - "grip-lines-vertical": 63397, - "grip-vertical": 62862, - "gripfire": 62380, - "grunt": 62381, - "guitar": 63398, - "gulp": 62382, - "h-square": 61693, - "hacker-news": 61908, - "hacker-news-square": 62383, - "hackerrank": 62967, - "hamburger": 63493, - "hammer": 63203, - "hamsa": 63077, - "hand-holding": 62653, - "hand-holding-heart": 62654, - "hand-holding-usd": 62656, - "hand-lizard": 62040, - "hand-middle-finger": 63494, - "hand-paper": 62038, - "hand-peace": 62043, - "hand-point-down": 61607, - "hand-point-left": 61605, - "hand-point-right": 61604, - "hand-point-up": 61606, - "hand-pointer": 62042, - "hand-rock": 62037, - "hand-scissors": 62039, - "hand-spock": 62041, - "hands": 62658, - "hands-helping": 62660, - "handshake": 62133, - "hanukiah": 63206, - "hard-hat": 63495, - "hashtag": 62098, - "hat-wizard": 63208, - "haykal": 63078, - "hdd": 61600, - "heading": 61916, - "headphones": 61477, - "headphones-alt": 62863, - "headset": 62864, - "heart": 61444, - "heart-broken": 63401, - "heartbeat": 61982, - "helicopter": 62771, - "highlighter": 62865, - "hiking": 63212, - "hippo": 63213, - "hips": 62546, - "hire-a-helper": 62384, - "history": 61914, - "hockey-puck": 62547, - "holly-berry": 63402, - "home": 61461, - "hooli": 62503, - "hornbill": 62866, - "horse": 63216, - "horse-head": 63403, - "hospital": 61688, - "hospital-alt": 62589, - "hospital-symbol": 62590, - "hot-tub": 62867, - "hotdog": 63503, - "hotel": 62868, - "hotjar": 62385, - "hourglass": 62036, - "hourglass-end": 62035, - "hourglass-half": 62034, - "hourglass-start": 62033, - "house-damage": 63217, - "houzz": 62076, - "hryvnia": 63218, - "html5": 61755, - "hubspot": 62386, - "i-cursor": 62022, - "ice-cream": 63504, - "icicles": 63405, - "id-badge": 62145, - "id-card": 62146, - "id-card-alt": 62591, - "igloo": 63406, - "image": 61502, - "images": 62210, - "imdb": 62168, - "inbox": 61468, - "indent": 61500, - "industry": 62069, - "infinity": 62772, - "info": 61737, - "info-circle": 61530, - "instagram": 61805, - "intercom": 63407, - "internet-explorer": 62059, - "invision": 63408, - "ioxhost": 61960, - "italic": 61491, - "itunes": 62388, - "itunes-note": 62389, - "java": 62692, - "jedi": 63081, - "jedi-order": 62734, - "jenkins": 62390, - "jira": 63409, - "joget": 62391, - "joint": 62869, - "joomla": 61866, - "journal-whills": 63082, - "js": 62392, - "js-square": 62393, - "jsfiddle": 61900, - "kaaba": 63083, - "kaggle": 62970, - "key": 61572, - "keybase": 62709, - "keyboard": 61724, - "keycdn": 62394, - "khanda": 63085, - "kickstarter": 62395, - "kickstarter-k": 62396, - "kiss": 62870, - "kiss-beam": 62871, - "kiss-wink-heart": 62872, - "kiwi-bird": 62773, - "korvue": 62511, - "landmark": 63087, - "language": 61867, - "laptop": 61705, - "laptop-code": 62972, - "laptop-medical": 63506, - "laravel": 62397, - "lastfm": 61954, - "lastfm-square": 61955, - "laugh": 62873, - "laugh-beam": 62874, - "laugh-squint": 62875, - "laugh-wink": 62876, - "layer-group": 62973, - "leaf": 61548, - "leanpub": 61970, - "lemon": 61588, - "less": 62493, - "less-than": 62774, - "less-than-equal": 62775, - "level-down-alt": 62398, - "level-up-alt": 62399, - "life-ring": 61901, - "lightbulb": 61675, - "line": 62400, - "link": 61633, - "linkedin": 61580, - "linkedin-in": 61665, - "linode": 62136, - "linux": 61820, - "lira-sign": 61845, - "list": 61498, - "list-alt": 61474, - "list-ol": 61643, - "list-ul": 61642, - "location-arrow": 61732, - "lock": 61475, - "lock-open": 62401, - "long-arrow-alt-down": 62217, - "long-arrow-alt-left": 62218, - "long-arrow-alt-right": 62219, - "long-arrow-alt-up": 62220, - "low-vision": 62120, - "luggage-cart": 62877, - "lyft": 62403, - "magento": 62404, - "magic": 61648, - "magnet": 61558, - "mail-bulk": 63092, - "mailchimp": 62878, - "male": 61827, - "mandalorian": 62735, - "map": 62073, - "map-marked": 62879, - "map-marked-alt": 62880, - "map-marker": 61505, - "map-marker-alt": 62405, - "map-pin": 62070, - "map-signs": 62071, - "markdown": 62991, - "marker": 62881, - "mars": 61986, - "mars-double": 61991, - "mars-stroke": 61993, - "mars-stroke-h": 61995, - "mars-stroke-v": 61994, - "mask": 63226, - "mastodon": 62710, - "maxcdn": 61750, - "medal": 62882, - "medapps": 62406, - "medium": 62010, - "medium-m": 62407, - "medkit": 61690, - "medrt": 62408, - "meetup": 62176, - "megaport": 62883, - "meh": 61722, - "meh-blank": 62884, - "meh-rolling-eyes": 62885, - "memory": 62776, - "mendeley": 63411, - "menorah": 63094, - "mercury": 61987, - "meteor": 63315, - "microchip": 62171, - "microphone": 61744, - "microphone-alt": 62409, - "microphone-alt-slash": 62777, - "microphone-slash": 61745, - "microscope": 62992, - "microsoft": 62410, - "minus": 61544, - "minus-circle": 61526, - "minus-square": 61766, - "mitten": 63413, - "mix": 62411, - "mixcloud": 62089, - "mizuni": 62412, - "mobile": 61707, - "mobile-alt": 62413, - "modx": 62085, - "monero": 62416, - "money-bill": 61654, - "money-bill-alt": 62417, - "money-bill-wave": 62778, - "money-bill-wave-alt": 62779, - "money-check": 62780, - "money-check-alt": 62781, - "monument": 62886, - "moon": 61830, - "mortar-pestle": 62887, - "mosque": 63096, - "motorcycle": 61980, - "mountain": 63228, - "mouse-pointer": 62021, - "mug-hot": 63414, - "music": 61441, - "napster": 62418, - "neos": 62994, - "network-wired": 63231, - "neuter": 61996, - "newspaper": 61930, - "nimblr": 62888, - "nintendo-switch": 62488, - "node": 62489, - "node-js": 62419, - "not-equal": 62782, - "notes-medical": 62593, - "npm": 62420, - "ns8": 62421, - "nutritionix": 62422, - "object-group": 62023, - "object-ungroup": 62024, - "odnoklassniki": 62051, - "odnoklassniki-square": 62052, - "oil-can": 62995, - "old-republic": 62736, - "om": 63097, - "opencart": 62013, - "openid": 61851, - "opera": 62058, - "optin-monster": 62012, - "osi": 62490, - "otter": 63232, - "outdent": 61499, - "page4": 62423, - "pagelines": 61836, - "pager": 63509, - "paint-brush": 61948, - "paint-roller": 62890, - "palette": 62783, - "palfed": 62424, - "pallet": 62594, - "paper-plane": 61912, - "paperclip": 61638, - "parachute-box": 62669, - "paragraph": 61917, - "parking": 62784, - "passport": 62891, - "pastafarianism": 63099, - "paste": 61674, - "patreon": 62425, - "pause": 61516, - "pause-circle": 62091, - "paw": 61872, - "paypal": 61933, - "peace": 63100, - "pen": 62212, - "pen-alt": 62213, - "pen-fancy": 62892, - "pen-nib": 62893, - "pen-square": 61771, - "pencil-alt": 62211, - "pencil-ruler": 62894, - "penny-arcade": 63236, - "people-carry": 62670, - "pepper-hot": 63510, - "percent": 62101, - "percentage": 62785, - "periscope": 62426, - "person-booth": 63318, - "phabricator": 62427, - "phoenix-framework": 62428, - "phoenix-squadron": 62737, - "phone": 61589, - "phone-slash": 62429, - "phone-square": 61592, - "phone-volume": 62112, - "php": 62551, - "pied-piper": 62126, - "pied-piper-alt": 61864, - "pied-piper-hat": 62693, - "pied-piper-pp": 61863, - "piggy-bank": 62675, - "pills": 62596, - "pinterest": 61650, - "pinterest-p": 62001, - "pinterest-square": 61651, - "pizza-slice": 63512, - "place-of-worship": 63103, - "plane": 61554, - "plane-arrival": 62895, - "plane-departure": 62896, - "play": 61515, - "play-circle": 61764, - "playstation": 62431, - "plug": 61926, - "plus": 61543, - "plus-circle": 61525, - "plus-square": 61694, - "podcast": 62158, - "poll": 63105, - "poll-h": 63106, - "poo": 62206, - "poo-storm": 63322, - "poop": 63001, - "portrait": 62432, - "pound-sign": 61780, - "power-off": 61457, - "pray": 63107, - "praying-hands": 63108, - "prescription": 62897, - "prescription-bottle": 62597, - "prescription-bottle-alt": 62598, - "print": 61487, - "procedures": 62599, - "product-hunt": 62088, - "project-diagram": 62786, - "pushed": 62433, - "puzzle-piece": 61742, - "python": 62434, - "qq": 61910, - "qrcode": 61481, - "question": 61736, - "question-circle": 61529, - "quidditch": 62552, - "quinscape": 62553, - "quora": 62148, - "quote-left": 61709, - "quote-right": 61710, - "quran": 63111, - "r-project": 62711, - "radiation": 63417, - "radiation-alt": 63418, - "rainbow": 63323, - "random": 61556, - "raspberry-pi": 63419, - "ravelry": 62169, - "react": 62491, - "reacteurope": 63325, - "readme": 62677, - "rebel": 61904, - "receipt": 62787, - "recycle": 61880, - "red-river": 62435, - "reddit": 61857, - "reddit-alien": 62081, - "reddit-square": 61858, - "redhat": 63420, - "redo": 61470, - "redo-alt": 62201, - "registered": 62045, - "renren": 61835, - "reply": 62437, - "reply-all": 61730, - "replyd": 62438, - "republican": 63326, - "researchgate": 62712, - "resolving": 62439, - "restroom": 63421, - "retweet": 61561, - "rev": 62898, - "ribbon": 62678, - "ring": 63243, - "road": 61464, - "robot": 62788, - "rocket": 61749, - "rocketchat": 62440, - "rockrms": 62441, - "route": 62679, - "rss": 61598, - "rss-square": 61763, - "ruble-sign": 61784, - "ruler": 62789, - "ruler-combined": 62790, - "ruler-horizontal": 62791, - "ruler-vertical": 62792, - "running": 63244, - "rupee-sign": 61782, - "sad-cry": 62899, - "sad-tear": 62900, - "safari": 62055, - "sass": 62494, - "satellite": 63423, - "satellite-dish": 63424, - "save": 61639, - "schlix": 62442, - "school": 62793, - "screwdriver": 62794, - "scribd": 62090, - "scroll": 63246, - "sd-card": 63426, - "search": 61442, - "search-dollar": 63112, - "search-location": 63113, - "search-minus": 61456, - "search-plus": 61454, - "searchengin": 62443, - "seedling": 62680, - "sellcast": 62170, - "sellsy": 61971, - "server": 62003, - "servicestack": 62444, - "shapes": 63007, - "share": 61540, - "share-alt": 61920, - "share-alt-square": 61921, - "share-square": 61773, - "shekel-sign": 61963, - "shield-alt": 62445, - "ship": 61978, - "shipping-fast": 62603, - "shirtsinbulk": 61972, - "shoe-prints": 62795, - "shopping-bag": 62096, - "shopping-basket": 62097, - "shopping-cart": 61562, - "shopware": 62901, - "shower": 62156, - "shuttle-van": 62902, - "sign": 62681, - "sign-in-alt": 62198, - "sign-language": 62119, - "sign-out-alt": 62197, - "signal": 61458, - "signature": 62903, - "sim-card": 63428, - "simplybuilt": 61973, - "sistrix": 62446, - "sitemap": 61672, - "sith": 62738, - "skating": 63429, - "sketch": 63430, - "skiing": 63433, - "skiing-nordic": 63434, - "skull": 62796, - "skull-crossbones": 63252, - "skyatlas": 61974, - "skype": 61822, - "slack": 61848, - "slack-hash": 62447, - "slash": 63253, - "sleigh": 63436, - "sliders-h": 61918, - "slideshare": 61927, - "smile": 61720, - "smile-beam": 62904, - "smile-wink": 62682, - "smog": 63327, - "smoking": 62605, - "smoking-ban": 62797, - "sms": 63437, - "snapchat": 62123, - "snapchat-ghost": 62124, - "snapchat-square": 62125, - "snowboarding": 63438, - "snowflake": 62172, - "snowman": 63440, - "snowplow": 63442, - "socks": 63126, - "solar-panel": 62906, - "sort": 61660, - "sort-alpha-down": 61789, - "sort-alpha-up": 61790, - "sort-amount-down": 61792, - "sort-amount-up": 61793, - "sort-down": 61661, - "sort-numeric-down": 61794, - "sort-numeric-up": 61795, - "sort-up": 61662, - "soundcloud": 61886, - "sourcetree": 63443, - "spa": 62907, - "space-shuttle": 61847, - "speakap": 62451, - "spider": 63255, - "spinner": 61712, - "splotch": 62908, - "spotify": 61884, - "spray-can": 62909, - "square": 61640, - "square-full": 62556, - "square-root-alt": 63128, - "squarespace": 62910, - "stack-exchange": 61837, - "stack-overflow": 61804, - "stamp": 62911, - "star": 61445, - "star-and-crescent": 63129, - "star-half": 61577, - "star-half-alt": 62912, - "star-of-david": 63130, - "star-of-life": 63009, - "staylinked": 62453, - "steam": 61878, - "steam-square": 61879, - "steam-symbol": 62454, - "step-backward": 61512, - "step-forward": 61521, - "stethoscope": 61681, - "sticker-mule": 62455, - "sticky-note": 62025, - "stop": 61517, - "stop-circle": 62093, - "stopwatch": 62194, - "store": 62798, - "store-alt": 62799, - "strava": 62504, - "stream": 62800, - "street-view": 61981, - "strikethrough": 61644, - "stripe": 62505, - "stripe-s": 62506, - "stroopwafel": 62801, - "studiovinari": 62456, - "stumbleupon": 61860, - "stumbleupon-circle": 61859, - "subscript": 61740, - "subway": 62009, - "suitcase": 61682, - "suitcase-rolling": 62913, - "sun": 61829, - "superpowers": 62173, - "superscript": 61739, - "supple": 62457, - "surprise": 62914, - "suse": 63446, - "swatchbook": 62915, - "swimmer": 62916, - "swimming-pool": 62917, - "synagogue": 63131, - "sync": 61473, - "sync-alt": 62193, - "syringe": 62606, - "table": 61646, - "table-tennis": 62557, - "tablet": 61706, - "tablet-alt": 62458, - "tablets": 62608, - "tachometer-alt": 62461, - "tag": 61483, - "tags": 61484, - "tape": 62683, - "tasks": 61614, - "taxi": 61882, - "teamspeak": 62713, - "teeth": 63022, - "teeth-open": 63023, - "telegram": 62150, - "telegram-plane": 62462, - "temperature-high": 63337, - "temperature-low": 63339, - "tencent-weibo": 61909, - "tenge": 63447, - "terminal": 61728, - "text-height": 61492, - "text-width": 61493, - "th": 61450, - "th-large": 61449, - "th-list": 61451, - "the-red-yeti": 63133, - "theater-masks": 63024, - "themeco": 62918, - "themeisle": 62130, - "thermometer": 62609, - "thermometer-empty": 62155, - "thermometer-full": 62151, - "thermometer-half": 62153, - "thermometer-quarter": 62154, - "thermometer-three-quarters": 62152, - "think-peaks": 63281, - "thumbs-down": 61797, - "thumbs-up": 61796, - "thumbtack": 61581, - "ticket-alt": 62463, - "times": 61453, - "times-circle": 61527, - "tint": 61507, - "tint-slash": 62919, - "tired": 62920, - "toggle-off": 61956, - "toggle-on": 61957, - "toilet": 63448, - "toilet-paper": 63262, - "toolbox": 62802, - "tools": 63449, - "tooth": 62921, - "torah": 63136, - "torii-gate": 63137, - "tractor": 63266, - "trade-federation": 62739, - "trademark": 62044, - "traffic-light": 63031, - "train": 62008, - "tram": 63450, - "transgender": 61988, - "transgender-alt": 61989, - "trash": 61944, - "trash-alt": 62189, - "trash-restore": 63529, - "trash-restore-alt": 63530, - "tree": 61883, - "trello": 61825, - "tripadvisor": 62050, - "trophy": 61585, - "truck": 61649, - "truck-loading": 62686, - "truck-monster": 63035, - "truck-moving": 62687, - "truck-pickup": 63036, - "tshirt": 62803, - "tty": 61924, - "tumblr": 61811, - "tumblr-square": 61812, - "tv": 62060, - "twitch": 61928, - "twitter": 61593, - "twitter-square": 61569, - "typo3": 62507, - "uber": 62466, - "ubuntu": 63455, - "uikit": 62467, - "umbrella": 61673, - "umbrella-beach": 62922, - "underline": 61645, - "undo": 61666, - "undo-alt": 62186, - "uniregistry": 62468, - "universal-access": 62106, - "university": 61852, - "unlink": 61735, - "unlock": 61596, - "unlock-alt": 61758, - "untappd": 62469, - "upload": 61587, - "ups": 63456, - "usb": 62087, - "user": 61447, - "user-alt": 62470, - "user-alt-slash": 62714, - "user-astronaut": 62715, - "user-check": 62716, - "user-circle": 62141, - "user-clock": 62717, - "user-cog": 62718, - "user-edit": 62719, - "user-friends": 62720, - "user-graduate": 62721, - "user-injured": 63272, - "user-lock": 62722, - "user-md": 61680, - "user-minus": 62723, - "user-ninja": 62724, - "user-nurse": 63535, - "user-plus": 62004, - "user-secret": 61979, - "user-shield": 62725, - "user-slash": 62726, - "user-tag": 62727, - "user-tie": 62728, - "user-times": 62005, - "users": 61632, - "users-cog": 62729, - "usps": 63457, - "ussunnah": 62471, - "utensil-spoon": 62181, - "utensils": 62183, - "vaadin": 62472, - "vector-square": 62923, - "venus": 61985, - "venus-double": 61990, - "venus-mars": 61992, - "viacoin": 62007, - "viadeo": 62121, - "viadeo-square": 62122, - "vial": 62610, - "vials": 62611, - "viber": 62473, - "video": 61501, - "video-slash": 62690, - "vihara": 63143, - "vimeo": 62474, - "vimeo-square": 61844, - "vimeo-v": 62077, - "vine": 61898, - "vk": 61833, - "vnv": 62475, - "volleyball-ball": 62559, - "volume-down": 61479, - "volume-mute": 63145, - "volume-off": 61478, - "volume-up": 61480, - "vote-yea": 63346, - "vr-cardboard": 63273, - "vuejs": 62495, - "walking": 62804, - "wallet": 62805, - "warehouse": 62612, - "water": 63347, - "weebly": 62924, - "weibo": 61834, - "weight": 62614, - "weight-hanging": 62925, - "weixin": 61911, - "whatsapp": 62002, - "whatsapp-square": 62476, - "wheelchair": 61843, - "whmcs": 62477, - "wifi": 61931, - "wikipedia-w": 62054, - "wind": 63278, - "window-close": 62480, - "window-maximize": 62160, - "window-minimize": 62161, - "window-restore": 62162, - "windows": 61818, - "wine-bottle": 63279, - "wine-glass": 62691, - "wine-glass-alt": 62926, - "wix": 62927, - "wizards-of-the-coast": 63280, - "wolf-pack-battalion": 62740, - "won-sign": 61785, - "wordpress": 61850, - "wordpress-simple": 62481, - "wpbeginner": 62103, - "wpexplorer": 62174, - "wpforms": 62104, - "wpressr": 62436, - "wrench": 61613, - "x-ray": 62615, - "xbox": 62482, - "xing": 61800, - "xing-square": 61801, - "y-combinator": 62011, - "yahoo": 61854, - "yandex": 62483, - "yandex-international": 62484, - "yarn": 63459, - "yelp": 61929, - "yen-sign": 61783, - "yin-yang": 63149, - "yoast": 62129, - "youtube": 61799, - "youtube-square": 62513, - "zhihu": 63039 -} \ No newline at end of file diff --git a/app/src/main/res/raw/node_modules_reactnativevectoricons_glyphmaps_fontawesome5free_meta.json b/app/src/main/res/raw/node_modules_reactnativevectoricons_glyphmaps_fontawesome5free_meta.json deleted file mode 100644 index 051deea6..00000000 --- a/app/src/main/res/raw/node_modules_reactnativevectoricons_glyphmaps_fontawesome5free_meta.json +++ /dev/null @@ -1,1511 +0,0 @@ -{ - "brands": [ - "500px", - "accessible-icon", - "accusoft", - "acquisitions-incorporated", - "adn", - "adobe", - "adversal", - "affiliatetheme", - "algolia", - "alipay", - "amazon-pay", - "amazon", - "amilia", - "android", - "angellist", - "angrycreative", - "angular", - "app-store-ios", - "app-store", - "apper", - "apple-pay", - "apple", - "artstation", - "asymmetrik", - "atlassian", - "audible", - "autoprefixer", - "avianex", - "aviato", - "aws", - "bandcamp", - "behance-square", - "behance", - "bimobject", - "bitbucket", - "bitcoin", - "bity", - "black-tie", - "blackberry", - "blogger-b", - "blogger", - "bluetooth-b", - "bluetooth", - "btc", - "buromobelexperte", - "buysellads", - "canadian-maple-leaf", - "cc-amazon-pay", - "cc-amex", - "cc-apple-pay", - "cc-diners-club", - "cc-discover", - "cc-jcb", - "cc-mastercard", - "cc-paypal", - "cc-stripe", - "cc-visa", - "centercode", - "centos", - "chrome", - "cloudscale", - "cloudsmith", - "cloudversify", - "codepen", - "codiepie", - "confluence", - "connectdevelop", - "contao", - "cpanel", - "creative-commons-by", - "creative-commons-nc-eu", - "creative-commons-nc-jp", - "creative-commons-nc", - "creative-commons-nd", - "creative-commons-pd-alt", - "creative-commons-pd", - "creative-commons-remix", - "creative-commons-sa", - "creative-commons-sampling-plus", - "creative-commons-sampling", - "creative-commons-share", - "creative-commons-zero", - "creative-commons", - "critical-role", - "css3-alt", - "css3", - "cuttlefish", - "d-and-d-beyond", - "d-and-d", - "dashcube", - "delicious", - "deploydog", - "deskpro", - "dev", - "deviantart", - "dhl", - "diaspora", - "digg", - "digital-ocean", - "discord", - "discourse", - "dochub", - "docker", - "draft2digital", - "dribbble-square", - "dribbble", - "dropbox", - "drupal", - "dyalog", - "earlybirds", - "ebay", - "edge", - "elementor", - "ello", - "ember", - "empire", - "envira", - "erlang", - "ethereum", - "etsy", - "expeditedssl", - "facebook-f", - "facebook-messenger", - "facebook-square", - "facebook", - "fantasy-flight-games", - "fedex", - "fedora", - "figma", - "firefox", - "first-order-alt", - "first-order", - "firstdraft", - "flickr", - "flipboard", - "fly", - "font-awesome-alt", - "font-awesome-flag", - "font-awesome-logo-full", - "font-awesome", - "fonticons-fi", - "fonticons", - "fort-awesome-alt", - "fort-awesome", - "forumbee", - "foursquare", - "free-code-camp", - "freebsd", - "fulcrum", - "galactic-republic", - "galactic-senate", - "get-pocket", - "gg-circle", - "gg", - "git-square", - "git", - "github-alt", - "github-square", - "github", - "gitkraken", - "gitlab", - "gitter", - "glide-g", - "glide", - "gofore", - "goodreads-g", - "goodreads", - "google-drive", - "google-play", - "google-plus-g", - "google-plus-square", - "google-plus", - "google-wallet", - "google", - "gratipay", - "grav", - "gripfire", - "grunt", - "gulp", - "hacker-news-square", - "hacker-news", - "hackerrank", - "hips", - "hire-a-helper", - "hooli", - "hornbill", - "hotjar", - "houzz", - "html5", - "hubspot", - "imdb", - "instagram", - "intercom", - "internet-explorer", - "invision", - "ioxhost", - "itunes-note", - "itunes", - "java", - "jedi-order", - "jenkins", - "jira", - "joget", - "joomla", - "js-square", - "js", - "jsfiddle", - "kaggle", - "keybase", - "keycdn", - "kickstarter-k", - "kickstarter", - "korvue", - "laravel", - "lastfm-square", - "lastfm", - "leanpub", - "less", - "line", - "linkedin-in", - "linkedin", - "linode", - "linux", - "lyft", - "magento", - "mailchimp", - "mandalorian", - "markdown", - "mastodon", - "maxcdn", - "medapps", - "medium-m", - "medium", - "medrt", - "meetup", - "megaport", - "mendeley", - "microsoft", - "mix", - "mixcloud", - "mizuni", - "modx", - "monero", - "napster", - "neos", - "nimblr", - "nintendo-switch", - "node-js", - "node", - "npm", - "ns8", - "nutritionix", - "odnoklassniki-square", - "odnoklassniki", - "old-republic", - "opencart", - "openid", - "opera", - "optin-monster", - "osi", - "page4", - "pagelines", - "palfed", - "patreon", - "paypal", - "penny-arcade", - "periscope", - "phabricator", - "phoenix-framework", - "phoenix-squadron", - "php", - "pied-piper-alt", - "pied-piper-hat", - "pied-piper-pp", - "pied-piper", - "pinterest-p", - "pinterest-square", - "pinterest", - "playstation", - "product-hunt", - "pushed", - "python", - "qq", - "quinscape", - "quora", - "r-project", - "raspberry-pi", - "ravelry", - "react", - "reacteurope", - "readme", - "rebel", - "red-river", - "reddit-alien", - "reddit-square", - "reddit", - "redhat", - "renren", - "replyd", - "researchgate", - "resolving", - "rev", - "rocketchat", - "rockrms", - "safari", - "sass", - "schlix", - "scribd", - "searchengin", - "sellcast", - "sellsy", - "servicestack", - "shirtsinbulk", - "shopware", - "simplybuilt", - "sistrix", - "sith", - "sketch", - "skyatlas", - "skype", - "slack-hash", - "slack", - "slideshare", - "snapchat-ghost", - "snapchat-square", - "snapchat", - "soundcloud", - "sourcetree", - "speakap", - "spotify", - "squarespace", - "stack-exchange", - "stack-overflow", - "staylinked", - "steam-square", - "steam-symbol", - "steam", - "sticker-mule", - "strava", - "stripe-s", - "stripe", - "studiovinari", - "stumbleupon-circle", - "stumbleupon", - "superpowers", - "supple", - "suse", - "teamspeak", - "telegram-plane", - "telegram", - "tencent-weibo", - "the-red-yeti", - "themeco", - "themeisle", - "think-peaks", - "trade-federation", - "trello", - "tripadvisor", - "tumblr-square", - "tumblr", - "twitch", - "twitter-square", - "twitter", - "typo3", - "uber", - "ubuntu", - "uikit", - "uniregistry", - "untappd", - "ups", - "usb", - "usps", - "ussunnah", - "vaadin", - "viacoin", - "viadeo-square", - "viadeo", - "viber", - "vimeo-square", - "vimeo-v", - "vimeo", - "vine", - "vk", - "vnv", - "vuejs", - "weebly", - "weibo", - "weixin", - "whatsapp-square", - "whatsapp", - "whmcs", - "wikipedia-w", - "windows", - "wix", - "wizards-of-the-coast", - "wolf-pack-battalion", - "wordpress-simple", - "wordpress", - "wpbeginner", - "wpexplorer", - "wpforms", - "wpressr", - "xbox", - "xing-square", - "xing", - "y-combinator", - "yahoo", - "yandex-international", - "yandex", - "yarn", - "yelp", - "yoast", - "youtube-square", - "youtube", - "zhihu" - ], - "regular": [ - "address-book", - "address-card", - "angry", - "arrow-alt-circle-down", - "arrow-alt-circle-left", - "arrow-alt-circle-right", - "arrow-alt-circle-up", - "bell-slash", - "bell", - "bookmark", - "building", - "calendar-alt", - "calendar-check", - "calendar-minus", - "calendar-plus", - "calendar-times", - "calendar", - "caret-square-down", - "caret-square-left", - "caret-square-right", - "caret-square-up", - "chart-bar", - "check-circle", - "check-square", - "circle", - "clipboard", - "clock", - "clone", - "closed-captioning", - "comment-alt", - "comment-dots", - "comment", - "comments", - "compass", - "copy", - "copyright", - "credit-card", - "dizzy", - "dot-circle", - "edit", - "envelope-open", - "envelope", - "eye-slash", - "eye", - "file-alt", - "file-archive", - "file-audio", - "file-code", - "file-excel", - "file-image", - "file-pdf", - "file-powerpoint", - "file-video", - "file-word", - "file", - "flag", - "flushed", - "folder-open", - "folder", - "font-awesome-logo-full", - "frown-open", - "frown", - "futbol", - "gem", - "grimace", - "grin-alt", - "grin-beam-sweat", - "grin-beam", - "grin-hearts", - "grin-squint-tears", - "grin-squint", - "grin-stars", - "grin-tears", - "grin-tongue-squint", - "grin-tongue-wink", - "grin-tongue", - "grin-wink", - "grin", - "hand-lizard", - "hand-paper", - "hand-peace", - "hand-point-down", - "hand-point-left", - "hand-point-right", - "hand-point-up", - "hand-pointer", - "hand-rock", - "hand-scissors", - "hand-spock", - "handshake", - "hdd", - "heart", - "hospital", - "hourglass", - "id-badge", - "id-card", - "image", - "images", - "keyboard", - "kiss-beam", - "kiss-wink-heart", - "kiss", - "laugh-beam", - "laugh-squint", - "laugh-wink", - "laugh", - "lemon", - "life-ring", - "lightbulb", - "list-alt", - "map", - "meh-blank", - "meh-rolling-eyes", - "meh", - "minus-square", - "money-bill-alt", - "moon", - "newspaper", - "object-group", - "object-ungroup", - "paper-plane", - "pause-circle", - "play-circle", - "plus-square", - "question-circle", - "registered", - "sad-cry", - "sad-tear", - "save", - "share-square", - "smile-beam", - "smile-wink", - "smile", - "snowflake", - "square", - "star-half", - "star", - "sticky-note", - "stop-circle", - "sun", - "surprise", - "thumbs-down", - "thumbs-up", - "times-circle", - "tired", - "trash-alt", - "user-circle", - "user", - "window-close", - "window-maximize", - "window-minimize", - "window-restore" - ], - "solid": [ - "ad", - "address-book", - "address-card", - "adjust", - "air-freshener", - "align-center", - "align-justify", - "align-left", - "align-right", - "allergies", - "ambulance", - "american-sign-language-interpreting", - "anchor", - "angle-double-down", - "angle-double-left", - "angle-double-right", - "angle-double-up", - "angle-down", - "angle-left", - "angle-right", - "angle-up", - "angry", - "ankh", - "apple-alt", - "archive", - "archway", - "arrow-alt-circle-down", - "arrow-alt-circle-left", - "arrow-alt-circle-right", - "arrow-alt-circle-up", - "arrow-circle-down", - "arrow-circle-left", - "arrow-circle-right", - "arrow-circle-up", - "arrow-down", - "arrow-left", - "arrow-right", - "arrow-up", - "arrows-alt-h", - "arrows-alt-v", - "arrows-alt", - "assistive-listening-systems", - "asterisk", - "at", - "atlas", - "atom", - "audio-description", - "award", - "baby-carriage", - "baby", - "backspace", - "backward", - "bacon", - "balance-scale", - "ban", - "band-aid", - "barcode", - "bars", - "baseball-ball", - "basketball-ball", - "bath", - "battery-empty", - "battery-full", - "battery-half", - "battery-quarter", - "battery-three-quarters", - "bed", - "beer", - "bell-slash", - "bell", - "bezier-curve", - "bible", - "bicycle", - "binoculars", - "biohazard", - "birthday-cake", - "blender-phone", - "blender", - "blind", - "blog", - "bold", - "bolt", - "bomb", - "bone", - "bong", - "book-dead", - "book-medical", - "book-open", - "book-reader", - "book", - "bookmark", - "bowling-ball", - "box-open", - "box", - "boxes", - "braille", - "brain", - "bread-slice", - "briefcase-medical", - "briefcase", - "broadcast-tower", - "broom", - "brush", - "bug", - "building", - "bullhorn", - "bullseye", - "burn", - "bus-alt", - "bus", - "business-time", - "calculator", - "calendar-alt", - "calendar-check", - "calendar-day", - "calendar-minus", - "calendar-plus", - "calendar-times", - "calendar-week", - "calendar", - "camera-retro", - "camera", - "campground", - "candy-cane", - "cannabis", - "capsules", - "car-alt", - "car-battery", - "car-crash", - "car-side", - "car", - "caret-down", - "caret-left", - "caret-right", - "caret-square-down", - "caret-square-left", - "caret-square-right", - "caret-square-up", - "caret-up", - "carrot", - "cart-arrow-down", - "cart-plus", - "cash-register", - "cat", - "certificate", - "chair", - "chalkboard-teacher", - "chalkboard", - "charging-station", - "chart-area", - "chart-bar", - "chart-line", - "chart-pie", - "check-circle", - "check-double", - "check-square", - "check", - "cheese", - "chess-bishop", - "chess-board", - "chess-king", - "chess-knight", - "chess-pawn", - "chess-queen", - "chess-rook", - "chess", - "chevron-circle-down", - "chevron-circle-left", - "chevron-circle-right", - "chevron-circle-up", - "chevron-down", - "chevron-left", - "chevron-right", - "chevron-up", - "child", - "church", - "circle-notch", - "circle", - "city", - "clinic-medical", - "clipboard-check", - "clipboard-list", - "clipboard", - "clock", - "clone", - "closed-captioning", - "cloud-download-alt", - "cloud-meatball", - "cloud-moon-rain", - "cloud-moon", - "cloud-rain", - "cloud-showers-heavy", - "cloud-sun-rain", - "cloud-sun", - "cloud-upload-alt", - "cloud", - "cocktail", - "code-branch", - "code", - "coffee", - "cog", - "cogs", - "coins", - "columns", - "comment-alt", - "comment-dollar", - "comment-dots", - "comment-medical", - "comment-slash", - "comment", - "comments-dollar", - "comments", - "compact-disc", - "compass", - "compress-arrows-alt", - "compress", - "concierge-bell", - "cookie-bite", - "cookie", - "copy", - "copyright", - "couch", - "credit-card", - "crop-alt", - "crop", - "cross", - "crosshairs", - "crow", - "crown", - "crutch", - "cube", - "cubes", - "cut", - "database", - "deaf", - "democrat", - "desktop", - "dharmachakra", - "diagnoses", - "dice-d20", - "dice-d6", - "dice-five", - "dice-four", - "dice-one", - "dice-six", - "dice-three", - "dice-two", - "dice", - "digital-tachograph", - "directions", - "divide", - "dizzy", - "dna", - "dog", - "dollar-sign", - "dolly-flatbed", - "dolly", - "donate", - "door-closed", - "door-open", - "dot-circle", - "dove", - "download", - "drafting-compass", - "dragon", - "draw-polygon", - "drum-steelpan", - "drum", - "drumstick-bite", - "dumbbell", - "dumpster-fire", - "dumpster", - "dungeon", - "edit", - "egg", - "eject", - "ellipsis-h", - "ellipsis-v", - "envelope-open-text", - "envelope-open", - "envelope-square", - "envelope", - "equals", - "eraser", - "ethernet", - "euro-sign", - "exchange-alt", - "exclamation-circle", - "exclamation-triangle", - "exclamation", - "expand-arrows-alt", - "expand", - "external-link-alt", - "external-link-square-alt", - "eye-dropper", - "eye-slash", - "eye", - "fast-backward", - "fast-forward", - "fax", - "feather-alt", - "feather", - "female", - "fighter-jet", - "file-alt", - "file-archive", - "file-audio", - "file-code", - "file-contract", - "file-csv", - "file-download", - "file-excel", - "file-export", - "file-image", - "file-import", - "file-invoice-dollar", - "file-invoice", - "file-medical-alt", - "file-medical", - "file-pdf", - "file-powerpoint", - "file-prescription", - "file-signature", - "file-upload", - "file-video", - "file-word", - "file", - "fill-drip", - "fill", - "film", - "filter", - "fingerprint", - "fire-alt", - "fire-extinguisher", - "fire", - "first-aid", - "fish", - "fist-raised", - "flag-checkered", - "flag-usa", - "flag", - "flask", - "flushed", - "folder-minus", - "folder-open", - "folder-plus", - "folder", - "font-awesome-logo-full", - "font", - "football-ball", - "forward", - "frog", - "frown-open", - "frown", - "funnel-dollar", - "futbol", - "gamepad", - "gas-pump", - "gavel", - "gem", - "genderless", - "ghost", - "gift", - "gifts", - "glass-cheers", - "glass-martini-alt", - "glass-martini", - "glass-whiskey", - "glasses", - "globe-africa", - "globe-americas", - "globe-asia", - "globe-europe", - "globe", - "golf-ball", - "gopuram", - "graduation-cap", - "greater-than-equal", - "greater-than", - "grimace", - "grin-alt", - "grin-beam-sweat", - "grin-beam", - "grin-hearts", - "grin-squint-tears", - "grin-squint", - "grin-stars", - "grin-tears", - "grin-tongue-squint", - "grin-tongue-wink", - "grin-tongue", - "grin-wink", - "grin", - "grip-horizontal", - "grip-lines-vertical", - "grip-lines", - "grip-vertical", - "guitar", - "h-square", - "hamburger", - "hammer", - "hamsa", - "hand-holding-heart", - "hand-holding-usd", - "hand-holding", - "hand-lizard", - "hand-middle-finger", - "hand-paper", - "hand-peace", - "hand-point-down", - "hand-point-left", - "hand-point-right", - "hand-point-up", - "hand-pointer", - "hand-rock", - "hand-scissors", - "hand-spock", - "hands-helping", - "hands", - "handshake", - "hanukiah", - "hard-hat", - "hashtag", - "hat-wizard", - "haykal", - "hdd", - "heading", - "headphones-alt", - "headphones", - "headset", - "heart-broken", - "heart", - "heartbeat", - "helicopter", - "highlighter", - "hiking", - "hippo", - "history", - "hockey-puck", - "holly-berry", - "home", - "horse-head", - "horse", - "hospital-alt", - "hospital-symbol", - "hospital", - "hot-tub", - "hotdog", - "hotel", - "hourglass-end", - "hourglass-half", - "hourglass-start", - "hourglass", - "house-damage", - "hryvnia", - "i-cursor", - "ice-cream", - "icicles", - "id-badge", - "id-card-alt", - "id-card", - "igloo", - "image", - "images", - "inbox", - "indent", - "industry", - "infinity", - "info-circle", - "info", - "italic", - "jedi", - "joint", - "journal-whills", - "kaaba", - "key", - "keyboard", - "khanda", - "kiss-beam", - "kiss-wink-heart", - "kiss", - "kiwi-bird", - "landmark", - "language", - "laptop-code", - "laptop-medical", - "laptop", - "laugh-beam", - "laugh-squint", - "laugh-wink", - "laugh", - "layer-group", - "leaf", - "lemon", - "less-than-equal", - "less-than", - "level-down-alt", - "level-up-alt", - "life-ring", - "lightbulb", - "link", - "lira-sign", - "list-alt", - "list-ol", - "list-ul", - "list", - "location-arrow", - "lock-open", - "lock", - "long-arrow-alt-down", - "long-arrow-alt-left", - "long-arrow-alt-right", - "long-arrow-alt-up", - "low-vision", - "luggage-cart", - "magic", - "magnet", - "mail-bulk", - "male", - "map-marked-alt", - "map-marked", - "map-marker-alt", - "map-marker", - "map-pin", - "map-signs", - "map", - "marker", - "mars-double", - "mars-stroke-h", - "mars-stroke-v", - "mars-stroke", - "mars", - "mask", - "medal", - "medkit", - "meh-blank", - "meh-rolling-eyes", - "meh", - "memory", - "menorah", - "mercury", - "meteor", - "microchip", - "microphone-alt-slash", - "microphone-alt", - "microphone-slash", - "microphone", - "microscope", - "minus-circle", - "minus-square", - "minus", - "mitten", - "mobile-alt", - "mobile", - "money-bill-alt", - "money-bill-wave-alt", - "money-bill-wave", - "money-bill", - "money-check-alt", - "money-check", - "monument", - "moon", - "mortar-pestle", - "mosque", - "motorcycle", - "mountain", - "mouse-pointer", - "mug-hot", - "music", - "network-wired", - "neuter", - "newspaper", - "not-equal", - "notes-medical", - "object-group", - "object-ungroup", - "oil-can", - "om", - "otter", - "outdent", - "pager", - "paint-brush", - "paint-roller", - "palette", - "pallet", - "paper-plane", - "paperclip", - "parachute-box", - "paragraph", - "parking", - "passport", - "pastafarianism", - "paste", - "pause-circle", - "pause", - "paw", - "peace", - "pen-alt", - "pen-fancy", - "pen-nib", - "pen-square", - "pen", - "pencil-alt", - "pencil-ruler", - "people-carry", - "pepper-hot", - "percent", - "percentage", - "person-booth", - "phone-slash", - "phone-square", - "phone-volume", - "phone", - "piggy-bank", - "pills", - "pizza-slice", - "place-of-worship", - "plane-arrival", - "plane-departure", - "plane", - "play-circle", - "play", - "plug", - "plus-circle", - "plus-square", - "plus", - "podcast", - "poll-h", - "poll", - "poo-storm", - "poo", - "poop", - "portrait", - "pound-sign", - "power-off", - "pray", - "praying-hands", - "prescription-bottle-alt", - "prescription-bottle", - "prescription", - "print", - "procedures", - "project-diagram", - "puzzle-piece", - "qrcode", - "question-circle", - "question", - "quidditch", - "quote-left", - "quote-right", - "quran", - "radiation-alt", - "radiation", - "rainbow", - "random", - "receipt", - "recycle", - "redo-alt", - "redo", - "registered", - "reply-all", - "reply", - "republican", - "restroom", - "retweet", - "ribbon", - "ring", - "road", - "robot", - "rocket", - "route", - "rss-square", - "rss", - "ruble-sign", - "ruler-combined", - "ruler-horizontal", - "ruler-vertical", - "ruler", - "running", - "rupee-sign", - "sad-cry", - "sad-tear", - "satellite-dish", - "satellite", - "save", - "school", - "screwdriver", - "scroll", - "sd-card", - "search-dollar", - "search-location", - "search-minus", - "search-plus", - "search", - "seedling", - "server", - "shapes", - "share-alt-square", - "share-alt", - "share-square", - "share", - "shekel-sign", - "shield-alt", - "ship", - "shipping-fast", - "shoe-prints", - "shopping-bag", - "shopping-basket", - "shopping-cart", - "shower", - "shuttle-van", - "sign-in-alt", - "sign-language", - "sign-out-alt", - "sign", - "signal", - "signature", - "sim-card", - "sitemap", - "skating", - "skiing-nordic", - "skiing", - "skull-crossbones", - "skull", - "slash", - "sleigh", - "sliders-h", - "smile-beam", - "smile-wink", - "smile", - "smog", - "smoking-ban", - "smoking", - "sms", - "snowboarding", - "snowflake", - "snowman", - "snowplow", - "socks", - "solar-panel", - "sort-alpha-down", - "sort-alpha-up", - "sort-amount-down", - "sort-amount-up", - "sort-down", - "sort-numeric-down", - "sort-numeric-up", - "sort-up", - "sort", - "spa", - "space-shuttle", - "spider", - "spinner", - "splotch", - "spray-can", - "square-full", - "square-root-alt", - "square", - "stamp", - "star-and-crescent", - "star-half-alt", - "star-half", - "star-of-david", - "star-of-life", - "star", - "step-backward", - "step-forward", - "stethoscope", - "sticky-note", - "stop-circle", - "stop", - "stopwatch", - "store-alt", - "store", - "stream", - "street-view", - "strikethrough", - "stroopwafel", - "subscript", - "subway", - "suitcase-rolling", - "suitcase", - "sun", - "superscript", - "surprise", - "swatchbook", - "swimmer", - "swimming-pool", - "synagogue", - "sync-alt", - "sync", - "syringe", - "table-tennis", - "table", - "tablet-alt", - "tablet", - "tablets", - "tachometer-alt", - "tag", - "tags", - "tape", - "tasks", - "taxi", - "teeth-open", - "teeth", - "temperature-high", - "temperature-low", - "tenge", - "terminal", - "text-height", - "text-width", - "th-large", - "th-list", - "th", - "theater-masks", - "thermometer-empty", - "thermometer-full", - "thermometer-half", - "thermometer-quarter", - "thermometer-three-quarters", - "thermometer", - "thumbs-down", - "thumbs-up", - "thumbtack", - "ticket-alt", - "times-circle", - "times", - "tint-slash", - "tint", - "tired", - "toggle-off", - "toggle-on", - "toilet-paper", - "toilet", - "toolbox", - "tools", - "tooth", - "torah", - "torii-gate", - "tractor", - "trademark", - "traffic-light", - "train", - "tram", - "transgender-alt", - "transgender", - "trash-alt", - "trash-restore-alt", - "trash-restore", - "trash", - "tree", - "trophy", - "truck-loading", - "truck-monster", - "truck-moving", - "truck-pickup", - "truck", - "tshirt", - "tty", - "tv", - "umbrella-beach", - "umbrella", - "underline", - "undo-alt", - "undo", - "universal-access", - "university", - "unlink", - "unlock-alt", - "unlock", - "upload", - "user-alt-slash", - "user-alt", - "user-astronaut", - "user-check", - "user-circle", - "user-clock", - "user-cog", - "user-edit", - "user-friends", - "user-graduate", - "user-injured", - "user-lock", - "user-md", - "user-minus", - "user-ninja", - "user-nurse", - "user-plus", - "user-secret", - "user-shield", - "user-slash", - "user-tag", - "user-tie", - "user-times", - "user", - "users-cog", - "users", - "utensil-spoon", - "utensils", - "vector-square", - "venus-double", - "venus-mars", - "venus", - "vial", - "vials", - "video-slash", - "video", - "vihara", - "volleyball-ball", - "volume-down", - "volume-mute", - "volume-off", - "volume-up", - "vote-yea", - "vr-cardboard", - "walking", - "wallet", - "warehouse", - "water", - "weight-hanging", - "weight", - "wheelchair", - "wifi", - "wind", - "window-close", - "window-maximize", - "window-minimize", - "window-restore", - "wine-bottle", - "wine-glass-alt", - "wine-glass", - "won-sign", - "wrench", - "x-ray", - "yen-sign", - "yin-yang" - ] -} diff --git a/app/src/main/res/values-night/colors.xml b/app/src/main/res/values-night/colors.xml new file mode 100644 index 00000000..d998e256 --- /dev/null +++ b/app/src/main/res/values-night/colors.xml @@ -0,0 +1,48 @@ + + + #40B887 + #2F9176 + #38D9A9 + + #101010 + #000000 + #000000 + #CCCCCC + #000000 + #AAAAAA + #0E0E0E + #CC000000 + + #999999 + #333333 + #5F5F5F + #333333 + + #40B887 + #2F9176 + #38D9A9 + #E3F6F1 + #77F255DA + + #454545 + #0A0A0A + + #CCCCCC + #AAAAAA + #E5E5E5 + #F1F1F1 + #FFBB00 + #FF0000 + #E35454 + #FFFFFF + #3971DB + #2196f3 + + #F6A637 + #FF4A7D + #26BCF7 + + #D5D5D5 + + #CAEDB9 + \ No newline at end of file diff --git a/app/src/main/res/values-v21/styles.xml b/app/src/main/res/values-v21/styles.xml new file mode 100644 index 00000000..cd72b05c --- /dev/null +++ b/app/src/main/res/values-v21/styles.xml @@ -0,0 +1,20 @@ + + + + + + + diff --git a/app/src/main/res/values/colors.xml b/app/src/main/res/values/colors.xml index 1c703f9e..46392492 100644 --- a/app/src/main/res/values/colors.xml +++ b/app/src/main/res/values/colors.xml @@ -1,12 +1,48 @@ - #40B89A - #303F9F - #FFFFFF + #40B887 + #2F9176 + #38D9A9 - #FF0000 - #00C000 - #FFFFFF + #EFEFEF + #FFFFFF + #FFFFFF + #333333 + #FFFFFF + #444444 + #F1F1F1 + #CC000000 + + #333333 + #333333 + #777777 + #F1F1F1 + + #40B887 #2F9176 #38D9A9 + #E3F6F1 + #77F255DA + + #DFDFDF + #F1F1F1 + + #CCCCCC + #AAAAAA + #E5E5E5 + #F1F1F1 + #FFBB00 + #FF0000 + #E35454 + #FFFFFF + #3971DB + #2196f3 + + #F6A637 + #FF4A7D + #26BCF7 + + #D5D5D5 + + #CAEDB9 \ No newline at end of file diff --git a/app/src/main/res/values/dimens.xml b/app/src/main/res/values/dimens.xml new file mode 100644 index 00000000..4ab4520f --- /dev/null +++ b/app/src/main/res/values/dimens.xml @@ -0,0 +1,8 @@ + + + 16dp + 16dp + 8dp + 176dp + 16dp + \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 9ceff696..a413f849 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -1,14 +1,232 @@ - LBRY - io.lbry.browser.LBRY_ENGAGEMENT_CHANNEL + Open navigation drawer + Close navigation drawer + Navigation header + Settings - Running - Service Status - Stopped - START - STOP + Home + Gallery + Slideshow - Unit tests - RUN + Home Second + + + + Find Content + Your Content + Wallet + + Following + Editor\'s Choice + Your Tags + All Content + Channels + Library + Publishes + New Publish + Rewards + Invites + Settings + About + + Sign In + App startup failed. Please check your data connection and try again. If this problem persists, please email hello@lbry.com + + + Welcome to LBRY. + LBRY is a community-controlled content platform where you can find and publish videos, music, books, and more. + By continuing, I agree to the <a href="https://lbry.com/termsofservice">Terms of Service</a> and confirm I am over the age of 13. + Please wait while we get some things ready... + Use LBRY » + + + Search videos, music and more + + + Find Channels to follow + Channels you follow + Discover + LBRY works better if you follow at least 5 creators you like. Sign in to show creators you follow if you already have an account. + Please select up to 5 creators to continue. + %1$d remaining... + Done + ALL + + + Anonymous + + + Tags + + Share + Repost + Tip + Edit + Delete + Download + Report + Loading decentralized data... + Related Content + + Unsupported Content + Sorry, we are unable to display this content in the app. You can find the file named %1$s in your downloads folder. + + + There\'s nothing here yet.\nPlease check back later. + Content + Website + + + User interface + Other + Enable dark theme + Show URL suggestsions + Notifications + Subscriptions + Content Interests + + Keep the LBRY service running in the background for improved wallet and network performance + Participate in the data network + + + %1$s - Search + %1$s - Tag + Search for \'%1$s\' + Explore the \'%1$s\' tag + View content at %1$s + View the %1$s channel + + + No results found for "%1$s". Please enter a different search term. + You can search for anything including movies, music, ebooks, software and more. + + + Balance + You currently have + You can convert your credits to USD and withdraw the converted amount using an exchange. <a href="https://lbry.com/faq/exchanges">Learn more</a>. + <a href="https://bittrex.com/Account/Register?referralCode=4M1-P30-BON">Convert credits to USD on Bittrex</a> + You also have + You staked + in tips + in your publishes + in your supports + Earn more tips by uploading cool videos + The background service is still initializing. You can explore and watch content in the mean time. + + A backup of your wallet is synced with lbry.tv + Your wallet is not currently synced with lbry.tv. You are responsible for backing up your wallet. + <a href="https://lbry.com/faq/account-sync">What does this mean?</a> + <a href="https://lbry.com/faq/how-to-backup-wallet#android">What does this mean?</a> + + Receive Credits + Use this wallet address to receive credits sent by another user (or yourself). + Get new address + You can generate a new address at any time, and any previous addresses will conintue to work. Using multiple addresses can be helpful for keeping track of incoming payments from multiple sources. + + Send Credits + Recipient address + bbFxRyXXXXXXXXXXXZD8nE7XTLUxYnddTs + Amount + Send + + Recent Transactions + View All + Receive + Spend + Publish + Support + Abandon + Channel + + Wallet Sync + Sync status + On + Off + Email + No connected email + <a href="https://lbry.com/faq/how-to-backup-wallet#android">Manual backup</a> + <a href="https://lbry.com/faq/how-to-backup-wallet#sync">Sync FAQ</a> + 0 + LBC + + Account Recommended + A lbry.tv account allows you to earn rewards, backup your wallet, and keep everything in sync. + Without an account, you assume all responsibility for securing your wallet and LBRY data. + Skip Account + Sign up + Address copied + + Please enter a valid address to send credits to + Insufficient balance + Please enter a valid amount + Your credits could not be sent at this time. Please try again later. + Loading transactions... + There are no recent transactions to display. + fee %1$s + + + You sent %1$s credit + You sent %1$s credits + + + + Sort content by + Content from + Trending content + New content + Top content + Trending + Top + New + Past 24 hours + Past week + Past month + Past year + All time + from + for + Everyone + Customize + The selected view is not yet available. + + + Please provide an email address. + you@example.com + An email has been sent to + Please click the link in the message to complete signing in. + Resend + Continue + Please enter a valid email address + Please follow the instructions in the email sent to your address to continue. + You have successfully signed in to lbry.tv + Retrieving account information... + Applying wallet data... + Please enter the password you used to secure your wallet. + Please enter a password to secure your wallet. + Note: for wallet security purposes, LBRY is unable to reset your password. + Password + Enable sync + The wallet sync operation could not be completed at this time. Please try again later. If this problem persists, please send an email to hello@lbry.com. + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/values/styles.xml b/app/src/main/res/values/styles.xml new file mode 100644 index 00000000..7c5e55a2 --- /dev/null +++ b/app/src/main/res/values/styles.xml @@ -0,0 +1,35 @@ + + + + + + + + + + + + diff --git a/app/src/main/res/values/themes.xml b/app/src/main/res/values/themes.xml deleted file mode 100644 index 12896d72..00000000 --- a/app/src/main/res/values/themes.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - - - \ No newline at end of file diff --git a/app/src/main/res/xml/filepaths.xml b/app/src/main/res/xml/filepaths.xml deleted file mode 100644 index 6c65d1c7..00000000 --- a/app/src/main/res/xml/filepaths.xml +++ /dev/null @@ -1,3 +0,0 @@ - - - \ No newline at end of file diff --git a/app/src/main/res/xml/settings.xml b/app/src/main/res/xml/settings.xml new file mode 100644 index 00000000..7ebffcff --- /dev/null +++ b/app/src/main/res/xml/settings.xml @@ -0,0 +1,52 @@ + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/test/java/io/lbry/browser/ExampleUnitTest.java b/app/src/test/java/io/lbry/browser/ExampleUnitTest.java new file mode 100644 index 00000000..8374b9c4 --- /dev/null +++ b/app/src/test/java/io/lbry/browser/ExampleUnitTest.java @@ -0,0 +1,17 @@ +package io.lbry.browser; + +import org.junit.Test; + +import static org.junit.Assert.*; + +/** + * Example local unit test, which will execute on the development machine (host). + * + * @see Testing documentation + */ +public class ExampleUnitTest { + @Test + public void addition_isCorrect() { + assertEquals(4, 2 + 2); + } +} \ No newline at end of file diff --git a/build.gradle b/build.gradle index a38cb3cc..2e8e3867 100644 --- a/build.gradle +++ b/build.gradle @@ -1,19 +1,16 @@ // Top-level build file where you can add configuration options common to all sub-projects/modules. buildscript { - ext { - buildToolsVersion = "28.0.3" - minSdkVersion = 21 - compileSdkVersion = 28 - targetSdkVersion = 28 - } + repositories { google() jcenter() + } dependencies { - classpath 'com.android.tools.build:gradle:3.6.0' + classpath 'com.android.tools.build:gradle:3.6.2' classpath 'com.google.gms:google-services:4.2.0' + // NOTE: Do not place your application dependencies here; they belong // in the individual module build.gradle files } @@ -23,18 +20,10 @@ allprojects { repositories { google() jcenter() - maven { url 'https://jitpack.io' } - mavenLocal() - maven { - // All of React Native (JS, Obj-C sources, Android binaries) is installed from npm - url("$rootDir/../node_modules/react-native/android") - } - maven { - // Android JSC is installed from npm - url("$rootDir/../node_modules/jsc-android/dist") - } - flatDir { - dirs 'libs' - } + } } + +task clean(type: Delete) { + delete rootProject.buildDir +} diff --git a/gradle.properties b/gradle.properties index 03440afb..199d16ed 100644 --- a/gradle.properties +++ b/gradle.properties @@ -1,23 +1,20 @@ # Project-wide Gradle settings. - # IDE (e.g. Android Studio) users: # Gradle settings configured through the IDE *will override* # any settings specified in this file. - # For more details on how to configure your build environment visit # http://www.gradle.org/docs/current/userguide/build_environment.html - # Specifies the JVM arguments used for the daemon process. # The setting is particularly useful for tweaking memory settings. -# Default value: -Xmx10248m -XX:MaxPermSize=256m -org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8 - +org.gradle.jvmargs=-Xmx1536m # When configured, Gradle will run in incubating parallel mode. # This option should only be used with decoupled projects. More details, visit # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects -org.gradle.parallel=true - -org.gradle.configureondemand=true - +# org.gradle.parallel=true +# AndroidX package structure to make it clearer which packages are bundled with the +# Android operating system, and which are packaged with your app's APK +# https://developer.android.com/topic/libraries/support-library/androidx-rn android.useAndroidX=true +# Automatically convert third-party libraries to use AndroidX android.enableJetifier=true + diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar index 5c2d1cf0..f6b961fd 100644 Binary files a/gradle/wrapper/gradle-wrapper.jar and b/gradle/wrapper/gradle-wrapper.jar differ diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties index 9a201d9e..d3bb33d6 100644 --- a/gradle/wrapper/gradle-wrapper.properties +++ b/gradle/wrapper/gradle-wrapper.properties @@ -1,4 +1,4 @@ -#Wed Feb 26 16:43:39 WAT 2020 +#Fri Apr 03 15:21:09 WAT 2020 distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists zipStoreBase=GRADLE_USER_HOME diff --git a/gradlew b/gradlew index b0d6d0ab..cccdd3d5 100644 --- a/gradlew +++ b/gradlew @@ -1,21 +1,5 @@ #!/usr/bin/env sh -# -# Copyright 2015 the original author or authors. -# -# Licensed under the Apache License, Version 2.0 (the "License"); -# you may not use this file except in compliance with the License. -# You may obtain a copy of the License at -# -# http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an "AS IS" BASIS, -# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -# See the License for the specific language governing permissions and -# limitations under the License. -# - ############################################################################## ## ## Gradle start up script for UN*X @@ -44,7 +28,7 @@ APP_NAME="Gradle" APP_BASE_NAME=`basename "$0"` # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. -DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' +DEFAULT_JVM_OPTS="" # Use the maximum available, or set MAX_FD != -1 to use that value. MAX_FD="maximum" diff --git a/gradlew.bat b/gradlew.bat index 15e1ee37..e95643d6 100644 --- a/gradlew.bat +++ b/gradlew.bat @@ -1,19 +1,3 @@ -@rem -@rem Copyright 2015 the original author or authors. -@rem -@rem Licensed under the Apache License, Version 2.0 (the "License"); -@rem you may not use this file except in compliance with the License. -@rem You may obtain a copy of the License at -@rem -@rem http://www.apache.org/licenses/LICENSE-2.0 -@rem -@rem Unless required by applicable law or agreed to in writing, software -@rem distributed under the License is distributed on an "AS IS" BASIS, -@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -@rem See the License for the specific language governing permissions and -@rem limitations under the License. -@rem - @if "%DEBUG%" == "" @echo off @rem ########################################################################## @rem @@ -30,7 +14,7 @@ set APP_BASE_NAME=%~n0 set APP_HOME=%DIRNAME% @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. -set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" +set DEFAULT_JVM_OPTS= @rem Find java.exe if defined JAVA_HOME goto findJavaFromJavaHome diff --git a/settings.gradle b/settings.gradle index 79f9cb53..7d91e8cc 100644 --- a/settings.gradle +++ b/settings.gradle @@ -1,24 +1,2 @@ -rootProject.name = 'LbryAndroid' -include ':@react-native-community_async-storage' -project(':@react-native-community_async-storage').projectDir = new File(settingsDir, '../node_modules/@react-native-community/async-storage/android') -include ':react-native-camera' -project(':react-native-camera').projectDir = new File(settingsDir, '../node_modules/react-native-camera/android') -include ':react-native-exception-handler' -project(':react-native-exception-handler').projectDir = new File(settingsDir, '../node_modules/react-native-exception-handler/android') -include ':react-native-fast-image' -project(':react-native-fast-image').projectDir = new File(settingsDir, '../node_modules/react-native-fast-image/android') -include ':react-native-fs' -project(':react-native-fs').projectDir = new File(settingsDir, '../node_modules/react-native-fs/android') -include ':react-native-gesture-handler' -project(':react-native-gesture-handler').projectDir = new File(settingsDir, '../node_modules/react-native-gesture-handler/android') -include ':react-native-reanimated' -project(':react-native-reanimated').projectDir = new File(settingsDir, '../node_modules/react-native-reanimated/android') -include ':react-native-snackbar' -project(':react-native-snackbar').projectDir = new File(settingsDir, '../node_modules/react-native-snackbar/android') -include ':react-native-video' -project(':react-native-video').projectDir = new File(settingsDir, '../node_modules/react-native-video/android-exoplayer') -include ':react-native-webview' -project(':react-native-webview').projectDir = new File(settingsDir, '../node_modules/react-native-webview/android') -include ':rn-fetch-blob' -project(':rn-fetch-blob').projectDir = new File(settingsDir, '../node_modules/rn-fetch-blob/android') -include ':app' \ No newline at end of file +rootProject.name='LBRY' +include ':app'