diff --git a/.gitignore b/.gitignore index 02f7d626..c21e214d 100644 --- a/.gitignore +++ b/.gitignore @@ -60,7 +60,6 @@ buck-out/ # Other Files app/google-services.json -app/src/main/assets/index.android.bundle *.log .vagrant *.hprof diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index 285c216d..33fca8b4 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -3,18 +3,12 @@ stages: - deploy - release -variables: - REACT_NATIVE_BRANCH: "master" build apk: stage: build image: lbry/android-base:platform-28 before_script: - apt-get -y update && apt-get -y install build-essential ca-certificates curl git gpg-agent openjdk-8-jdk software-properties-common wget zipalign - - wget -q https://nodejs.org/dist/latest-v10.x/node-v10.19.0-linux-x64.tar.gz && tar xf node-v10.19.0-linux-x64.tar.gz -C /opt - - ln -fs /opt/node-v10.19.0-linux-x64/bin/node /usr/bin/node - - ln -fs /opt/node-v10.19.0-linux-x64/bin/npm /usr/bin/npm - - ln -fs /opt/node-v10.19.0-linux-x64/bin/npx /usr/bin/npx - chmod u+x $CI_PROJECT_DIR/gradlew - export ANDROID_SDK_ROOT=~/.buildozer/android/platform/android-sdk-23 - export BUILD_VERSION=$($CI_PROJECT_DIR/gradlew -q printVersionName --console=plain | tail -1) @@ -26,25 +20,13 @@ build apk: script: - export PATH=/usr/bin:$PATH - echo "$PGP_PRIVATE_KEY" | gpg --batch --import - - curl -sS https://dl.yarnpkg.com/debian/pubkey.gpg | apt-key add - - - echo "deb https://dl.yarnpkg.com/debian/ stable main" | tee /etc/apt/sources.list.d/yarn.list - echo "deb https://dl.bintray.com/sobolevn/deb git-secret main" | tee -a /etc/apt/sources.list - wget -O - https://api.bintray.com/users/sobolevn/keys/gpg/public.key | apt-key add - - - apt-get -y update && apt-get -y install yarn git-secret + - apt-get -y update && apt-get -y install git-secret - git secret reveal - - npm install -g react-native-cli - - cd ~/ - - git clone --single-branch --branch $REACT_NATIVE_BRANCH https://github.com/lbryio/lbry-react-native - - cd lbry-react-native - - chmod u+x bundle-android.sh - yarn - - rm -rf android # temporary, should be a submodule init? - - cp -rf $CI_PROJECT_DIR/ android/ - - cd android - chmod u+x ./release.sh - ./release.sh - - mkdir -p $CI_PROJECT_DIR/bin/ && cp bin/*.apk $CI_PROJECT_DIR/bin/ - - cd $CI_PROJECT_DIR - cp bin/browser-$BUILD_VERSION-release__arm.apk /dev/null - cp bin/browser-$BUILD_VERSION-release__arm64.apk /dev/null 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 aa3e40f4..81794584 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 1403 - versionName "0.14.3" - missingDimensionStrategy 'react-native-camera', 'general' - multiDexEnabled true - } - dexOptions { - javaMaxHeapSize "2048M" - preDexLibraries false - jumboMode true + minSdkVersion 21 + targetSdkVersion 29 + versionCode 1503 + versionName "0.15.3" + + testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" } + productFlavors { __32bit { versionCode android.defaultConfig.versionCode * 10 + 1 @@ -167,66 +34,70 @@ android { } } } - signingConfigs { - debug { - storeFile file('debug.keystore') - storePassword 'android' - keyAlias 'androiddebugkey' - keyPassword 'android' + + buildTypes { + release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' } } - buildTypes { - debug { - signingConfig signingConfigs.debug - } - release { - } +} + +task printVersionName { + doLast { + println android.defaultConfig.versionName } } 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 'androidx.camera:camera-camera2:1.0.0-beta03' + implementation 'androidx.camera:camera-lifecycle:1.0.0-beta03' + implementation 'androidx.camera:camera-view:1.0.0-alpha10' - __32bitImplementation files('libs/lbrysdk-0.73.1-release__arm.aar') - __64bitImplementation files('libs/lbrysdk-0.73.1-release__arm64.aar') + 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' - if (enableHermes) { - def hermesPath = "../../node_modules/hermes-engine/android/"; - debugImplementation files(hermesPath + "hermes-debug.aar") - releaseImplementation files(hermesPath + "hermes-release.aar") - } else { - implementation jscFlavor - } -} + 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.exoplayer:extension-mediasession:2.11.4' -// 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' + implementation 'com.google.android:flexbox:2.0.1' + + implementation 'com.hbb20:ccp:2.3.8' + + implementation 'com.github.chrisbanes:PhotoView:2.3.0' + implementation 'com.atlassian.commonmark:commonmark:0.14.0' + + implementation 'com.arthenica:mobile-ffmpeg-full-gpl:4.3.1.LTS' + + 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.74.0-release__arm.aar') + __64bitImplementation files('libs/lbrysdk-0.74.0-release__arm64.aar') } 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/libs/lbrysdk-0.73.1-release__arm.aar b/app/libs/lbrysdk-0.74.0-release__arm.aar similarity index 70% rename from app/libs/lbrysdk-0.73.1-release__arm.aar rename to app/libs/lbrysdk-0.74.0-release__arm.aar index 6058d44a..f917e371 100644 Binary files a/app/libs/lbrysdk-0.73.1-release__arm.aar and b/app/libs/lbrysdk-0.74.0-release__arm.aar differ diff --git a/app/libs/lbrysdk-0.73.1-release__arm64.aar b/app/libs/lbrysdk-0.74.0-release__arm64.aar similarity index 71% rename from app/libs/lbrysdk-0.73.1-release__arm64.aar rename to app/libs/lbrysdk-0.74.0-release__arm64.aar index 525e693d..fe87414a 100644 Binary files a/app/libs/lbrysdk-0.73.1-release__arm64.aar and b/app/libs/lbrysdk-0.74.0-release__arm64.aar 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..28eca8a7 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -1,30 +1,30 @@ + - + - - + + @@ -37,16 +37,18 @@ - - - + android:supportsPictureInPicture="true" + android:theme="@style/AppTheme.NoActionBar" + android:launchMode="singleTask" + android:windowSoftInputMode="adjustResize"> + + + + @@ -54,14 +56,19 @@ - - - - - - + + - - + \ 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/FirstRunActivity.java b/app/src/main/java/io/lbry/browser/FirstRunActivity.java new file mode 100644 index 00000000..8197fd70 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/FirstRunActivity.java @@ -0,0 +1,138 @@ +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.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.LbryAnalytics; +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(); + } + } + + public void onResume() { + super.onResume(); + LbryAnalytics.setCurrentScreen(this, "First Run", "FirstRun"); + } + + 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 + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + boolean firstAuthCompleted = sp.getBoolean(MainActivity.PREFERENCE_KEY_INTERNAL_FIRST_AUTH_COMPLETED, false); + if (!firstAuthCompleted) { + LbryAnalytics.logEvent(LbryAnalytics.EVENT_FIRST_USER_AUTH); + sp.edit().putBoolean(MainActivity.PREFERENCE_KEY_INTERNAL_FIRST_AUTH_COMPLETED, true).apply(); + } + + 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_INTERNAL_FIRST_RUN_COMPLETED, true).apply(); + + // first_run_completed event + LbryAnalytics.logEvent(LbryAnalytics.EVENT_FIRST_RUN_COMPLETED); + 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 index f5c4c4a8..62536467 100644 --- a/app/src/main/java/io/lbry/browser/LbrynetMessagingService.java +++ b/app/src/main/java/io/lbry/browser/LbrynetMessagingService.java @@ -12,14 +12,16 @@ import android.os.Build; import android.os.Bundle; import androidx.core.app.NotificationCompat; import androidx.core.content.ContextCompat; +import androidx.preference.PreferenceManager; + 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.browser.utils.LbryAnalytics; import io.lbry.lbrysdk.LbrynetService; -import io.lbry.browser.reactmodules.UtilityModule; import java.io.UnsupportedEncodingException; import java.net.URLEncoder; @@ -30,22 +32,15 @@ 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); } @@ -67,7 +62,7 @@ public class LbrynetMessagingService extends FirebaseMessagingService { if (firebaseAnalytics != null) { Bundle bundle = new Bundle(); bundle.putString("name", name); - firebaseAnalytics.logEvent("lbry_notification_receive", bundle); + firebaseAnalytics.logEvent(LbryAnalytics.EVENT_LBRY_NOTIFICATION_RECEIVE, bundle); } sendNotification(title, body, type, url, name, contentTitle, channelUrl, publishTime); @@ -97,7 +92,7 @@ public class LbrynetMessagingService extends FirebaseMessagingService { // 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. @@ -113,29 +108,6 @@ public class LbrynetMessagingService extends FirebaseMessagingService { // 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)); @@ -160,27 +132,27 @@ public class LbrynetMessagingService extends FirebaseMessagingService { // 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); + NOTIFICATION_CHANNEL_ID, "LBRY Engagement", NotificationManager.IMPORTANCE_DEFAULT); notificationManager.createNotificationChannel(channel); } - notificationManager.notify(9898, notificationBuilder.build()); + notificationManager.notify(3, notificationBuilder.build()); } public List getEnabledTypes() { - SharedPreferences sp = getSharedPreferences(MainActivity.SHARED_PREFERENCES_NAME, Context.MODE_PRIVATE); + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); List enabledTypes = new ArrayList(); - if (sp.getBoolean(UtilityModule.RECEIVE_SUBSCRIPTION_NOTIFICATIONS, true)) { + if (sp.getBoolean(MainActivity.PREFERENCE_KEY_NOTIFICATION_SUBSCRIPTIONS, true)) { enabledTypes.add(TYPE_SUBSCRIPTION); } - if (sp.getBoolean(UtilityModule.RECEIVE_REWARD_NOTIFICATIONS, true)) { + if (sp.getBoolean(MainActivity.PREFERENCE_KEY_NOTIFICATION_REWARDS, true)) { enabledTypes.add(TYPE_REWARD); } - if (sp.getBoolean(UtilityModule.RECEIVE_INTERESTS_NOTIFICATIONS, true)) { + if (sp.getBoolean(MainActivity.PREFERENCE_KEY_NOTIFICATION_CONTENT_INTERESTS, true)) { enabledTypes.add(TYPE_INTERESTS); } - if (sp.getBoolean(UtilityModule.RECEIVE_CREATOR_NOTIFICATIONS, true)) { + if (sp.getBoolean(MainActivity.PREFERENCE_KEY_NOTIFICATION_CREATOR, true)) { enabledTypes.add(TYPE_CREATOR); } diff --git a/app/src/main/java/io/lbry/browser/MainActivity.java b/app/src/main/java/io/lbry/browser/MainActivity.java index f5b0d6f0..ed711318 100644 --- a/app/src/main/java/io/lbry/browser/MainActivity.java +++ b/app/src/main/java/io/lbry/browser/MainActivity.java @@ -1,193 +1,2395 @@ package io.lbry.browser; -import android.annotation.SuppressLint; -import android.annotation.TargetApi; +import android.app.Activity; +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.ComponentName; 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.content.pm.PackageManager; +import android.content.res.Configuration; +import android.content.res.TypedArray; +import android.database.sqlite.SQLiteDatabase; +import android.graphics.Color; +import android.graphics.drawable.Drawable; import android.net.Uri; +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 android.support.v4.media.session.MediaSessionCompat; +import android.text.Editable; +import android.text.TextWatcher; +import android.util.Base64; +import android.util.Log; +import android.view.KeyEvent; +import android.view.View; +import android.view.Menu; +import android.view.WindowManager; +import android.view.inputmethod.EditorInfo; +import android.view.inputmethod.InputMethodManager; +import android.widget.EditText; +import android.widget.ImageView; +import android.widget.TextView; +import android.widget.Toast; + +import com.google.android.exoplayer2.SimpleExoPlayer; +import com.google.android.exoplayer2.ext.cast.CastPlayer; +import com.google.android.exoplayer2.ext.mediasession.MediaSessionConnector; +import com.google.android.exoplayer2.ui.PlayerView; +import com.google.android.exoplayer2.upstream.cache.Cache; +import com.google.android.gms.cast.framework.CastContext; +import com.google.android.gms.tasks.OnCompleteListener; +import com.google.android.gms.tasks.Task; +import com.google.android.material.snackbar.Snackbar; +import com.google.firebase.iid.FirebaseInstanceId; +import com.google.firebase.iid.InstanceIdResult; + +import androidx.annotation.Nullable; +import androidx.appcompat.app.ActionBar; +import androidx.appcompat.app.ActionBarDrawerToggle; +import androidx.appcompat.app.AppCompatDelegate; import androidx.core.app.ActivityCompat; 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.FileProvider; +import androidx.core.content.res.ResourcesCompat; +import androidx.core.graphics.drawable.DrawableCompat; +import androidx.core.view.GravityCompat; +import androidx.core.view.OnApplyWindowInsetsListener; +import androidx.core.view.ViewCompat; +import androidx.core.view.WindowInsetsCompat; +import androidx.fragment.app.Fragment; +import androidx.fragment.app.FragmentManager; +import androidx.fragment.app.FragmentTransaction; +import androidx.media.session.MediaButtonReceiver; +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.File; +import java.io.FileInputStream; +import java.io.InputStreamReader; +import java.net.ConnectException; +import java.text.DecimalFormat; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collections; +import java.util.Date; +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 java.util.logging.Level; +import java.util.logging.Logger; + +import io.lbry.browser.adapter.NavigationMenuAdapter; +import io.lbry.browser.adapter.UrlSuggestionListAdapter; +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.dialog.ContentScopeDialogFragment; +import io.lbry.browser.exceptions.LbryUriException; +import io.lbry.browser.listener.CameraPermissionListener; +import io.lbry.browser.listener.DownloadActionListener; +import io.lbry.browser.listener.FetchChannelsListener; +import io.lbry.browser.listener.FetchClaimsListener; +import io.lbry.browser.listener.FilePickerListener; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.listener.StoragePermissionListener; +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.Tag; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.model.WalletSync; +import io.lbry.browser.model.lbryinc.Reward; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ClaimListTask; +import io.lbry.browser.tasks.lbryinc.ClaimRewardTask; +import io.lbry.browser.tasks.lbryinc.FetchRewardsTask; +import io.lbry.browser.tasks.LighthouseAutoCompleteTask; +import io.lbry.browser.tasks.MergeSubscriptionsTask; +import io.lbry.browser.tasks.claim.ResolveTask; +import io.lbry.browser.tasks.localdata.FetchRecentUrlHistoryTask; +import io.lbry.browser.tasks.wallet.DefaultSyncTaskHandler; +import io.lbry.browser.tasks.wallet.LoadSharedUserStateTask; +import io.lbry.browser.tasks.wallet.SaveSharedUserStateTask; +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.tasks.wallet.WalletBalanceTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.ui.channel.ChannelFormFragment; +import io.lbry.browser.ui.channel.ChannelFragment; +import io.lbry.browser.ui.channel.ChannelManagerFragment; +import io.lbry.browser.ui.findcontent.EditorsChoiceFragment; +import io.lbry.browser.ui.findcontent.FileViewFragment; +import io.lbry.browser.ui.findcontent.FollowingFragment; +import io.lbry.browser.ui.library.LibraryFragment; +import io.lbry.browser.ui.other.AboutFragment; +import io.lbry.browser.ui.publish.PublishFormFragment; +import io.lbry.browser.ui.publish.PublishFragment; +import io.lbry.browser.ui.publish.PublishesFragment; +import io.lbry.browser.ui.findcontent.SearchFragment; +import io.lbry.browser.ui.other.SettingsFragment; +import io.lbry.browser.ui.findcontent.AllContentFragment; +import io.lbry.browser.ui.wallet.InvitesFragment; +import io.lbry.browser.ui.wallet.RewardsFragment; +import io.lbry.browser.ui.wallet.WalletFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; +import io.lbry.lbrysdk.DownloadManager; import io.lbry.lbrysdk.LbrynetService; import io.lbry.lbrysdk.ServiceHelper; import io.lbry.lbrysdk.Utils; +import lombok.Getter; +import lombok.Setter; +import okhttp3.OkHttpClient; -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; + private Map specialRouteFragmentClassMap; + private boolean inPictureInPictureMode; + public static SimpleExoPlayer appPlayer; + public static Cache playerCache; + public static boolean playerReassigned; + public static CastContext castContext; + public static CastPlayer castPlayer; + public static Claim nowPlayingClaim; + public static String nowPlayingClaimUrl; + public static boolean startingFilePickerActivity = false; + public static boolean startingShareActivity = false; + public static boolean startingPermissionRequest = false; + public static boolean startingSignInFlowActivity = false; + public static boolean startingCameraRequest = false; + private boolean enteringPIPMode = false; + private boolean fullSyncInProgress = false; + private int queuedSyncCount = 0; + private String cameraOutputFilename; -public class MainActivity extends FragmentActivity implements DefaultHardwareBackBtnHandler, PermissionAwareActivity { + @Setter + private BackPressInterceptor backPressInterceptor; - 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(); + @Getter + private String firebaseMessagingToken; - private BroadcastReceiver notificationsReceiver; - private BroadcastReceiver smsReceiver; + private Map openNavFragments; + private static final Map fragmentClassNavIdMap = new HashMap<>(); + static { + Logger.getLogger(OkHttpClient.class.getName()).setLevel(Level.FINE); + + fragmentClassNavIdMap.put(FollowingFragment.class, NavMenuItem.ID_ITEM_FOLLOWING); + fragmentClassNavIdMap.put(EditorsChoiceFragment.class, NavMenuItem.ID_ITEM_EDITORS_CHOICE); + fragmentClassNavIdMap.put(AllContentFragment.class, NavMenuItem.ID_ITEM_ALL_CONTENT); + + fragmentClassNavIdMap.put(PublishFragment.class, NavMenuItem.ID_ITEM_NEW_PUBLISH); + fragmentClassNavIdMap.put(ChannelManagerFragment.class, NavMenuItem.ID_ITEM_CHANNELS); + fragmentClassNavIdMap.put(LibraryFragment.class, NavMenuItem.ID_ITEM_LIBRARY); + fragmentClassNavIdMap.put(PublishesFragment.class, NavMenuItem.ID_ITEM_PUBLISHES); + + fragmentClassNavIdMap.put(WalletFragment.class, NavMenuItem.ID_ITEM_WALLET); + fragmentClassNavIdMap.put(RewardsFragment.class, NavMenuItem.ID_ITEM_REWARDS); + fragmentClassNavIdMap.put(InvitesFragment.class, NavMenuItem.ID_ITEM_INVITES); + + fragmentClassNavIdMap.put(SettingsFragment.class, NavMenuItem.ID_ITEM_SETTINGS); + fragmentClassNavIdMap.put(AboutFragment.class, NavMenuItem.ID_ITEM_ABOUT); + + // Internal (sub-)pages + fragmentClassNavIdMap.put(FileViewFragment.class, NavMenuItem.ID_ITEM_FOLLOWING); + fragmentClassNavIdMap.put(ChannelFragment.class, NavMenuItem.ID_ITEM_FOLLOWING); + fragmentClassNavIdMap.put(SearchFragment.class, NavMenuItem.ID_ITEM_FOLLOWING); + } + + public static final int REQUEST_STORAGE_PERMISSION = 1001; + public static final int REQUEST_CAMERA_PERMISSION = 1002; + public static final int REQUEST_SIMPLE_SIGN_IN = 2001; + public static final int REQUEST_WALLET_SYNC_SIGN_IN = 2002; + public static final int REQUEST_REWARDS_VERIFY_SIGN_IN = 2003; + + public static final int REQUEST_FILE_PICKER = 5001; + public static final int REQUEST_VIDEO_CAPTURE = 5002; + public static final int REQUEST_TAKE_PHOTO = 5003; + + // 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"; + public static final String ACTION_WALLET_BALANCE_UPDATED = "io.lbry.browser.Broadcast.WalletBalanceUpdated"; + public static final String ACTION_OPEN_CHANNEL_URL = "io.lbry.browser.Broadcast.OpenChannelUrl"; + public static final String ACTION_OPEN_WALLET_PAGE = "io.lbry.browser.Broadcast.OpenWalletPage"; + public static final String ACTION_OPEN_REWARDS_PAGE = "io.lbry.browser.Broadcast.OpenRewardsPage"; + public static final String ACTION_SAVE_SHARED_USER_STATE = "io.lbry.browser.Broadcast.SaveSharedUserState"; + + // preference keys + public static final String PREFERENCE_KEY_DARK_MODE = "io.lbry.browser.preference.userinterface.DarkMode"; + public static final String PREFERENCE_KEY_SHOW_MATURE_CONTENT = "io.lbry.browser.preference.userinterface.ShowMatureContent"; + public static final String PREFERENCE_KEY_SHOW_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_NOTIFICATION_CREATOR = "io.lbry.browser.preference.notifications.Creator"; + 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"; + public static final String PREFERENCE_KEY_INTERNAL_REWARDS_NOT_INTERESTED = "io.lbry.browser.preference.internal.RewardsNotInterested"; + public static final String PREFERENCE_KEY_INTERNAL_NEW_ANDROID_REWARD_CLAIMED = "io.lbry.browser.preference.internal.NewAndroidRewardClaimed"; + + public static final String PREFERENCE_KEY_INTERNAL_FIRST_RUN_COMPLETED = "io.lbry.browser.preference.internal.FirstRunCompleted"; + public static final String PREFERENCE_KEY_INTERNAL_FIRST_AUTH_COMPLETED = "io.lbry.browser.preference.internal.FirstAuthCompleted"; + + private final int CHECK_SDK_READY_INTERVAL = 1000; + + 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; + private List recentUrlHistory; + private boolean hasLoadedFirstBalance; + + // broadcast receivers private BroadcastReceiver serviceActionsReceiver; - private BroadcastReceiver downloadEventReceiver; - private FirebaseAnalytics firebaseAnalytics; - private ReactRootView mReactRootView; - private ReactInstanceManager mReactInstanceManager; + private BroadcastReceiver requestsReceiver; - /** - * Flag which indicates whether or not the service is running. Will be updated in the - * onResume method. - */ + private static boolean appStarted; private boolean serviceRunning; private CheckSdkReadyTask checkSdkReadyTask; + private MediaSessionCompat mediaSession; private boolean receivedStopService; - private PermissionListener permissionListener; - public static boolean lbrySdkReady; + private ActionBarDrawerToggle toggle; + private SyncSetTask syncSetTask = null; + private List pendingSyncSetQueue; + @Getter + private DatabaseHelper dbHelper; + private int selectedMenuItemId = -1; + private List cameraPermissionListeners; + private List downloadActionListeners; + private List filePickerListeners; + private List sdkStatusListeners; + private List storagePermissionListeners; + private List walletBalanceListeners; + private List fetchClaimsListeners; + private List fetchChannelsListeners; + @Getter + private ScheduledExecutorService scheduler = Executors.newSingleThreadScheduledExecutor(); + private boolean walletBalanceUpdateScheduled; + private boolean shouldOpenUserSelectedMenuItem; + private boolean walletSyncScheduled; + private String pendingAllContentTag; + private String pendingChannelUrl; + private boolean pendingOpenWalletPage; + private boolean pendingOpenRewardsPage; + private boolean pendingFollowingReload; - protected String getMainComponentName() { - return "LBRYApp"; + // startup stages (to be able to determine how far a user made it if startup fails) + // and display a more useful message for troubleshooting + private static final int STARTUP_STAGE_INSTALL_ID_LOADED = 1; + private static final int STARTUP_STAGE_KNOWN_TAGS_LOADED = 2; + private static final int STARTUP_STAGE_EXCHANGE_RATE_LOADED = 3; + private static final int STARTUP_STAGE_USER_AUTHENTICATED = 4; + private static final int STARTUP_STAGE_NEW_INSTALL_DONE = 5; + private static final int STARTUP_STAGE_SUBSCRIPTIONS_LOADED = 6; + private static final int STARTUP_STAGE_SUBSCRIPTIONS_RESOLVED = 7; + + public boolean isDarkMode() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + return sp.getBoolean(PREFERENCE_KEY_DARK_MODE, false); } - - public static LaunchTiming CurrentLaunchTiming; - + @Override protected void onCreate(Bundle savedInstanceState) { - CurrentLaunchTiming = new LaunchTiming(new Date()); + AppCompatDelegate.setDefaultNightMode(isDarkMode() ? AppCompatDelegate.MODE_NIGHT_YES : AppCompatDelegate.MODE_NIGHT_NO); + + initKeyStore(); + loadAuthToken(); + + if (!isDarkMode()) { + getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR); + } + initSpecialRouteMap(); + + LbryAnalytics.init(this); + FirebaseInstanceId.getInstance().getInstanceId().addOnCompleteListener(new OnCompleteListener() { + @Override + public void onComplete(Task task) { + if (!task.isSuccessful()) { + return; + } + + // Get new Instance ID token + firebaseMessagingToken = task.getResult().getToken(); + } + }); + super.onCreate(savedInstanceState); - currentActivity = this; + dbHelper = new DatabaseHelper(this); + checkNotificationOpenIntent(getIntent()); + setContentView(R.layout.activity_main); + Toolbar toolbar = findViewById(R.id.toolbar); + setSupportActionBar(toolbar); - SoLoader.init(this, false); + // TODO: Check Google Play Services availability + // castContext = CastContext.getSharedInstance(this); - // Register the stop service receiver (so that we close the activity if the user requests the service to stop) + ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.content_main), new OnApplyWindowInsetsListener() { + @Override + public WindowInsetsCompat onApplyWindowInsets(View v, WindowInsetsCompat insets) { + ViewCompat.onApplyWindowInsets(findViewById(R.id.url_suggestions_container), + insets.replaceSystemWindowInsets(0, 0, 0, insets.getSystemWindowInsetBottom())); + return ViewCompat.onApplyWindowInsets(v, + insets.replaceSystemWindowInsets(insets.getSystemWindowInsetLeft(), 0, + 0, insets.getSystemWindowInsetBottom())); + } + }); + + // register receivers + registerRequestsReceiver(); registerServiceActionsReceiver(); - // Register SMS receiver for handling verification texts - registerSmsReceiver(); + View decorView = getWindow().getDecorView(); + decorView.setOnSystemUiVisibilityChangeListener(new View.OnSystemUiVisibilityChangeListener() { + @Override + public void onSystemUiVisibilityChange(int visibility) { + if ((visibility & View.SYSTEM_UI_FLAG_FULLSCREEN) == 0) { + // not fullscreen + } + } + }); - // Register the receiver to emit download events - registerDownloadEventReceiver(); + // setup uri bar + setupUriBar(); - // 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)); + /*View decorView = getWindow().getDecorView(); + decorView.setOnSystemUiVisibilityChangeListener(new View.OnSystemUiVisibilityChangeListener() { + @Override + public void onSystemUiVisibilityChange(int visibility) { + if ((visibility & View.SYSTEM_UI_FLAG_FULLSCREEN) == 0) { + findViewById(R.id.app_bar_main_container).setFitsSystemWindows(false); + findViewById(R.id.drawer_layout).setFitsSystemWindows(false); + } else { + findViewById(R.id.app_bar_main_container).setFitsSystemWindows(true); + findViewById(R.id.drawer_layout).setFitsSystemWindows(true); + } + } + });*/ + + // other + pendingSyncSetQueue = new ArrayList<>(); + openNavFragments = new HashMap<>(); + cameraPermissionListeners = new ArrayList<>(); + downloadActionListeners = new ArrayList<>(); + filePickerListeners = new ArrayList<>(); + sdkStatusListeners = new ArrayList<>(); + storagePermissionListeners = new ArrayList<>(); + walletBalanceListeners = new ArrayList<>(); + fetchClaimsListeners = new ArrayList<>(); + fetchChannelsListeners = new ArrayList<>(); + + sdkStatusListeners.add(this); + + FragmentManager fragmentManager = getSupportFragmentManager(); + fragmentManager.addOnBackStackChangedListener(backStackChangedListener); + + 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; + } + } + + @Override + public void onDrawerSlide(View drawerView, float slideOffset) { + if (slideOffset != 0) { + clearWunderbarFocus(findViewById(R.id.wunderbar)); + } + super.onDrawerSlide(drawerView, slideOffset); + } + }; + 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(); + getWindow().clearFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON); + 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 && !Helper.isNullOrEmpty(nowPlayingClaimUrl)) { + openFileUrl(nowPlayingClaimUrl); + } + } + }); + + // 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() && !Arrays.asList( + NavMenuItem.ID_ITEM_FOLLOWING, NavMenuItem.ID_ITEM_ALL_CONTENT, NavMenuItem.ID_ITEM_WALLET).contains(menuItem.getId())) { + // already open + navMenuAdapter.setCurrentItem(menuItem); + closeDrawer(); + return; + } + + navMenuAdapter.setCurrentItem(menuItem); + shouldOpenUserSelectedMenuItem = true; + selectedMenuItemId = menuItem.getId(); + closeDrawer(); + } + }); + navItemsView.setAdapter(navMenuAdapter); + + findViewById(R.id.sign_in_button_container).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + walletSyncSignIn(); + } + }); + } + + private void initSpecialRouteMap() { + specialRouteFragmentClassMap = new HashMap<>(); + specialRouteFragmentClassMap.put("about", AboutFragment.class); + specialRouteFragmentClassMap.put("allContent", AllContentFragment.class); + specialRouteFragmentClassMap.put("channels", ChannelManagerFragment.class); + specialRouteFragmentClassMap.put("invite", InvitesFragment.class); + specialRouteFragmentClassMap.put("invites", InvitesFragment.class); + specialRouteFragmentClassMap.put("library", LibraryFragment.class); + specialRouteFragmentClassMap.put("publish", PublishFragment.class); + specialRouteFragmentClassMap.put("publishes", PublishesFragment.class); + specialRouteFragmentClassMap.put("following", FollowingFragment.class); + specialRouteFragmentClassMap.put("rewards", RewardsFragment.class); + specialRouteFragmentClassMap.put("settings", SettingsFragment.class); + specialRouteFragmentClassMap.put("subscriptions", FollowingFragment.class); + specialRouteFragmentClassMap.put("wallet", WalletFragment.class); + specialRouteFragmentClassMap.put("discover", FollowingFragment.class); + } + + protected void onNewIntent(Intent intent) { + super.onNewIntent(intent); + checkUrlIntent(intent); + checkNotificationOpenIntent(intent); + } + + public void addCameraPermissionListener(CameraPermissionListener listener) { + if (!cameraPermissionListeners.contains(listener)) { + cameraPermissionListeners.add(listener); + } + } + + public void removeCameraPermissionListener(CameraPermissionListener listener) { + cameraPermissionListeners.remove(listener); + } + + public void addDownloadActionListener(DownloadActionListener listener) { + if (!downloadActionListeners.contains(listener)) { + downloadActionListeners.add(listener); + } + } + + public void removeDownloadActionListener(DownloadActionListener listener) { + downloadActionListeners.remove(listener); + } + + public void addFilePickerListener(FilePickerListener listener) { + if (!filePickerListeners.contains(listener)) { + filePickerListeners.add(listener); + } + } + + public void removeFilePickerListener(FilePickerListener listener) { + filePickerListeners.remove(listener); + } + + public void addSdkStatusListener(SdkStatusListener listener) { + if (!sdkStatusListeners.contains(listener)) { + sdkStatusListeners.add(listener); + } + } + + public void removeSdkStatusListener(SdkStatusListener listener) { + sdkStatusListeners.remove(listener); + } + + public void addStoragePermissionListener(StoragePermissionListener listener) { + if (!storagePermissionListeners.contains(listener)) { + storagePermissionListeners.add(listener); + } + } + + public void removeStoragePermissionListener(StoragePermissionListener listener) { + storagePermissionListeners.remove(listener); + } + + public void addWalletBalanceListener(WalletBalanceListener listener) { + if (!walletBalanceListeners.contains(listener)) { + walletBalanceListeners.add(listener); + } + } + + public void removeWalletBalanceListener(WalletBalanceListener listener) { + walletBalanceListeners.remove(listener); + } + + public void removeNavFragment(Class fragmentClass, int navItemId) { + String key = buildNavFragmentKey(fragmentClass, navItemId, null); + if (openNavFragments.containsKey(key)) { + openNavFragments.remove(key); + } + } + + public void addFetchChannelsListener(FetchChannelsListener listener) { + if (!fetchChannelsListeners.contains(listener)) { + fetchChannelsListeners.add(listener); + } + } + public void removeFetchChannelsListener(FetchChannelsListener listener) { + fetchChannelsListeners.remove(listener); + } + + public void addFetchClaimsListener(FetchClaimsListener listener) { + if (!fetchClaimsListeners.contains(listener)) { + fetchClaimsListeners.add(listener); + } + } + public void removeFetchClaimsListener(FetchClaimsListener listener) { + fetchClaimsListeners.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_EDITORS_CHOICE: + openFragment(EditorsChoiceFragment.class, true, NavMenuItem.ID_ITEM_EDITORS_CHOICE); + break; + case NavMenuItem.ID_ITEM_ALL_CONTENT: + openFragment(AllContentFragment.class, true, NavMenuItem.ID_ITEM_ALL_CONTENT); + break; + + case NavMenuItem.ID_ITEM_NEW_PUBLISH: + openFragment(PublishFragment.class, true, NavMenuItem.ID_ITEM_NEW_PUBLISH); + break; + case NavMenuItem.ID_ITEM_CHANNELS: + openFragment(ChannelManagerFragment.class, true, NavMenuItem.ID_ITEM_CHANNELS); + break; + case NavMenuItem.ID_ITEM_LIBRARY: + openFragment(LibraryFragment.class, true, NavMenuItem.ID_ITEM_LIBRARY); + break; + case NavMenuItem.ID_ITEM_PUBLISHES: + openFragment(PublishesFragment.class, true, NavMenuItem.ID_ITEM_PUBLISHES); + break; + + case NavMenuItem.ID_ITEM_WALLET: + openFragment(WalletFragment.class, true, NavMenuItem.ID_ITEM_WALLET); + break; + case NavMenuItem.ID_ITEM_REWARDS: + openFragment(RewardsFragment.class, true, NavMenuItem.ID_ITEM_REWARDS); + break; + case NavMenuItem.ID_ITEM_INVITES: + openFragment(InvitesFragment.class, true, NavMenuItem.ID_ITEM_INVITES); + break; + + case NavMenuItem.ID_ITEM_SETTINGS: + openFragment(SettingsFragment.class, true, NavMenuItem.ID_ITEM_SETTINGS); + break; + case NavMenuItem.ID_ITEM_ABOUT: + openFragment(AboutFragment.class, true, NavMenuItem.ID_ITEM_ABOUT); + break; + } + } + + public void openChannelClaim(Claim claim) { + Map params = new HashMap<>(); + params.put("url", !Helper.isNullOrEmpty(claim.getShortUrl()) ? claim.getShortUrl() : claim.getPermanentUrl()); + params.put("claim", getCachedClaimForUrl(claim.getPermanentUrl())); + openFragment(ChannelFragment.class, true, NavMenuItem.ID_ITEM_FOLLOWING, params); + } + + public void openChannelForm(Claim claim) { + Map params = new HashMap<>(); + if (claim != null) { + params.put("claim", claim); + } + openFragment(ChannelFormFragment.class, true, NavMenuItem.ID_ITEM_CHANNELS, params); + } + + public void openPublishesOnSuccessfulPublish() { + // close publish form + getSupportFragmentManager().popBackStack(); + openFragment(PublishesFragment.class, true, NavMenuItem.ID_ITEM_PUBLISHES); + } + + public void openPublishForm(Claim claim) { + Map params = new HashMap<>(); + if (claim != null) { + params.put("claim", claim); + } + openFragment(PublishFormFragment.class, true, NavMenuItem.ID_ITEM_NEW_PUBLISH, params); + } + + 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); + } + + private Claim getCachedClaimForUrl(String url) { + ClaimCacheKey key = new ClaimCacheKey(); + key.setUrl(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); + } + + public void openAllContentFragmentWithTag(String tag) { + Map params = new HashMap<>(); + params.put("singleTag", tag); + openFragment(AllContentFragment.class, true, NavMenuItem.ID_ITEM_ALL_CONTENT, params); + } + + public void openFileUrl(String url) { + Map params = new HashMap<>(); + params.put("url", url); + openFragment(FileViewFragment.class, true, NavMenuItem.ID_ITEM_FOLLOWING, params); + } + + public void openFileClaim(Claim claim) { + Map params = new HashMap<>(); + params.put("claimId", claim.getClaimId()); + params.put("url", !Helper.isNullOrEmpty(claim.getShortUrl()) ? claim.getShortUrl() : claim.getPermanentUrl()); + openFragment(FileViewFragment.class, true, NavMenuItem.ID_ITEM_FOLLOWING, params); + } + + 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.floating_balance_main_container).setVisibility(View.GONE); + findViewById(R.id.global_now_playing_card).setVisibility(View.GONE); + findViewById(R.id.global_sdk_initializing_status).setVisibility(View.GONE); + getSupportActionBar().hide(); + + PlayerView pipPlayer = findViewById(R.id.pip_player); + pipPlayer.setVisibility(View.VISIBLE); + pipPlayer.setPlayer(appPlayer); + pipPlayer.setUseController(false); + playerReassigned = true; + } + private void renderFullMode() { + getSupportActionBar().show(); + findViewById(R.id.content_main).setVisibility(View.VISIBLE); + findViewById(R.id.floating_balance_main_container).setVisibility(View.VISIBLE); + Fragment fragment = getCurrentFragment(); + if (!(fragment instanceof FileViewFragment)) { + findViewById(R.id.global_now_playing_card).setVisibility(View.VISIBLE); + } + if (!Lbry.SDK_READY) { + findViewById(R.id.global_sdk_initializing_status).setVisibility(View.VISIBLE); + } + + PlayerView pipPlayer = findViewById(R.id.pip_player); + pipPlayer.setVisibility(View.INVISIBLE); + pipPlayer.setPlayer(null); + playerReassigned = true; + } + + @Override + protected void onDestroy() { + unregisterReceivers(); + NotificationManagerCompat notificationManager = NotificationManagerCompat.from(this); + if (receivedStopService || !isServiceRunning(this, LbrynetService.class)) { + notificationManager.cancelAll(); + } + if (dbHelper != null) { + dbHelper.close(); + } + if (mediaSession != null) { + mediaSession.release(); + } + stopExoplayer(); + nowPlayingClaim = null; + nowPlayingClaimUrl = null; + appStarted = false; + + if (!keepSdkBackground()) { + sendBroadcast(new Intent(LbrynetService.ACTION_STOP_SERVICE)); + } + + super.onDestroy(); + } + + public static void stopExoplayer() { + if (appPlayer != null) { + appPlayer.stop(true); + appPlayer.release(); + appPlayer = null; + } + if (playerCache != null) { + playerCache.release(); + playerCache = 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; + updateFloatingWalletBalance(); + updateUsdWalletBalanceInNav(); + + sendBroadcast(new Intent(ACTION_WALLET_BALANCE_UPDATED)); + } + + @Override + public void onError(Exception error) { + + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + @Override + protected void onResume() { + super.onResume(); + + applyNavbarSigninPadding(); + checkFirstRun(); + checkNowPlaying(); + fetchRewards(); + + // check (and start) the LBRY SDK service serviceRunning = isServiceRunning(this, LbrynetService.class); if (!serviceRunning) { - CurrentLaunchTiming.setColdStart(true); + Lbry.SDK_READY = false; + findViewById(R.id.global_sdk_initializing_status).setVisibility(View.VISIBLE); ServiceHelper.start(this, "", LbrynetService.class, "lbrynetservice"); } checkSdkReady(); + showSignedInUser(); + checkPendingOpens(); - checkNotificationOpenIntent(getIntent()); + if (Lbry.SDK_READY) { + findViewById(R.id.global_sdk_initializing_status).setVisibility(View.GONE); + } + } - 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); + private void checkPendingOpens() { + if (pendingFollowingReload) { + loadFollowingContent(); + pendingFollowingReload = false; + } + if (!Helper.isNullOrEmpty(pendingAllContentTag)) { + openAllContentFragmentWithTag(pendingAllContentTag); + pendingAllContentTag = null; + } else if (!Helper.isNullOrEmpty(pendingChannelUrl)) { + openChannelUrl(pendingChannelUrl); + pendingChannelUrl = null; + } else if (pendingOpenWalletPage) { + openFragment(WalletFragment.class, true, NavMenuItem.ID_ITEM_WALLET); + } else if (pendingOpenRewardsPage) { + openFragment(RewardsFragment.class, true, NavMenuItem.ID_ITEM_REWARDS); + } + } - registerNotificationsReceiver(); + @Override + protected void onPause() { + if (!enteringPIPMode && appPlayer != null) { + appPlayer.setPlayWhenReady(false); + } + super.onPause(); + } - setContentView(mReactRootView); + public static void suspendGlobalPlayer(Context context) { + if (MainActivity.appPlayer != null) { + MainActivity.appPlayer.setPlayWhenReady(false); + } + if (context instanceof MainActivity) { + ((MainActivity) context).hideGlobalNowPlaying(); + } + } + public static void resumeGlobalPlayer(Context context) { + if (context instanceof MainActivity) { + ((MainActivity) context).checkNowPlaying(); + } + } + + private void toggleUrlSuggestions(boolean visible) { + View container = findViewById(R.id.url_suggestions_container); + View closeIcon = findViewById(R.id.wunderbar_close); + EditText wunderbar = findViewById(R.id.wunderbar); + wunderbar.setPadding(0, 0, visible ? getScaledValue(36) : 0, 0); + + container.setVisibility(visible ? View.VISIBLE : View.GONE); + closeIcon.setVisibility(visible ? View.VISIBLE : View.GONE); + } + + public int getScaledValue(int value) { + float scale = getResources().getDisplayMetrics().density; + return Helper.getScaledValue(value, scale); + } + + + public boolean canShowUrlSuggestions() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + return sp.getBoolean(MainActivity.PREFERENCE_KEY_SHOW_URL_SUGGESTIONS, false); + } + + public boolean keepSdkBackground() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + return sp.getBoolean(MainActivity.PREFERENCE_KEY_KEEP_SDK_BACKGROUND, true); + } + + private void setupUriBar() { + findViewById(R.id.wunderbar_close).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + clearWunderbarFocus(view); + } + }); + + EditText wunderbar = findViewById(R.id.wunderbar); + wunderbar.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + if (hasFocus) { + InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE); + imm.showSoftInput(view, 0); + } + + if (canShowUrlSuggestions()) { + toggleUrlSuggestions(hasFocus); + if (hasFocus && Helper.isNullOrEmpty(Helper.getValue(((EditText) view).getText()))) { + displayUrlSuggestionsForNoInput(); + } + } + } + }); + + 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 && canShowUrlSuggestions()) { + handleUriInputChanged(charSequence.toString().trim()); + } + } + + @Override + public void afterTextChanged(Editable editable) { + + } + }); + wunderbar.setOnEditorActionListener(new TextView.OnEditorActionListener() { + @Override + public boolean onEditorAction(TextView textView, int actionId, KeyEvent keyEvent) { + if (actionId == EditorInfo.IME_ACTION_GO) { + String input = Helper.getValue(wunderbar.getText()); + boolean handled = false; + if (input.startsWith(LbryUri.PROTO_DEFAULT) && !input.equalsIgnoreCase(LbryUri.PROTO_DEFAULT)) { + try { + LbryUri uri = LbryUri.parse(input); + if (uri.isChannel()) { + openChannelUrl(uri.toString()); + clearWunderbarFocus(wunderbar); + handled = true; + } else { + openFileUrl(uri.toString()); + clearWunderbarFocus(wunderbar); + handled = true; + } + } catch (LbryUriException ex) { + // pass + } + } + if (!handled) { + // search + launchSearch(input); + clearWunderbarFocus(wunderbar); + } + + return true; + } + + return false; + } + }); + + 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 + if (urlSuggestion.getClaim() != null) { + openChannelClaim(urlSuggestion.getClaim()); + } else { + openChannelUrl(urlSuggestion.getUri().toString()); + } + break; + case UrlSuggestion.TYPE_FILE: + if (urlSuggestion.getClaim() != null) { + openFileClaim(urlSuggestion.getClaim()); + } else { + openFileUrl(urlSuggestion.getUri().toString()); + } + break; + case UrlSuggestion.TYPE_SEARCH: + launchSearch(urlSuggestion.getText()); + break; + case UrlSuggestion.TYPE_TAG: + // open tag page + openAllContentFragmentWithTag(urlSuggestion.getText()); + break; + } + clearWunderbarFocus(findViewById(R.id.wunderbar)); + } + }); + + RecyclerView urlSuggestionList = findViewById(R.id.url_suggestions); + LinearLayoutManager llm = new LinearLayoutManager(this); + urlSuggestionList.setLayoutManager(llm); + urlSuggestionList.setAdapter(urlSuggestionListAdapter); + } + + public void clearWunderbarFocus(View view) { + findViewById(R.id.wunderbar).clearFocus(); + findViewById(R.id.app_bar_main_container).requestFocus(); + InputMethodManager imm = (InputMethodManager) getSystemService(Activity.INPUT_METHOD_SERVICE); + imm.hideSoftInputFromWindow(view.getWindowToken(), 0); + } + public View getWunderbar() { + return findViewById(R.id.wunderbar); + } + + 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 ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + if (findViewById(R.id.url_suggestions_container).getVisibility() == View.VISIBLE) { + for (int i = 0; i < claims.size(); i++) { + // build a simple url from the claim for matching + Claim claim = claims.get(i); + Claim actualClaim = claim; + boolean isRepost = false; + if (Claim.TYPE_REPOST.equalsIgnoreCase(claim.getValueType())) { + actualClaim = claim.getRepostedClaim(); + isRepost = true; + } + if (Helper.isNullOrEmpty(claim.getName())) { + continue; + } + + LbryUri simpleUrl = new LbryUri(); + if (actualClaim.getName().startsWith("@") && !isRepost) { + // channel + simpleUrl.setChannelName(actualClaim.getName()); + } else { + simpleUrl.setStreamName(claim.getName()); + } + + urlSuggestionListAdapter.setClaimForUrl(simpleUrl, actualClaim); + } + urlSuggestionListAdapter.notifyDataSetChanged(); + } + } + + @Override + public void onError(Exception error) { + + } + }); + task.executeOnExecutor(AsyncTask.SERIAL_EXECUTOR); + } + + private void displayUrlSuggestionsForNoInput() { + urlSuggestionListAdapter.clear(); + loadDefaultSuggestionsForBlankUrl(); + } + + private void handleUriInputChanged(String text) { + // build the default suggestions + urlSuggestionListAdapter.clear(); + if (Helper.isNullOrEmpty(text) || text.trim().equals("@")) { + displayUrlSuggestionsForNoInput(); + return; + } + + List defaultSuggestions = buildDefaultSuggestions(text); + urlSuggestionListAdapter.addUrlSuggestions(defaultSuggestions); + if (LbryUri.PROTO_DEFAULT.equalsIgnoreCase(text)) { + return; + } + + LighthouseAutoCompleteTask task = new LighthouseAutoCompleteTask(text, null, new LighthouseAutoCompleteTask.AutoCompleteResultHandler() { + @Override + public void onSuccess(List suggestions) { + String wunderBarText = Helper.getValue(((EditText) findViewById(R.id.wunderbar)).getText()); + if (wunderBarText.equalsIgnoreCase(text)) { + urlSuggestionListAdapter.addUrlSuggestions(suggestions); + List urls = urlSuggestionListAdapter.getItemUrls(); + resolveUrlSuggestions(urls); + } + } + + @Override + public void onError(Exception error) { + + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void loadDefaultSuggestionsForBlankUrl() { + if (recentUrlHistory != null && recentUrlHistory.size() > 0) { + urlSuggestionListAdapter.addUrlSuggestions(recentUrlHistory); + } + + FetchRecentUrlHistoryTask task = new FetchRecentUrlHistoryTask(DatabaseHelper.getInstance(), new FetchRecentUrlHistoryTask.FetchRecentUrlHistoryHandler() { + @Override + public void onSuccess(List recentHistory) { + List suggestions = new ArrayList<>(recentHistory); + List lbrySuggestions = buildLbryUrlSuggestions(); + if (suggestions.size() < 10) { + for (int i = suggestions.size(), j = 0; i < 10 && j < lbrySuggestions.size(); i++, j++) { + suggestions.add(lbrySuggestions.get(j)); + } + } else if (suggestions.size() == 0) { + suggestions.addAll(lbrySuggestions); + } + + for (UrlSuggestion suggestion : suggestions) { + suggestion.setUseTextAsDescription(true); + } + + recentUrlHistory = new ArrayList<>(suggestions); + urlSuggestionListAdapter.clear(); + urlSuggestionListAdapter.addUrlSuggestions(recentUrlHistory); + List urls = urlSuggestionListAdapter.getItemUrls(); + resolveUrlSuggestions(urls); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private List buildLbryUrlSuggestions() { + List suggestions = new ArrayList<>(); + suggestions.add(new UrlSuggestion( + UrlSuggestion.TYPE_FILE, "What is LBRY?", LbryUri.tryParse("lbry://what#19b9c243bea0c45175e6a6027911abbad53e983e"))); + suggestions.add(new UrlSuggestion( + UrlSuggestion.TYPE_CHANNEL, "LBRYCast", LbryUri.tryParse("lbry://@lbrycast#4c29f8b013adea4d5cca1861fb2161d5089613ea"))); + suggestions.add(new UrlSuggestion( + UrlSuggestion.TYPE_CHANNEL, "The LBRY Channel", LbryUri.tryParse("lbry://@lbry#3fda836a92faaceedfe398225fb9b2ee2ed1f01a"))); + for (UrlSuggestion suggestion : suggestions) { + suggestion.setUseTextAsDescription(true); + } + return suggestions; + } + + private List buildDefaultSuggestions(String text) { + List suggestions = new ArrayList(); + + if (LbryUri.PROTO_DEFAULT.equalsIgnoreCase(text)) { + loadDefaultSuggestionsForBlankUrl(); + return recentUrlHistory != null ? recentUrlHistory : new ArrayList<>(); + } + + // First item is always search + if (!text.startsWith(LbryUri.PROTO_DEFAULT)) { + UrlSuggestion searchSuggestion = new UrlSuggestion(UrlSuggestion.TYPE_SEARCH, text); + suggestions.add(searchSuggestion); + } + + if (!text.matches(LbryUri.REGEX_INVALID_URI)) { + boolean isUrlWithScheme = text.startsWith(LbryUri.PROTO_DEFAULT); + boolean isChannel = text.startsWith("@"); + LbryUri uri = null; + if (isUrlWithScheme && text.length() > 7) { + try { + uri = LbryUri.parse(text); + isChannel = uri.isChannel(); + } catch (LbryUriException ex) { + // pass + } + } + + if (!isChannel) { + if (uri == null) { + 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 should not contain spaces + if (isChannel) { + if (uri == null) { + uri = new LbryUri(); + uri.setChannelName(text); + } + UrlSuggestion suggestion = new UrlSuggestion(UrlSuggestion.TYPE_CHANNEL, text); + suggestion.setUri(uri); + suggestions.add(suggestion); + } + } + + if (!isUrlWithScheme && !isChannel) { + UrlSuggestion suggestion = new UrlSuggestion(UrlSuggestion.TYPE_TAG, text); + suggestions.add(suggestion); + } + } + + return suggestions; + } + + public void checkNowPlaying() { + Fragment fragment = getCurrentFragment(); + if (fragment instanceof FileViewFragment) { + return; + } + + 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); + playerReassigned = true; + + getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON); + } + } + + public void hideGlobalNowPlaying() { + findViewById(R.id.global_now_playing_card).setVisibility(View.GONE); + } + + public void unsetFitsSystemWindows(View view) { + view.setFitsSystemWindows(false); + } + + public void enterFullScreenMode() { + hideFloatingWalletBalance(); + ActionBar actionBar = getSupportActionBar(); + if (actionBar != null) { + actionBar.hide(); + } + findViewById(R.id.global_sdk_initializing_status).setVisibility(View.GONE); + findViewById(R.id.app_bar_main_container).setFitsSystemWindows(false); + + View decorView = getWindow().getDecorView(); + decorView.setSystemUiVisibility( + View.SYSTEM_UI_FLAG_HIDE_NAVIGATION | + View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY | + View.SYSTEM_UI_FLAG_FULLSCREEN | + View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | + View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION | + View.SYSTEM_UI_FLAG_LAYOUT_STABLE); + } + + public int getStatusBarHeight() { + int height = 0; + int resourceId = getResources().getIdentifier("status_bar_height", "dimen", "android"); + if (resourceId > 0) { + height = getResources().getDimensionPixelSize(resourceId); + } + return height; + } + + public void exitFullScreenMode() { + View appBarMainContainer = findViewById(R.id.app_bar_main_container); + View decorView = getWindow().getDecorView(); + int flags = isDarkMode() ? (View.SYSTEM_UI_FLAG_LAYOUT_STABLE | View.SYSTEM_UI_FLAG_VISIBLE) : + (View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR | View.SYSTEM_UI_FLAG_LAYOUT_STABLE | View.SYSTEM_UI_FLAG_VISIBLE); + appBarMainContainer.setFitsSystemWindows(true); + decorView.setSystemUiVisibility(flags); + + if (!Lbry.SDK_READY) { + findViewById(R.id.global_sdk_initializing_status).setVisibility(View.VISIBLE); + } + showFloatingWalletBalance(); + ActionBar actionBar = getSupportActionBar(); + if (actionBar != null) { + actionBar.show(); + } + } + + 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_INTERNAL_FIRST_RUN_COMPLETED, false); + if (!firstRunCompleted) { + startActivity(new Intent(this, FirstRunActivity.class)); + } else if (!appStarted) { + // first run completed, startup + startup(); + return; + } + + if (getSupportFragmentManager().getBackStackEntryCount() == 0) { + openFragment(FollowingFragment.class, false, NavMenuItem.ID_ITEM_FOLLOWING); + } + } + + 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 + Log.e(TAG, "Could not decrypt existing auth token.", ex); + } + } } private void checkSdkReady() { - if (!lbrySdkReady) { + 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(); + scheduleWalletSyncTask(); + initFloatingWalletBalance(); + } + } + + public void onSdkReady() { + if (Lbryio.isSignedIn()) { + checkSyncedWallet(); + } + + findViewById(R.id.global_sdk_initializing_status).setVisibility(View.GONE); + + syncWalletAndLoadPreferences(); + scheduleWalletBalanceUpdate(); + scheduleWalletSyncTask(); + fetchOwnChannels(); + fetchOwnClaims(); + + initFloatingWalletBalance(); + + checkAndClaimNewAndroidReward(); + } + + public void checkAndClaimNewAndroidReward() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + boolean rewardClaimed = sp.getBoolean(PREFERENCE_KEY_INTERNAL_NEW_ANDROID_REWARD_CLAIMED, false); + if (!rewardClaimed) { + ClaimRewardTask task = new ClaimRewardTask( + Reward.TYPE_NEW_ANDROID, + null, + null, + this, + new ClaimRewardTask.ClaimRewardHandler() { + @Override + public void onSuccess(double amountClaimed, String message) { + if (Helper.isNullOrEmpty(message)) { + message = getResources().getQuantityString( + R.plurals.claim_reward_message, + amountClaimed == 1 ? 1 : 2, + new DecimalFormat(Helper.LBC_CURRENCY_FORMAT_PATTERN).format(amountClaimed)); + } + Snackbar.make(findViewById(R.id.content_main), message, Snackbar.LENGTH_LONG).show(); + if (sp != null) { + sp.edit().putBoolean(PREFERENCE_KEY_INTERNAL_NEW_ANDROID_REWARD_CLAIMED, true).apply(); + } + } + + @Override + public void onError(Exception error) { + // pass. fail silently + if (sp != null) { + sp.edit().putBoolean(PREFERENCE_KEY_INTERNAL_NEW_ANDROID_REWARD_CLAIMED, true).apply(); + } + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + public void initMediaSession() { + ComponentName mediaButtonReceiver = new ComponentName(getApplicationContext(), MediaButtonReceiver.class); + mediaSession = new MediaSessionCompat(getApplicationContext(), "LBRYMediaSession", mediaButtonReceiver, null); + MediaSessionConnector connector = new MediaSessionConnector(mediaSession); + connector.setPlayer(MainActivity.appPlayer); + mediaSession.setActive(true); + } + + public void showFloatingWalletBalance() { + findViewById(R.id.floating_balance_main_container).setVisibility(View.VISIBLE); + } + public void hideFloatingWalletBalance() { + findViewById(R.id.floating_balance_main_container).setVisibility(View.GONE); + } + public void hideFloatingRewardsValue() { + findViewById(R.id.floating_reward_container).setVisibility(View.INVISIBLE); + } + + private void initFloatingWalletBalance() { + findViewById(R.id.floating_balance_container).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + openFragment(WalletFragment.class, true, NavMenuItem.ID_ITEM_WALLET); + } + }); + findViewById(R.id.floating_reward_container).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + openFragment(RewardsFragment.class, true, NavMenuItem.ID_ITEM_REWARDS); + } + }); + } + + private void updateUsdWalletBalanceInNav() { + double usdBalance = Lbry.walletBalance.getAvailable().doubleValue() * Lbryio.LBCUSDRate; + if (navMenuAdapter != null) { + navMenuAdapter.setExtraLabelForItem( + NavMenuItem.ID_ITEM_WALLET, + Lbryio.LBCUSDRate > 0 ? String.format("$%s", Helper.SIMPLE_CURRENCY_FORMAT.format(usdBalance)) : null + ); + } + } + + private void updateFloatingWalletBalance() { + if (!hasLoadedFirstBalance) { + findViewById(R.id.floating_balance_loading).setVisibility(View.GONE); + findViewById(R.id.floating_balance_value).setVisibility(View.VISIBLE); + hasLoadedFirstBalance = true; + } + + ((TextView) findViewById(R.id.floating_balance_value)).setText(Helper.shortCurrencyFormat( + Lbry.walletBalance == null ? 0 : Lbry.walletBalance.getAvailable().doubleValue())); + } + + private void scheduleWalletBalanceUpdate() { + if (scheduler != null && !walletBalanceUpdateScheduled) { + scheduler.scheduleAtFixedRate(new Runnable() { + @Override + public void run() { + updateWalletBalance(); + } + }, 0, 5, TimeUnit.SECONDS); + walletBalanceUpdateScheduled = true; + } + } + + private void scheduleWalletSyncTask() { + if (scheduler != null && !walletSyncScheduled) { + scheduler.scheduleAtFixedRate(new Runnable() { + @Override + public void run() { + syncWalletAndLoadPreferences(); + } + }, 0, 5, TimeUnit.MINUTES); + walletSyncScheduled = true; + } + } + + public void saveSharedUserState() { + if (!userSyncEnabled()) { + return; + } + SaveSharedUserStateTask saveTask = new SaveSharedUserStateTask(new SaveSharedUserStateTask.SaveSharedUserStateHandler() { + @Override + public void onSuccess() { + // push wallet sync changes + pushCurrentWalletSync(); + } + + @Override + public void onError(Exception error) { + // pass + } + }); + saveTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void loadSharedUserState() { + // load wallet preferences + LoadSharedUserStateTask loadTask = new LoadSharedUserStateTask(MainActivity.this, new LoadSharedUserStateTask.LoadSharedUserStateHandler() { + @Override + public void onSuccess(List subscriptions, List followedTags) { + if (subscriptions != null && subscriptions.size() > 0) { + // reload subscriptions if wallet fragment is FollowingFragment + //openNavFragments.get + MergeSubscriptionsTask mergeTask = new MergeSubscriptionsTask( + subscriptions, MainActivity.this, new MergeSubscriptionsTask.MergeSubscriptionsHandler() { + @Override + public void onSuccess(List subscriptions, List diff) { + Lbryio.subscriptions = new ArrayList<>(subscriptions); + if (diff != null && diff.size() > 0) { + saveSharedUserState(); + } + for (Fragment fragment : openNavFragments.values()) { + if (fragment instanceof FollowingFragment) { + // reload local subscriptions + ((FollowingFragment) fragment).fetchLoadedSubscriptions(true); + } + } + } + + @Override + public void onError(Exception error) { + Log.e(TAG, String.format("merge subscriptions failed: %s", error.getMessage()), error); + } + }); + mergeTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + if (followedTags != null && followedTags.size() > 0) { + List previousTags = new ArrayList<>(Lbry.followedTags); + Lbry.followedTags = new ArrayList<>(followedTags); + for (Fragment fragment : openNavFragments.values()) { + if (fragment instanceof AllContentFragment) { + AllContentFragment acFragment = (AllContentFragment) fragment; + if (!acFragment.isSingleTagView() && + acFragment.getCurrentContentScope() == ContentScopeDialogFragment.ITEM_TAGS && + !previousTags.equals(followedTags)) { + acFragment.fetchClaimSearchContent(true); + } + } + } + } + } + + @Override + public void onError(Exception error) { + Log.e(TAG, String.format("load shared user state failed: %s", error != null ? error.getMessage() : "no error message"), error); + } + }); + loadTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void pushCurrentWalletSync() { + String password = Utils.getSecureValue(SECURE_VALUE_KEY_SAVED_PASSWORD, this, Lbry.KEYSTORE); + SyncApplyTask fetchTask = new SyncApplyTask(true, password, new DefaultSyncTaskHandler() { + @Override + public void onSyncApplySuccess(String hash, String data) { + SyncSetTask setTask = new SyncSetTask(Lbryio.lastRemoteHash, hash, data, null); + setTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + @Override + public void onSyncApplyError(Exception error) { } + }); + fetchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private boolean userSyncEnabled() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + boolean walletSyncEnabled = sp.getBoolean(PREFERENCE_KEY_INTERNAL_WALLET_SYNC_ENABLED, false); + return walletSyncEnabled && Lbryio.isSignedIn(); + } + + public void syncSet(String hash, String data) { + if (syncSetTask == null || syncSetTask.getStatus() == AsyncTask.Status.FINISHED) { + syncSetTask = new SyncSetTask(Lbryio.lastRemoteHash, hash, data, new DefaultSyncTaskHandler() { + @Override + public void onSyncSetSuccess(String hash) { + Lbryio.lastRemoteHash = hash; + WalletSync walletSync = new WalletSync(hash, data); + Lbryio.lastWalletSync = walletSync; + + if (pendingSyncSetQueue.size() > 0) { + fullSyncInProgress = true; + WalletSync nextSync = pendingSyncSetQueue.remove(0); + syncSet(nextSync.getHash(), nextSync.getData()); + } else if (queuedSyncCount > 0) { + queuedSyncCount--; + syncApplyAndSet(); + } + + fullSyncInProgress = false; + } + @Override + public void onSyncSetError(Exception error) { + // log app exceptions + if (pendingSyncSetQueue.size() > 0) { + WalletSync nextSync = pendingSyncSetQueue.remove(0); + syncSet(nextSync.getHash(), nextSync.getData()); + } else if (queuedSyncCount > 0) { + queuedSyncCount--; + syncApplyAndSet(); + } + + fullSyncInProgress = false; + } + }); + syncSetTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } else { + WalletSync pending = new WalletSync(hash, data); + pendingSyncSetQueue.add(pending); + } + } + + public void syncApplyAndSet() { + fullSyncInProgress = true; + String password = Utils.getSecureValue(SECURE_VALUE_KEY_SAVED_PASSWORD, this, Lbry.KEYSTORE); + SyncApplyTask fetchTask = new SyncApplyTask(true, password, new DefaultSyncTaskHandler() { + @Override + public void onSyncApplySuccess(String hash, String data) { + if (!hash.equalsIgnoreCase(Lbryio.lastRemoteHash)) { + syncSet(hash, data); + } else { + fullSyncInProgress = false; + queuedSyncCount = 0; + } + } + @Override + public void onSyncApplyError(Exception error) { + fullSyncInProgress = false; + if (queuedSyncCount > 0) { + queuedSyncCount--; + syncApplyAndSet(); + } + } + }); + fetchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void syncWalletAndLoadPreferences() { + if (!userSyncEnabled()) { + return; + } + if (fullSyncInProgress) { + queuedSyncCount++; + } + + fullSyncInProgress = true; + String password = Utils.getSecureValue(SECURE_VALUE_KEY_SAVED_PASSWORD, this, Lbry.KEYSTORE); + SyncGetTask task = new SyncGetTask(password, true, null, new DefaultSyncTaskHandler() { + @Override + public void onSyncGetSuccess(WalletSync walletSync) { + Lbryio.lastWalletSync = walletSync; + Lbryio.lastRemoteHash = walletSync.getHash(); + loadSharedUserState(); + } + + @Override + public void onSyncGetWalletNotFound() { + // pass. This actually shouldn't happen at this point. + // But if it does, send what we have + if (Lbryio.isSignedIn() && userSyncEnabled()) { + syncApplyAndSet(); + } + } + + @Override + public void onSyncGetError(Exception error) { + // pass + Log.e(TAG, String.format("sync get failed: %s", error != null ? error.getMessage() : "no error message"), error); + + fullSyncInProgress = false; + if (queuedSyncCount > 0) { + queuedSyncCount--; + syncApplyAndSet(); + } + } + + @Override + public void onSyncApplySuccess(String hash, String data) { + if (!hash.equalsIgnoreCase(Lbryio.lastRemoteHash)) { + syncSet(hash, data); + } else { + fullSyncInProgress = false; + queuedSyncCount = 0; + } + + loadSharedUserState(); + } + + @Override + public void onSyncApplyError(Exception error) { + // pass + Log.e(TAG, String.format("sync apply failed: %s", error != null ? error.getMessage() : "no error message"), error); + fullSyncInProgress = false; + if (queuedSyncCount > 0) { + queuedSyncCount--; + syncApplyAndSet(); + } + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void registerRequestsReceiver() { + IntentFilter intentFilter = new IntentFilter(); + intentFilter.addAction(ACTION_AUTH_TOKEN_GENERATED); + intentFilter.addAction(ACTION_USER_SIGN_IN_SUCCESS); + intentFilter.addAction(ACTION_OPEN_ALL_CONTENT_TAG); + intentFilter.addAction(ACTION_OPEN_CHANNEL_URL); + intentFilter.addAction(ACTION_OPEN_WALLET_PAGE); + intentFilter.addAction(ACTION_OPEN_REWARDS_PAGE); + intentFilter.addAction(ACTION_SAVE_SHARED_USER_STATE); + 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_USER_SIGN_IN_SUCCESS.equalsIgnoreCase(action)) { + handleUserSignInSuccess(intent); + } else if (ACTION_OPEN_ALL_CONTENT_TAG.equalsIgnoreCase(action)) { + handleOpenContentTag(intent); + } else if (ACTION_OPEN_CHANNEL_URL.equalsIgnoreCase(action)) { + handleOpenChannelUrl(intent); + } else if (ACTION_OPEN_WALLET_PAGE.equalsIgnoreCase(action)) { + pendingOpenWalletPage = true; + } else if (ACTION_OPEN_REWARDS_PAGE.equalsIgnoreCase(action)) { + pendingOpenRewardsPage = true; + } else if (ACTION_SAVE_SHARED_USER_STATE.equalsIgnoreCase(action)) { + saveSharedUserState(); + } + } + + 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(Intent intent) { + String url = intent.getStringExtra("url"); + pendingChannelUrl = url; + } + }; + registerReceiver(requestsReceiver, intentFilter); + } + + private void loadFollowingContent() { + for (Fragment fragment : openNavFragments.values()) { + if (fragment instanceof FollowingFragment) { + ((FollowingFragment) fragment).loadFollowing(); + } + } + } + public void showMessage(int stringResourceId) { + Snackbar.make(findViewById(R.id.content_main), stringResourceId, Snackbar.LENGTH_LONG).show(); + } + public void showMessage(String message) { + Snackbar.make(findViewById(R.id.content_main), message, Snackbar.LENGTH_LONG).show(); + } + public void showError(String message) { + Snackbar.make(findViewById(R.id.content_main), message, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + + @Override + public void onBackPressed() { + + if (findViewById(R.id.url_suggestions_container).getVisibility() == View.VISIBLE) { + clearWunderbarFocus(findViewById(R.id.wunderbar)); + return; + } + if (backPressInterceptor != null && backPressInterceptor.onBackPressed()) { + return; + } + + DrawerLayout drawer = findViewById(R.id.drawer_layout); + if (drawer.isDrawerOpen(GravityCompat.START)) { + drawer.closeDrawer(GravityCompat.START); + } else { + boolean handled = false; + // TODO: Refactor both forms as back press interceptors? + ChannelFormFragment channelFormFragment = null; + PublishFormFragment publishFormFragment = null; + for (Fragment fragment : openNavFragments.values()) { + if (fragment instanceof ChannelFormFragment) { + channelFormFragment = ((ChannelFormFragment) fragment); + break; + } + if (fragment instanceof PublishFormFragment) { + publishFormFragment = ((PublishFormFragment) fragment); + break; + } + } + if (channelFormFragment != null && channelFormFragment.isSaveInProgress()) { + handled = true; + return; + } + if (publishFormFragment != null && (publishFormFragment.isSaveInProgress() || publishFormFragment.isTranscodeInProgress())) { + if (publishFormFragment.isTranscodeInProgress()) { + showMessage(R.string.transcode_in_progress); + } + handled = true; + return; + } + + if (!handled) { + // 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); + } + + public void rewardsSignIn() { + Intent intent = new Intent(this, VerificationActivity.class); + intent.putExtra("flow", VerificationActivity.VERIFICATION_FLOW_REWARDS); + startActivityForResult(intent, REQUEST_REWARDS_VERIFY_SIGN_IN); + } + + @Override + public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) { + switch (requestCode) { + case REQUEST_STORAGE_PERMISSION: + if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { + for (StoragePermissionListener listener : storagePermissionListeners) { + listener.onStoragePermissionGranted(); + } + } else { + for (StoragePermissionListener listener : storagePermissionListeners) { + listener.onStoragePermissionRefused(); + } + } + startingPermissionRequest = false; + break; + + case REQUEST_CAMERA_PERMISSION: + if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { + for (CameraPermissionListener listener : cameraPermissionListeners) { + listener.onCameraPermissionGranted(); + } + } else { + for (CameraPermissionListener listener : cameraPermissionListeners) { + listener.onCameraPermissionRefused(); + } + } + startingPermissionRequest = false; + break; + } + } + + @Override + protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) { + super.onActivityResult(requestCode, resultCode, data); + if (requestCode == REQUEST_FILE_PICKER) { + startingFilePickerActivity = false; + if (resultCode == RESULT_OK) { + Uri fileUri = data.getData(); + String filePath = Helper.getRealPathFromURI_API19(this, fileUri); + for (FilePickerListener listener : filePickerListeners) { + listener.onFilePicked(filePath); + } + } else { + for (FilePickerListener listener : filePickerListeners) { + listener.onFilePickerCancelled(); + } + } + } else if (requestCode == REQUEST_SIMPLE_SIGN_IN || requestCode == REQUEST_WALLET_SYNC_SIGN_IN) { + if (resultCode == RESULT_OK) { + // user signed in + showSignedInUser(); + + if (requestCode == REQUEST_WALLET_SYNC_SIGN_IN) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + sp.edit().putBoolean(MainActivity.PREFERENCE_KEY_INTERNAL_WALLET_SYNC_ENABLED, true).apply(); + + for (Fragment fragment : openNavFragments.values()) { + if (fragment instanceof WalletFragment) { + ((WalletFragment) fragment).onWalletSyncEnabled(); + } + } + scheduleWalletSyncTask(); + } + } + } else if (requestCode == REQUEST_VIDEO_CAPTURE || requestCode == REQUEST_TAKE_PHOTO) { + if (resultCode == RESULT_OK) { + PublishFragment publishFragment = null; + for (Fragment fragment : openNavFragments.values()) { + if (fragment instanceof PublishFragment) { + publishFragment = (PublishFragment) fragment; + break; + } + } + + Map params = new HashMap<>(); + params.put("directFilePath", cameraOutputFilename); + if (publishFragment != null) { + params.put("suggestedUrl", publishFragment.getSuggestedPublishUrl()); + } + openFragment(PublishFormFragment.class, true, NavMenuItem.ID_ITEM_NEW_PUBLISH, params); + } + cameraOutputFilename = null; + } + } + + public void requestVideoCapture() { + Intent intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE); + if (intent.resolveActivity(getPackageManager()) != null) { + String outputPath = String.format("%s/record", Utils.getAppInternalStorageDir(this)); + File dir = new File(outputPath); + if (!dir.isDirectory()) { + dir.mkdirs(); + } + + cameraOutputFilename = String.format("%s/VID_%s.mp4", outputPath, Helper.FILESTAMP_FORMAT.format(new Date())); + Uri outputUri = FileProvider.getUriForFile(this, String.format("%s.fileprovider", getPackageName()), new File(cameraOutputFilename)); + intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri); + startActivityForResult(intent, REQUEST_VIDEO_CAPTURE); + return; + } + + showError(getString(R.string.cannot_capture_video)); + } + + public void requestTakePhoto() { + Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); + if (intent.resolveActivity(getPackageManager()) != null) { + String outputPath = String.format("%s/photos", Utils.getAppInternalStorageDir(this)); + File dir = new File(outputPath); + if (!dir.isDirectory()) { + dir.mkdirs(); + } + + cameraOutputFilename = String.format("%s/IMG_%s.jpg", outputPath, Helper.FILESTAMP_FORMAT.format(new Date())); + Uri outputUri = FileProvider.getUriForFile(this, String.format("%s.fileprovider", getPackageName()), new File(cameraOutputFilename)); + intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri); + startActivityForResult(intent, REQUEST_TAKE_PHOTO); + return; + } + + showError(getString(R.string.cannot_take_photo)); + } + + + + private void applyNavbarSigninPadding() { + int statusBarHeight = getStatusBarHeight(); + + View signInButton = findViewById(R.id.sign_in_button_container); + View signedInEmailContainer = findViewById(R.id.signed_in_email_container); + signInButton.setPadding(0, statusBarHeight, 0, 0); + signedInEmailContainer.setPadding(0, statusBarHeight, 0, 0); + } + + private void showSignedInUser() { + if (Lbryio.isSignedIn()) { + findViewById(R.id.sign_in_button_container).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 renderStartupFailed(Map startupStages) { + Map startupStageIconIds = new HashMap<>(); + startupStageIconIds.put(STARTUP_STAGE_INSTALL_ID_LOADED, R.id.startup_stage_icon_install_id); + startupStageIconIds.put(STARTUP_STAGE_KNOWN_TAGS_LOADED, R.id.startup_stage_icon_known_tags); + startupStageIconIds.put(STARTUP_STAGE_EXCHANGE_RATE_LOADED, R.id.startup_stage_icon_exchange_rate); + startupStageIconIds.put(STARTUP_STAGE_USER_AUTHENTICATED, R.id.startup_stage_icon_user_authenticated); + startupStageIconIds.put(STARTUP_STAGE_NEW_INSTALL_DONE, R.id.startup_stage_icon_install_new); + startupStageIconIds.put(STARTUP_STAGE_SUBSCRIPTIONS_LOADED, R.id.startup_stage_icon_subscriptions_loaded); + startupStageIconIds.put(STARTUP_STAGE_SUBSCRIPTIONS_RESOLVED, R.id.startup_stage_icon_subscriptions_resolved); + + for (Integer key : startupStages.keySet()) { + boolean stageDone = startupStages.get(key); + ImageView icon = findViewById(startupStageIconIds.get(key)); + icon.setImageResource(stageDone ? R.drawable.ic_check : R.drawable.ic_close); + icon.setColorFilter(stageDone ? Color.WHITE : Color.RED); + } + + findViewById(R.id.splash_view_loading_container).setVisibility(View.GONE); + findViewById(R.id.splash_view_error_container).setVisibility(View.VISIBLE); + } + + private void startup() { + final Context context = this; + Lbry.startupInit(); + + // perform some tasks before launching + (new AsyncTask() { + private Map startupStages = new HashMap<>(); + + private void initStartupStages() { + startupStages.put(STARTUP_STAGE_INSTALL_ID_LOADED, false); + startupStages.put(STARTUP_STAGE_KNOWN_TAGS_LOADED, false); + startupStages.put(STARTUP_STAGE_EXCHANGE_RATE_LOADED, false); + startupStages.put(STARTUP_STAGE_USER_AUTHENTICATED, false); + startupStages.put(STARTUP_STAGE_NEW_INSTALL_DONE, false); + startupStages.put(STARTUP_STAGE_SUBSCRIPTIONS_LOADED, false); + startupStages.put(STARTUP_STAGE_SUBSCRIPTIONS_RESOLVED, false); + } + protected void onPreExecute() { + hideActionBar(); + lockDrawer(); + findViewById(R.id.splash_view).setVisibility(View.VISIBLE); + LbryAnalytics.setCurrentScreen(MainActivity.this, "Splash", "Splash"); + initStartupStages(); + } + 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?) + startupStages.put(STARTUP_STAGE_INSTALL_ID_LOADED, false); + return false; + } + + Lbry.INSTALLATION_ID = installId; + startupStages.put(STARTUP_STAGE_INSTALL_ID_LOADED, true); + + SQLiteDatabase db = dbHelper.getReadableDatabase(); + List fetchedTags = DatabaseHelper.getTags(db); + Lbry.knownTags = Helper.mergeKnownTags(fetchedTags); + Collections.sort(Lbry.knownTags, new Tag()); + Lbry.followedTags = Helper.filterFollowedTags(Lbry.knownTags); + startupStages.put(STARTUP_STAGE_KNOWN_TAGS_LOADED, true); + + // load the exchange rate + Lbryio.loadExchangeRate(); + if (Lbryio.LBCUSDRate == 0) { + return false; + } + startupStages.put(STARTUP_STAGE_EXCHANGE_RATE_LOADED, true); + + Lbryio.authenticate(context); + if (Lbryio.currentUser == null) { + throw new Exception("Did not retrieve authenticated user."); + } + startupStages.put(STARTUP_STAGE_USER_AUTHENTICATED, true); + + Lbryio.newInstall(context); + startupStages.put(STARTUP_STAGE_NEW_INSTALL_DONE, true); + + // (light) fetch subscriptions + if (Lbryio.subscriptions.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.subscriptions = subscriptions; + startupStages.put(STARTUP_STAGE_SUBSCRIPTIONS_LOADED, true); + + // resolve subscriptions + if (subUrls.size() > 0 && Lbryio.cacheResolvedSubscriptions.size() != Lbryio.subscriptions.size()) { + List resolvedSubs = Lbry.resolve(subUrls, Lbry.LBRY_TV_CONNECTION_STRING); + Lbryio.cacheResolvedSubscriptions = resolvedSubs; + } + // if no exceptions occurred here, subscriptions have been loaded and resolved + startupStages.put(STARTUP_STAGE_SUBSCRIPTIONS_RESOLVED, true); + } else { + // user has not subscribed to anything + startupStages.put(STARTUP_STAGE_SUBSCRIPTIONS_LOADED, true); + startupStages.put(STARTUP_STAGE_SUBSCRIPTIONS_RESOLVED, true); + } + } else { + startupStages.put(STARTUP_STAGE_SUBSCRIPTIONS_LOADED, true); + startupStages.put(STARTUP_STAGE_SUBSCRIPTIONS_RESOLVED, true); + } + } 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) { + // show which startup stage failed + renderStartupFailed(startupStages); + appStarted = false; + return; + } + + findViewById(R.id.splash_view).setVisibility(View.GONE); + unlockDrawer(); + showActionBar(); + + if (navMenuAdapter != null) { + navMenuAdapter.setCurrentItem(NavMenuItem.ID_ITEM_FOLLOWING); + } + + loadLastFragment(); + showSignedInUser(); + fetchRewards(); + + checkUrlIntent(getIntent()); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_APP_LAUNCH); + appStarted = true; + } + }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void fetchRewards() { + FetchRewardsTask task = new FetchRewardsTask(null, new FetchRewardsTask.FetchRewardsHandler() { + @Override + public void onSuccess(List rewards) { + Lbryio.updateRewardsLists(rewards); + for (Fragment fragment : openNavFragments.values()) { + if (fragment instanceof RewardsFragment) { + ((RewardsFragment) fragment).updateUnclaimedRewardsValue(); + } + } + + if (Lbryio.totalUnclaimedRewardAmount > 0) { + showFloatingUnclaimedRewards(); + updateRewardsUsdVale(); + } + } + + @Override + public void onError(Exception error) { + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void updateRewardsUsdVale() { + if (Lbryio.totalUnclaimedRewardAmount > 0) { + double usdRewardAmount = Lbryio.totalUnclaimedRewardAmount * Lbryio.LBCUSDRate; + if (navMenuAdapter != null) { + navMenuAdapter.setExtraLabelForItem( + NavMenuItem.ID_ITEM_REWARDS, + Lbryio.LBCUSDRate > 0 ? String.format("$%s", Helper.SIMPLE_CURRENCY_FORMAT.format(usdRewardAmount)) : null + ); + } + } + } + + public void showFloatingUnclaimedRewards() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(this); + boolean notInterestedInRewards = sp.getBoolean(PREFERENCE_KEY_INTERNAL_REWARDS_NOT_INTERESTED, false); + if (notInterestedInRewards) { + return; + } + + ((TextView) findViewById(R.id.floating_reward_value)).setText(Helper.shortCurrencyFormat(Lbryio.totalUnclaimedRewardAmount)); + findViewById(R.id.floating_reward_container).setVisibility(View.VISIBLE); + } + + private void checkUrlIntent(Intent intent) { + if (intent != null) { + Uri data = intent.getData(); + if (data != null) { + String url = data.toString(); + // check special urls + if (url.startsWith("lbry://?")) { + String specialPath = url.substring(8); + if (specialRouteFragmentClassMap.containsKey(specialPath)) { + Class fragmentClass = specialRouteFragmentClassMap.get(specialPath); + if (fragmentClassNavIdMap.containsKey(fragmentClass)) { + Map params = new HashMap<>(); + String tag = intent.getStringExtra("tag"); + params.put("singleTag", tag); + + openFragment( + specialRouteFragmentClassMap.get(specialPath), + true, + fragmentClassNavIdMap.get(fragmentClass), + !Helper.isNullOrEmpty(tag) ? params : null + ); + } + } + + // unrecognised path will open the following by default + } else { + try { + LbryUri uri = LbryUri.parse(url); + if (uri.isChannel()) { + openChannelUrl(uri.toString()); + } else { + openFileUrl(uri.toString()); + } + } catch (LbryUriException ex) { + // pass + } + } + } + } + } + + 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() { + if (startingShareActivity) { + // share activity triggered this, so reset the flag at this point 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); + startingShareActivity = false; } }, 1000); + return; } + if (startingPermissionRequest) { + return; + } + enterPIPMode(); + } + + protected boolean enterPIPMode() { + if (enteringPIPMode) { + return true; + } + + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O && + appPlayer != null && + !startingFilePickerActivity && + !startingSignInFlowActivity) { + 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 checkNotificationOpenIntent(Intent intent) { @@ -200,71 +2402,15 @@ public class MainActivity extends FragmentActivity implements DefaultHardwareBac } 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); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_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 - } - } - }; - registerReceiver(downloadEventReceiver, intentFilter); - } private void registerServiceActionsReceiver() { IntentFilter intentFilter = new IntentFilter(); + intentFilter.addAction(DownloadManager.ACTION_DOWNLOAD_EVENT); intentFilter.addAction(LbrynetService.LBRY_SDK_SERVICE_STARTED); intentFilter.addAction(LbrynetService.ACTION_STOP_SERVICE); serviceActionsReceiver = new BroadcastReceiver() { @@ -284,12 +2430,30 @@ public class MainActivity extends FragmentActivity implements DefaultHardwareBac notificationManager.notify(1, svcNotification); } }, 1000); + } else if (DownloadManager.ACTION_DOWNLOAD_EVENT.equalsIgnoreCase(action)) { + String downloadAction = intent.getStringExtra("action"); + String uri = intent.getStringExtra("uri"); + String outpoint = intent.getStringExtra("outpoint"); + String fileInfoJson = intent.getStringExtra("file_info"); + double progress = intent.getDoubleExtra("progress", 0); + if (uri == null || outpoint == null || (fileInfoJson == null && !"abort".equals(downloadAction))) { + return; + } + + for (DownloadActionListener listener : downloadActionListeners) { + listener.onDownloadAction(downloadAction, uri, outpoint, fileInfoJson, progress); + } } } }; registerReceiver(serviceActionsReceiver, intentFilter); } + private void unregisterReceivers() { + Helper.unregisterReceiver(requestsReceiver, this); + Helper.unregisterReceiver(serviceActionsReceiver, this); + } + private Notification buildServiceNotification() { NotificationCompat.Builder builder = new NotificationCompat.Builder(this, LbrynetService.NOTIFICATION_CHANNEL_ID); Intent contextIntent = new Intent(this, MainActivity.class); @@ -312,556 +2476,82 @@ 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_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 void setNowPlayingClaim(Claim claim, String url) { + nowPlayingClaim = claim; + nowPlayingClaimUrl = url; + if (claim != null) { + ((TextView) findViewById(R.id.global_now_playing_title)).setText(nowPlayingClaim.getTitle()); + ((TextView) findViewById(R.id.global_now_playing_channel_title)).setText(nowPlayingClaim.getPublisherTitle()); } } - public static Activity getActivity() { - Activity activity = new Activity(); - activity = currentActivity; - return activity; - } - - @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 void clearNowPlayingClaim() { + nowPlayingClaim = null; + nowPlayingClaimUrl = null; + 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); + if (appPlayer != null) { + appPlayer.setPlayWhenReady(false); } } - @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 +2560,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("file_manager") && startupStatus.getBoolean("wallet"); } } catch (ConnectException ex) { // pass @@ -894,37 +2577,224 @@ 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); + clearWunderbarFocus(findViewById(R.id.wunderbar)); + } + + @Override + public void onPictureInPictureModeChanged(boolean isInPictureInPictureMode, Configuration newConfig) { + inPictureInPictureMode = isInPictureInPictureMode; + 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 (appPlayer != null && inPictureInPictureMode) { + 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, Map params) { + if (params != null && params.containsKey("url")) { + return String.format("%s-%d-%s", fragmentClass.getName(), navItemId, params.get("url").toString()); + } + + 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, params); + 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 void fetchOwnChannels() { + ClaimListTask task = new ClaimListTask(Claim.TYPE_CHANNEL, null, new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + Lbry.ownChannels = Helper.filterDeletedClaims(new ArrayList<>(claims)); + for (FetchChannelsListener listener : fetchChannelsListeners) { + listener.onChannelsFetched(claims); + } + } + + @Override + public void onError(Exception error) { + // pass + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void fetchOwnClaims() { + ClaimListTask task = new ClaimListTask(Arrays.asList(Claim.TYPE_STREAM, Claim.TYPE_REPOST), null, new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + Lbry.ownClaims = Helper.filterDeletedClaims(new ArrayList<>(claims)); + for (FetchClaimsListener listener : fetchClaimsListeners) { + listener.onClaimsFetched(claims); + } + } + + @Override + public void onError(Exception error) { } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + 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); + } + + public static void requestPermission(String permission, int requestCode, String rationale, Context context, boolean forceRequest) { + if (ContextCompat.checkSelfPermission(context, permission) != PackageManager.PERMISSION_GRANTED) { + if (!forceRequest && ActivityCompat.shouldShowRequestPermissionRationale((Activity) context, permission)) { + Toast.makeText(context, rationale, Toast.LENGTH_LONG).show(); + } else { + startingPermissionRequest = true; + ActivityCompat.requestPermissions((Activity) context, new String[] { permission }, requestCode); } } } - - 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; - } + + public static boolean hasPermission(String permission, Context context) { + return (ContextCompat.checkSelfPermission(context, permission) == PackageManager.PERMISSION_GRANTED); + } + + public interface BackPressInterceptor { + boolean onBackPressed(); } } 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..cf819cb6 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/VerificationActivity.java @@ -0,0 +1,272 @@ +package io.lbry.browser; + +import android.content.Intent; +import android.graphics.Color; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.View; + +import androidx.fragment.app.FragmentActivity; +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.lbryinc.FetchCurrentUserTask; +import io.lbry.browser.utils.LbryAnalytics; +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)); + + 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(); + } + }); + } + + public void onResume() { + super.onResume(); + LbryAnalytics.setCurrentScreen(this, "Verification", "Verification"); + checkFlow(); + } + + public void checkFlow() { + ViewPager2 viewPager = findViewById(R.id.verification_pager); + if (Lbryio.isSignedIn()) { + boolean flowHandled = false; + if (flow == VERIFICATION_FLOW_WALLET) { + viewPager.setCurrentItem(VerificationPagerAdapter.PAGE_VERIFICATION_WALLET, false); + flowHandled = true; + } else if (flow == VERIFICATION_FLOW_REWARDS) { + User user = Lbryio.currentUser; + if (!user.isIdentityVerified()) { + // phone number verification required + viewPager.setCurrentItem(VerificationPagerAdapter.PAGE_VERIFICATION_PHONE, false); + flowHandled = true; + } else if (!user.isRewardApproved()) { + // manual verification required + viewPager.setCurrentItem(VerificationPagerAdapter.PAGE_VERIFICATION_MANUAL, false); + flowHandled = true; + } + } + + if (!flowHandled) { + // user has already been verified and or reward approved + setResult(RESULT_CANCELED); + finish(); + return; + } + } + } + + public void showLoading() { + findViewById(R.id.verification_loading_progress).setVisibility(View.VISIBLE); + findViewById(R.id.verification_pager).setVisibility(View.INVISIBLE); + findViewById(R.id.verification_close_button).setVisibility(View.GONE); + } + + public void hideLoading() { + findViewById(R.id.verification_loading_progress).setVisibility(View.GONE); + findViewById(R.id.verification_pager).setVisibility(View.VISIBLE); + } + + @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); + + Bundle bundle = new Bundle(); + bundle.putString("email", email); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_EMAIL_ADDED, bundle); + } + 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)); + + Bundle bundle = new Bundle(); + bundle.putString("email", email); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_EMAIL_VERIFIED, bundle); + + 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 + showLoading(); + 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) { + showFetchUserError(error.getMessage()); + hideLoading(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } else { + // change pager view depending on flow + showLoading(); + FetchCurrentUserTask task = new FetchCurrentUserTask(new FetchCurrentUserTask.FetchUserTaskHandler() { + @Override + public void onSuccess(User user) { + hideLoading(); + findViewById(R.id.verification_close_button).setVisibility(View.VISIBLE); + + Lbryio.currentUser = user; + ViewPager2 viewPager = findViewById(R.id.verification_pager); + // for rewards, (show phone verification if not done, or manual verification if required) + if (flow == VERIFICATION_FLOW_REWARDS) { + if (!user.isIdentityVerified()) { + // phone number verification required + viewPager.setCurrentItem(VerificationPagerAdapter.PAGE_VERIFICATION_PHONE, false); + } else if (!user.isRewardApproved()) { + // manual verification required + viewPager.setCurrentItem(VerificationPagerAdapter.PAGE_VERIFICATION_MANUAL, false); + } else { + // fully verified + setResult(RESULT_OK); + finish(); + } + } else if (flow == VERIFICATION_FLOW_WALLET) { + // for wallet sync, if password unlock is required, show password entry page + viewPager.setCurrentItem(VerificationPagerAdapter.PAGE_VERIFICATION_WALLET, false); + } + } + @Override + public void onError(Exception error) { + showFetchUserError(error.getMessage()); + hideLoading(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + @Override + public void onPhoneAdded(String countryCode, String phoneNumber) { + + } + + @Override + public void onPhoneVerified() { + showLoading(); + FetchCurrentUserTask task = new FetchCurrentUserTask(new FetchCurrentUserTask.FetchUserTaskHandler() { + @Override + public void onSuccess(User user) { + Lbryio.currentUser = user; + if (user.isIdentityVerified() && user.isRewardApproved()) { + // verified for rewards + LbryAnalytics.logEvent(LbryAnalytics.EVENT_REWARD_ELIGIBILITY_COMPLETED); + + setResult(RESULT_OK); + finish(); + return; + } + + // show manual verification page if the user is still not reward approved + ViewPager2 viewPager = findViewById(R.id.verification_pager); + viewPager.setCurrentItem(VerificationPagerAdapter.PAGE_VERIFICATION_MANUAL, false); + hideLoading(); + } + + @Override + public void onError(Exception error) { + showFetchUserError(error.getMessage()); + hideLoading(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void showFetchUserError(String message) { + Snackbar.make(findViewById(R.id.verification_pager), message, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + + @Override + public void onManualVerifyContinue() { + setResult(RESULT_OK); + finish(); + } + + @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..a4b0873d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/ChannelFilterListAdapter.java @@ -0,0 +1,125 @@ +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 + @Setter + 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 View allView; + 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); + allView = v.findViewById(R.id.channel_filter_all_container); + } + } + + public int getItemCount() { + return items != null ? items.size() : 0; + } + + public boolean isClaimSelected(Claim claim) { + return claim.equals(selectedItem); + } + + public void clearClaims() { + items = new ArrayList<>(items.subList(0, 1)); + notifyDataSetChanged(); + } + + 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.INVISIBLE : View.VISIBLE); + vh.allView.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() != 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..ce8bbce3 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/ClaimListAdapter.java @@ -0,0 +1,496 @@ +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.ProgressBar; +import android.widget.TextView; +import android.widget.Toast; + +import androidx.recyclerview.widget.RecyclerView; + +import com.bumptech.glide.Glide; +import com.bumptech.glide.request.RequestOptions; +import com.google.android.material.snackbar.Snackbar; + +import java.math.BigDecimal; +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.R; +import io.lbry.browser.listener.SelectionModeListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.LbryFile; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; +import lombok.Getter; +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 Map quickClaimIdMap; + private Map quickClaimUrlMap; + private Map notFoundClaimIdMap; + private Map notFoundClaimUrlMap; + + @Setter + private boolean hideFee; + @Setter + private boolean canEnterSelectionMode; + private Context context; + private List items; + private List selectedItems; + @Setter + private ClaimListItemListener listener; + @Getter + @Setter + private boolean inSelectionMode; + @Setter + private SelectionModeListener selectionModeListener; + private float scale; + + public ClaimListAdapter(List items, Context context) { + this.context = context; + this.items = new ArrayList<>(items); + this.selectedItems = new ArrayList<>(); + quickClaimIdMap = new HashMap<>(); + quickClaimUrlMap = new HashMap<>(); + notFoundClaimIdMap = new HashMap<>(); + notFoundClaimUrlMap = new HashMap<>(); + if (context != null) { + scale = context.getResources().getDisplayMetrics().density; + } + } + + public List getSelectedItems() { + return this.selectedItems; + } + public int getSelectedCount() { + return selectedItems != null ? selectedItems.size() : 0; + } + 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 removeFeaturedItem() { + int featuredIndex = -1; + for (int i = 0; i < items.size(); i++) { + if (items.get(i).isFeatured()) { + featuredIndex = i; + break; + } + } + if (featuredIndex > -1) { + items.remove(featuredIndex); + } + } + + public List getItems() { + return new ArrayList<>(this.items); + } + + public void updateSigningChannelForClaim(Claim resolvedClaim) { + for (Claim claim : items) { + if (claim.getClaimId().equalsIgnoreCase(resolvedClaim.getClaimId())) { + claim.setSigningChannel(resolvedClaim.getSigningChannel()); + } + } + } + + public void clearItems() { + clearSelectedItems(); + this.items.clear(); + quickClaimIdMap.clear(); + quickClaimUrlMap.clear(); + notFoundClaimIdMap.clear(); + notFoundClaimUrlMap.clear(); + notifyDataSetChanged(); + } + + public Claim getLastItem() { + return items.size() > 0 ? items.get(items.size() - 1) : null; + } + + 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); + } + } + + notFoundClaimUrlMap.clear(); + notFoundClaimIdMap.clear(); + notifyDataSetChanged(); + } + public void setItems(List claims) { + items = new ArrayList<>(claims); + notifyDataSetChanged(); + } + + public void removeItems(List claims) { + items.removeAll(claims); + notifyDataSetChanged(); + } + + public void removeItem(Claim claim) { + items.remove(claim); + selectedItems.remove(claim); + notifyDataSetChanged(); + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected View feeContainer; + protected TextView feeView; + protected ImageView thumbnailView; + protected View noThumbnailView; + protected TextView alphaView; + protected TextView vanityUrlView; + protected TextView durationView; + protected TextView titleView; + protected TextView publisherView; + protected TextView publishTimeView; + protected TextView pendingTextView; + protected View repostInfoView; + protected TextView repostChannelView; + protected View selectedOverlayView; + protected TextView fileSizeView; + protected ProgressBar downloadProgressView; + protected TextView deviceView; + public ViewHolder(View v) { + super(v); + feeContainer = v.findViewById(R.id.claim_fee_container); + feeView = v.findViewById(R.id.claim_fee); + 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); + pendingTextView = v.findViewById(R.id.claim_pending_text); + repostInfoView = v.findViewById(R.id.claim_repost_info); + repostChannelView = v.findViewById(R.id.claim_repost_channel); + selectedOverlayView = v.findViewById(R.id.claim_selected_overlay); + fileSizeView = v.findViewById(R.id.claim_file_size); + downloadProgressView = v.findViewById(R.id.claim_download_progress); + deviceView = v.findViewById(R.id.claim_view_device); + } + } + + @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; + } + + Claim claim = items.get(position); + String valueType = items.get(position).getValueType(); + Claim actualClaim = Claim.TYPE_REPOST.equalsIgnoreCase(valueType) ? claim.getRepostedClaim() : claim; + + return Claim.TYPE_CHANNEL.equalsIgnoreCase(actualClaim.getValueType()) ? VIEW_TYPE_CHANNEL : VIEW_TYPE_STREAM; + } + + public void updateFileForClaimByIdOrUrl(LbryFile file, String claimId, String url) { + updateFileForClaimByIdOrUrl(file, claimId, url, false); + } + public void updateFileForClaimByIdOrUrl(LbryFile file, String claimId, String url, boolean skipNotFound) { + if (!skipNotFound) { + if (notFoundClaimIdMap.containsKey(claimId) && notFoundClaimUrlMap.containsKey(url)) { + return; + } + } + if (quickClaimIdMap.containsKey(claimId)) { + quickClaimIdMap.get(claimId).setFile(file); + notifyDataSetChanged(); + return; + } + if (quickClaimUrlMap.containsKey(claimId)) { + quickClaimUrlMap.get(claimId).setFile(file); + notifyDataSetChanged(); + return; + } + + boolean claimFound = false; + for (int i = 0; i < items.size(); i++) { + Claim claim = items.get(i); + if (claimId.equalsIgnoreCase(claim.getClaimId()) || url.equalsIgnoreCase(claim.getPermanentUrl())) { + quickClaimIdMap.put(claimId, claim); + quickClaimUrlMap.put(url, claim); + claim.setFile(file); + notifyDataSetChanged(); + claimFound = true; + break; + } + } + + if (!claimFound) { + notFoundClaimIdMap.put(claimId, true); + notFoundClaimUrlMap.put(url, true); + } + } + public void clearFileForClaimOrUrl(String outpoint, String url) { + clearFileForClaimOrUrl(outpoint, url, false); + notifyDataSetChanged(); + } + + + public void clearFileForClaimOrUrl(String outpoint, String url, boolean remove) { + int claimIndex = -1; + for (int i = 0; i < items.size(); i++) { + Claim claim = items.get(i); + if (outpoint.equalsIgnoreCase(claim.getOutpoint()) || url.equalsIgnoreCase(claim.getPermanentUrl())) { + claimIndex = i; + claim.setFile(null); + break; + } + } + if (remove && claimIndex > -1) { + Claim removed = items.remove(claimIndex); + selectedItems.remove(removed); + } + + notifyDataSetChanged(); + } + + @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); + } + + public int getScaledValue(int value) { + return (int) (value * scale + 0.5f); + } + + @Override + public void onBindViewHolder(ClaimListAdapter.ViewHolder vh, int position) { + int type = getItemViewType(position); + int paddingTop = position == 0 ? 16 : 8; + int paddingBottom = position == getItemCount() - 1 ? 16 : 8; + int paddingTopScaled = getScaledValue(paddingTop); + int paddingBottomScaled = getScaledValue(paddingBottom); + vh.itemView.setPadding(vh.itemView.getPaddingLeft(), paddingTopScaled, vh.itemView.getPaddingRight(), paddingBottomScaled); + + Claim original = items.get(position); + boolean isRepost = Claim.TYPE_REPOST.equalsIgnoreCase(original.getValueType()); + final Claim item = Claim.TYPE_REPOST.equalsIgnoreCase(original.getValueType()) ? original.getRepostedClaim() : original; + 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()); + } + + boolean isPending = item.getConfirmations() == 0; + boolean isSelected = isClaimSelected(item); + vh.itemView.setSelected(isSelected); + vh.selectedOverlayView.setVisibility(isSelected ? View.VISIBLE : View.GONE); + vh.itemView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (isPending) { + Snackbar.make(vh.itemView, R.string.item_pending_blockchain, Snackbar.LENGTH_LONG).show(); + return; + } + + if (inSelectionMode) { + toggleSelectedClaim(original); + } else { + if (listener != null) { + listener.onClaimClicked(item); + } + } + } + }); + vh.itemView.setOnLongClickListener(new View.OnLongClickListener() { + @Override + public boolean onLongClick(View view) { + if (!canEnterSelectionMode) { + return false; + } + + if (isPending) { + Snackbar.make(vh.itemView, R.string.item_pending_blockchain, Snackbar.LENGTH_LONG).show(); + return false; + } + + if (!inSelectionMode) { + inSelectionMode = true; + if (selectionModeListener != null) { + selectionModeListener.onEnterSelectionMode(); + } + } + toggleSelectedClaim(original); + return true; + } + }); + + vh.publisherView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (listener != null && signingChannel != null) { + listener.onClaimClicked(signingChannel); + } + } + }); + + vh.publishTimeView.setVisibility(!isPending ? View.VISIBLE : View.GONE); + vh.pendingTextView.setVisibility(isPending ? View.VISIBLE : View.GONE); + vh.repostInfoView.setVisibility(isRepost ? View.VISIBLE : View.GONE); + vh.repostChannelView.setText(isRepost ? original.getSigningChannel().getName() : null); + vh.repostChannelView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (listener != null) { + listener.onClaimClicked(original.getSigningChannel()); + } + } + }); + + 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.feeContainer.setVisibility(item.isUnresolved() || !Claim.TYPE_STREAM.equalsIgnoreCase(item.getValueType()) ? View.GONE : View.VISIBLE); + 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.thumbnailView.setVisibility(View.VISIBLE); + } else { + vh.thumbnailView.setVisibility(View.GONE); + } + + BigDecimal cost = item.getActualCost(Lbryio.LBCUSDRate); + vh.feeContainer.setVisibility(cost.doubleValue() > 0 && !hideFee ? View.VISIBLE : View.GONE); + vh.feeView.setText(cost.doubleValue() > 0 ? Helper.shortCurrencyFormat(cost.doubleValue()) : "Paid"); + 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)); + + LbryFile claimFile = item.getFile(); + boolean isDownloading = false; + int progress = 0; + String fileSizeString = claimFile == null ? null : Helper.formatBytes(claimFile.getTotalBytes(), false); + if (claimFile != null && + !Helper.isNullOrEmpty(claimFile.getDownloadPath()) && + !claimFile.isCompleted() && + claimFile.getWrittenBytes() < claimFile.getTotalBytes()) { + isDownloading = true; + progress = claimFile.getTotalBytes() > 0 ? + Double.valueOf(((double) claimFile.getWrittenBytes() / (double) claimFile.getTotalBytes()) * 100.0).intValue() : 0; + fileSizeString = String.format("%s / %s", + Helper.formatBytes(claimFile.getWrittenBytes(), false), + Helper.formatBytes(claimFile.getTotalBytes(), false)); + } + + Helper.setViewText(vh.fileSizeView, claimFile != null && !Helper.isNullOrEmpty(claimFile.getDownloadPath()) ? fileSizeString : null); + Helper.setViewVisibility(vh.downloadProgressView, isDownloading ? View.VISIBLE : View.INVISIBLE); + Helper.setViewProgress(vh.downloadProgressView, progress); + Helper.setViewText(vh.deviceView, item.getDevice()); + } 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)); + } + } + } + + private void toggleSelectedClaim(Claim claim) { + if (selectedItems.contains(claim)) { + selectedItems.remove(claim); + } else { + selectedItems.add(claim); + } + + if (selectionModeListener != null) { + selectionModeListener.onItemSelectionToggled(); + } + + if (selectedItems.size() == 0) { + inSelectionMode = false; + if (selectionModeListener != null) { + selectionModeListener.onExitSelectionMode(); + } + } + + notifyDataSetChanged(); + } + + public interface ClaimListItemListener { + void onClaimClicked(Claim claim); + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/EditorsChoiceItemAdapter.java b/app/src/main/java/io/lbry/browser/adapter/EditorsChoiceItemAdapter.java new file mode 100644 index 00000000..557751aa --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/EditorsChoiceItemAdapter.java @@ -0,0 +1,117 @@ +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 java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.EditorsChoiceItem; +import io.lbry.browser.utils.Helper; +import lombok.Setter; + +public class EditorsChoiceItemAdapter extends RecyclerView.Adapter { + private static final int VIEW_TYPE_HEADER = 1; + private static final int VIEW_TYPE_CONTENT = 2; + + private Context context; + private List items; + @Setter + private EditorsChoiceItemListener listener; + + public EditorsChoiceItemAdapter(List items, Context context) { + this.context = context; + this.items = new ArrayList<>(items); + } + + public void addFeaturedItem(EditorsChoiceItem item) { + items.add(0, item); + notifyDataSetChanged(); + } + + public void addItems(List items) { + for (EditorsChoiceItem item : items) { + if (!this.items.contains(item)) { + this.items.add(item); + } + } + notifyDataSetChanged(); + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected ImageView thumbnailView; + protected TextView descriptionView; + protected TextView headerView; + protected TextView titleView; + protected View cardView; + + public ViewHolder(View v) { + super(v); + + cardView = v.findViewById(R.id.editors_choice_content_card); + descriptionView = v.findViewById(R.id.editors_choice_content_description); + titleView = v.findViewById(R.id.editors_choice_content_title); + + thumbnailView = v.findViewById(R.id.editors_choice_content_thumbnail); + headerView = v.findViewById(R.id.editors_choice_header_title); + } + } + + @Override + public int getItemCount() { + return items != null ? items.size() : 0; + } + + @Override + public int getItemViewType(int position) { + return items.get(position).isHeader() ? VIEW_TYPE_HEADER : VIEW_TYPE_CONTENT; + } + + @Override + public EditorsChoiceItemAdapter.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { + View v = LayoutInflater.from(context).inflate(R.layout.list_item_editors_choice, parent, false); + return new EditorsChoiceItemAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(EditorsChoiceItemAdapter.ViewHolder vh, int position) { + int type = getItemViewType(position); + EditorsChoiceItem item = items.get(position); + + vh.headerView.setVisibility(type == VIEW_TYPE_HEADER ? View.VISIBLE : View.GONE); + vh.cardView.setVisibility(type == VIEW_TYPE_CONTENT ? View.VISIBLE : View.GONE); + + vh.headerView.setText(item.getTitle()); + vh.titleView.setText(item.getTitle()); + vh.descriptionView.setText(item.getDescription()); + if (!Helper.isNullOrEmpty(item.getThumbnailUrl())) { + Glide.with(context.getApplicationContext()). + load(item.getThumbnailUrl()). + centerCrop(). + placeholder(R.drawable.bg_thumbnail_placeholder). + into(vh.thumbnailView); + } + + vh.cardView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (listener != null) { + listener.onEditorsChoiceItemClicked(item); + } + } + }); + } + + public interface EditorsChoiceItemListener { + void onEditorsChoiceItemClicked(EditorsChoiceItem item); + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/GalleryGridAdapter.java b/app/src/main/java/io/lbry/browser/adapter/GalleryGridAdapter.java new file mode 100644 index 00000000..b9edfcc0 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/GalleryGridAdapter.java @@ -0,0 +1,119 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.graphics.Rect; +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 java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.GalleryItem; +import io.lbry.browser.utils.Helper; +import lombok.Setter; + +public class GalleryGridAdapter extends RecyclerView.Adapter { + private Context context; + private List items; + @Setter + private GalleryItemClickListener listener; + + public GalleryGridAdapter(List items, Context context) { + this.items = new ArrayList<>(items); + this.context = context; + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected ImageView thumbnailView; + protected TextView durationView; + public ViewHolder(View v) { + super(v); + thumbnailView = v.findViewById(R.id.gallery_item_thumbnail); + durationView = v.findViewById(R.id.gallery_item_duration); + } + } + + public int getItemCount() { + return items != null ? items.size() : 0; + } + + public void addItem(GalleryItem item) { + if (!items.contains(item)) { + items.add(item); + notifyDataSetChanged(); + } + } + + public void addItems(List items) { + for (GalleryItem item : items) { + if (!this.items.contains(item)) { + this.items.add(item); + notifyDataSetChanged(); + } + } + } + + public void clearItems() { + items.clear(); + notifyDataSetChanged(); + } + + @Override + public GalleryGridAdapter.ViewHolder onCreateViewHolder(ViewGroup root, int viewType) { + View v = LayoutInflater.from(context).inflate(R.layout.list_item_gallery, root, false); + return new GalleryGridAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(GalleryGridAdapter.ViewHolder vh, int position) { + GalleryItem item = items.get(position); + String thumbnailUrl = item.getThumbnailPath(); + Glide.with(context.getApplicationContext()).load(thumbnailUrl).centerCrop().into(vh.thumbnailView); + vh.durationView.setVisibility(item.getDuration() > 0 ? View.VISIBLE : View.INVISIBLE); + vh.durationView.setText(item.getDuration() > 0 ? Helper.formatDuration(Double.valueOf(item.getDuration() / 1000.0).longValue()) : null); + + vh.itemView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (listener != null) { + listener.onGalleryItemClicked(item); + } + } + }); + } + + public interface GalleryItemClickListener { + void onGalleryItemClicked(GalleryItem item); + } + + public static class GalleryGridItemDecoration extends RecyclerView.ItemDecoration { + + private int spanCount; + private int spacing; + + public GalleryGridItemDecoration(int spanCount, int spacing) { + this.spanCount = spanCount; + this.spacing = spacing; + } + + @Override + public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) { + int position = parent.getChildAdapterPosition(view); // item position + int column = position % spanCount; // item column + + outRect.left = column * spacing / spanCount; // column * ((1f / spanCount) * spacing) + outRect.right = spacing - (column + 1) * spacing / spanCount; // spacing - (column + 1) * ((1f / spanCount) * spacing) + if (position >= spanCount) { + outRect.top = spacing; // item top + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/InlineChannelSpinnerAdapter.java b/app/src/main/java/io/lbry/browser/adapter/InlineChannelSpinnerAdapter.java new file mode 100644 index 00000000..9924a67a --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/InlineChannelSpinnerAdapter.java @@ -0,0 +1,80 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.database.DataSetObserver; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ArrayAdapter; +import android.widget.SpinnerAdapter; +import android.widget.TextView; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.Claim; + +public class InlineChannelSpinnerAdapter extends ArrayAdapter { + + private List channels; + private int layoutResourceId; + private LayoutInflater inflater; + + public InlineChannelSpinnerAdapter(Context context, int resource, List channels) { + super(context, resource, 0, channels); + inflater = LayoutInflater.from(context); + layoutResourceId = resource; + this.channels = new ArrayList<>(channels); + } + public void addPlaceholder(boolean includeAnonymous) { + Claim placeholder = new Claim(); + placeholder.setPlaceholder(true); + insert(placeholder, 0); + channels.add(0, placeholder); + + if (includeAnonymous) { + Claim anonymous = new Claim(); + anonymous.setPlaceholderAnonymous(true); + insert(anonymous, 1); + channels.add(1, anonymous); + } + } + + public int getItemPosition(Claim item) { + for (int i = 0; i < channels.size(); i++) { + Claim channel = channels.get(i); + if (item.getClaimId().equalsIgnoreCase(channel.getClaimId())) { + return i; + } + } + return -1; + } + + @Override + public View getDropDownView(int position, View view, ViewGroup parent) { + return createView(position, view, parent); + } + + @Override + public View getView(int position, View view, ViewGroup parent) { + return createView(position, view, parent); + } + private View createView(int position, View convertView, ViewGroup parent){ + View view = inflater.inflate(layoutResourceId, parent, false); + + Context context = getContext(); + Claim channel = getItem(position); + String name = channel.getName(); + if (channel.isPlaceholder()) { + name = context.getString(R.string.create_a_channel); + } else if (channel.isPlaceholderAnonymous()) { + name = context.getString(R.string.anonymous); + } + + TextView label = view.findViewById(R.id.channel_item_name); + label.setText(name); + + return view; + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/InviteeListAdapter.java b/app/src/main/java/io/lbry/browser/adapter/InviteeListAdapter.java new file mode 100644 index 00000000..7ff066b0 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/InviteeListAdapter.java @@ -0,0 +1,92 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.content.Intent; +import android.graphics.Typeface; +import android.net.Uri; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.TextView; + +import androidx.recyclerview.widget.RecyclerView; + +import java.text.DecimalFormat; +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.lbryinc.Invitee; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryUri; +import lombok.Setter; + +public class InviteeListAdapter extends RecyclerView.Adapter { + + private Context context; + private List items; + + public InviteeListAdapter(List invitees, Context context) { + this.context = context; + this.items = new ArrayList<>(invitees); + } + + public void clear() { + items.clear(); + notifyDataSetChanged(); + } + + public List getItems() { + return new ArrayList<>(items); + } + + public void addHeader() { + Invitee header = new Invitee(); + header.setHeader(true); + items.add(0, header); + } + + public void addInvitees(List Invitees) { + for (Invitee tx : Invitees) { + if (!items.contains(tx)) { + items.add(tx); + } + } + notifyDataSetChanged(); + } + + public int getItemCount() { + return items != null ? items.size() : 0; + } + + @Override + public InviteeListAdapter.ViewHolder onCreateViewHolder(ViewGroup root, int viewType) { + View v = LayoutInflater.from(context).inflate(R.layout.list_item_invitee, root, false); + return new InviteeListAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(InviteeListAdapter.ViewHolder vh, int position) { + Invitee item = items.get(position); + vh.emailView.setText(item.isHeader() ? context.getString(R.string.email) : item.getEmail()); + vh.emailView.setTypeface(null, item.isHeader() ? Typeface.BOLD : Typeface.NORMAL); + + String rewardText = context.getString( + item.isInviteRewardClaimed() ? R.string.claimed : + (item.isInviteRewardClaimable() ? R.string.claimable : R.string.unclaimable)); + vh.rewardView.setText(item.isHeader() ? context.getString(R.string.reward) : rewardText); + vh.rewardView.setTypeface(null, item.isHeader() ? Typeface.BOLD : Typeface.NORMAL); + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected TextView emailView; + protected TextView rewardView; + + public ViewHolder(View v) { + super(v); + emailView = v.findViewById(R.id.invitee_email); + rewardView = v.findViewById(R.id.invitee_reward); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/adapter/LanguageSpinnerAdapter.java b/app/src/main/java/io/lbry/browser/adapter/LanguageSpinnerAdapter.java new file mode 100644 index 00000000..1e5c9a44 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/LanguageSpinnerAdapter.java @@ -0,0 +1,53 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ArrayAdapter; +import android.widget.TextView; + +import androidx.annotation.NonNull; + +import io.lbry.browser.R; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.Language; +import io.lbry.browser.utils.Predefined; + +public class LanguageSpinnerAdapter extends ArrayAdapter { + private int layoutResourceId; + private LayoutInflater inflater; + + public LanguageSpinnerAdapter(Context context, int resource) { + super(context, resource, 0, Predefined.PUBLISH_LANGUAGES); + inflater = LayoutInflater.from(context); + layoutResourceId = resource; + } + + public int getItemPosition(String languageCode) { + for (int i = 0; i < Predefined.PUBLISH_LANGUAGES.size(); i++) { + Language lang = Predefined.PUBLISH_LANGUAGES.get(i); + if (lang.getCode().equalsIgnoreCase(languageCode)) { + return i; + } + } + return -1; + } + + @Override + public View getDropDownView(int position, View view, @NonNull ViewGroup parent) { + return createView(position, view, parent); + } + @Override + public View getView(int position, View view, @NonNull ViewGroup parent) { + return createView(position, view, parent); + } + private View createView(int position, View convertView, ViewGroup parent) { + Language item = getItem(position); + View view = inflater.inflate(layoutResourceId, parent, false); + TextView label = view.findViewById(R.id.item_display_name); + label.setText(item != null ? item.getStringResourceId() : 0); + + return view; + } +} \ No newline at end of file diff --git a/app/src/main/java/io/lbry/browser/adapter/LicenseSpinnerAdapter.java b/app/src/main/java/io/lbry/browser/adapter/LicenseSpinnerAdapter.java new file mode 100644 index 00000000..c6776cf9 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/LicenseSpinnerAdapter.java @@ -0,0 +1,52 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ArrayAdapter; +import android.widget.TextView; + +import androidx.annotation.NonNull; + +import io.lbry.browser.R; +import io.lbry.browser.model.Language; +import io.lbry.browser.model.License; +import io.lbry.browser.utils.Predefined; + +public class LicenseSpinnerAdapter extends ArrayAdapter { + private int layoutResourceId; + private LayoutInflater inflater; + + public LicenseSpinnerAdapter(Context context, int resource) { + super(context, resource, 0, Predefined.LICENSES); + inflater = LayoutInflater.from(context); + layoutResourceId = resource; + } + public int getItemPosition(String name) { + for (int i = 0; i < Predefined.LICENSES.size(); i++) { + License lic = Predefined.LICENSES.get(i); + if (lic.getName().equalsIgnoreCase(name)) { + return i; + } + } + return -1; + } + + @Override + public View getDropDownView(int position, View view, @NonNull ViewGroup parent) { + return createView(position, view, parent); + } + @Override + public View getView(int position, View view, @NonNull ViewGroup parent) { + return createView(position, view, parent); + } + private View createView(int position, View convertView, ViewGroup parent) { + License item = getItem(position); + View view = inflater.inflate(layoutResourceId, parent, false); + TextView label = view.findViewById(R.id.item_display_name); + label.setText(item != null ? item.getStringResourceId() : 0); + + return view; + } +} \ No newline at end of file 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..f5c75a55 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/NavigationMenuAdapter.java @@ -0,0 +1,117 @@ +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 io.lbry.browser.utils.Helper; +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 setExtraLabelForItem(int id, String extraLabel) { + for (NavMenuItem item : menuItems) { + if (item.getId() == id) { + item.setExtraLabel(extraLabel); + 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); + String displayTitle = !Helper.isNullOrEmpty(item.getExtraLabel()) ? String.format("%s (%s)", item.getTitle(), item.getExtraLabel()) : item.getTitle(); + vh.titleView.setText(displayTitle); + 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/RewardListAdapter.java b/app/src/main/java/io/lbry/browser/adapter/RewardListAdapter.java new file mode 100644 index 00000000..f3895452 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/RewardListAdapter.java @@ -0,0 +1,216 @@ +package io.lbry.browser.adapter; + +import android.content.Context; +import android.content.Intent; +import android.graphics.Color; +import android.net.Uri; +import android.text.Editable; +import android.text.TextWatcher; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.EditText; +import android.widget.TextView; + +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.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.lbryinc.Reward; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbryio; +import lombok.Getter; +import lombok.Setter; + +public class RewardListAdapter extends RecyclerView.Adapter { + + public static final int DISPLAY_MODE_ALL = 1; + public static final int DISPLAY_MODE_UNCLAIMED = 2; + + private Context context; + @Setter + private List all; + private List items; + @Setter + private RewardClickListener clickListener; + @Getter + private int displayMode; + + public RewardListAdapter(List all, Context context) { + this.all = new ArrayList<>(all); + this.items = new ArrayList<>(all); + this.context = context; + this.displayMode = DISPLAY_MODE_ALL; + + addCustomReward(); + } + + public void setRewards(List rewards) { + this.all = new ArrayList<>(rewards); + updateItemsForDisplayMode(); + notifyDataSetChanged(); + } + + public void setDisplayMode(int displayMode) { + this.displayMode = displayMode; + updateItemsForDisplayMode(); + notifyDataSetChanged(); + } + + private void updateItemsForDisplayMode() { + if (displayMode == DISPLAY_MODE_ALL) { + items = new ArrayList<>(all); + } else if (displayMode == DISPLAY_MODE_UNCLAIMED) { + items = new ArrayList<>(); + for (Reward reward : all) { + if (!reward.isClaimed()) { + items.add(reward); + } + } + } + addCustomReward(); + } + + private void addCustomReward() { + Reward custom = new Reward(); + custom.setCustom(true); + custom.setRewardTitle(context.getString(R.string.custom_reward_title)); + custom.setRewardDescription(context.getString(R.string.custom_reward_description)); + items.add(custom); + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected View iconClaimed; + protected View loading; + protected View upTo; + protected TextView textTitle; + protected TextView textDescription; + protected TextView textLbcValue; + protected TextView textUsdValue; + protected TextView textLinkTransaction; + protected EditText inputCustomCode; + protected MaterialButton buttonClaimCustom; + public ViewHolder(View v) { + super(v); + iconClaimed = v.findViewById(R.id.reward_item_claimed_icon); + upTo = v.findViewById(R.id.reward_item_up_to); + loading = v.findViewById(R.id.reward_item_loading); + textTitle = v.findViewById(R.id.reward_item_title); + textDescription = v.findViewById(R.id.reward_item_description); + textLbcValue = v.findViewById(R.id.reward_item_lbc_value); + textLinkTransaction = v.findViewById(R.id.reward_item_tx_link); + textUsdValue = v.findViewById(R.id.reward_item_usd_value); + inputCustomCode = v.findViewById(R.id.reward_item_custom_code_input); + buttonClaimCustom = v.findViewById(R.id.reward_item_custom_claim_button); + } + } + + public int getItemCount() { + return items != null ? items.size() : 0; + } + + public void addReward(Reward reward) { + if (!items.contains(reward)) { + items.add(reward); + } + notifyDataSetChanged(); + } + public List getRewards() { + return new ArrayList<>(items); + } + + @Override + public RewardListAdapter.ViewHolder onCreateViewHolder(ViewGroup root, int viewType) { + View v = LayoutInflater.from(context).inflate(R.layout.list_item_reward, root, false); + return new RewardListAdapter.ViewHolder(v); + } + + @Override + public void onBindViewHolder(RewardListAdapter.ViewHolder vh, int position) { + Reward reward = items.get(position); + String displayAmount = reward.getDisplayAmount(); + double rewardAmount = 0; + if (!"?".equals(displayAmount)) { + rewardAmount = Double.valueOf(displayAmount); + } + boolean hasTransaction = !Helper.isNullOrEmpty(reward.getTransactionId()) && reward.getTransactionId().length() > 7; + vh.iconClaimed.setVisibility(reward.isClaimed() ? View.VISIBLE : View.INVISIBLE); + vh.inputCustomCode.setVisibility(reward.isCustom() ? View.VISIBLE : View.GONE); + vh.buttonClaimCustom.setVisibility(reward.isCustom() ? View.VISIBLE : View.GONE); + vh.textTitle.setText(reward.getRewardTitle()); + vh.textDescription.setText(reward.getRewardDescription()); + vh.upTo.setVisibility(reward.shouldDisplayRange() ? View.VISIBLE : View.GONE); + vh.textLbcValue.setText(reward.isCustom() ? "?" : Helper.LBC_CURRENCY_FORMAT.format(Helper.parseDouble(reward.getDisplayAmount(), 0))); + vh.textLinkTransaction.setVisibility(hasTransaction ? View.VISIBLE : View.GONE); + vh.textLinkTransaction.setText(hasTransaction ? reward.getTransactionId().substring(0, 7) : null); + vh.textLinkTransaction.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", Helper.EXPLORER_TX_PREFIX, reward.getTransactionId()))); + context.startActivity(intent); + } + } + }); + + vh.textUsdValue.setText(reward.isCustom() || Lbryio.LBCUSDRate == 0 ? null : + String.format("≈$%s", Helper.SIMPLE_CURRENCY_FORMAT.format(rewardAmount * Lbryio.LBCUSDRate))); + + vh.itemView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (reward.isClaimed()) { + return; + } + + if (clickListener != null) { + clickListener.onRewardClicked(reward, vh.loading); + } + } + }); + + vh.inputCustomCode.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) { + String value = charSequence.toString().trim(); + vh.buttonClaimCustom.setEnabled(value.length() > 0); + } + + @Override + public void afterTextChanged(Editable editable) { + + } + }); + + vh.buttonClaimCustom.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + String claimCode = Helper.getValue(vh.inputCustomCode.getText()); + if (Helper.isNullOrEmpty(claimCode)) { + Snackbar.make(view, R.string.please_enter_claim_code, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + return; + } + + if (clickListener != null) { + clickListener.onCustomClaimButtonClicked(claimCode, vh.inputCustomCode, vh.buttonClaimCustom, vh.loading); + } + } + }); + } + + public interface RewardClickListener { + void onRewardClicked(Reward reward, View loadingView); + void onCustomClaimButtonClicked(String code, EditText inputCustomCode, MaterialButton buttonClaim, View loadingView); + } +} 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..ff152d3f --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/SuggestedChannelGridAdapter.java @@ -0,0 +1,127 @@ +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 View noThumbnailView; + protected ImageView thumbnailView; + protected TextView alphaView; + protected TextView titleView; + protected TextView tagView; + public ViewHolder(View v) { + super(v); + noThumbnailView = v.findViewById(R.id.suggested_channel_no_thumbnail); + 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 void clearItems() { + items.clear(); + notifyDataSetChanged(); + } + + 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(); + + int bgColor = Helper.generateRandomColorForValue(claim.getClaimId()); + Helper.setIconViewBackgroundColor(vh.noThumbnailView, bgColor, false, context); + vh.noThumbnailView.setVisibility(Helper.isNullOrEmpty(thumbnailUrl) ? View.INVISIBLE : View.VISIBLE); + vh.alphaView.setText(claim.getName().substring(1, 2)); + if (!Helper.isNullOrEmpty(thumbnailUrl)) { + vh.thumbnailView.setVisibility(View.VISIBLE); + Glide.with(context.getApplicationContext()).load(thumbnailUrl).apply(RequestOptions.circleCropTransform()).into(vh.thumbnailView); + } else { + vh.thumbnailView.setVisibility(View.GONE); + } + 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..3ed25b93 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/TagListAdapter.java @@ -0,0 +1,108 @@ +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 java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.Tag; +import lombok.Getter; +import lombok.Setter; + +public class TagListAdapter extends RecyclerView.Adapter { + + public static final int CUSTOMIZE_MODE_NONE = 0; + public static final int CUSTOMIZE_MODE_ADD = 1; + public static final int CUSTOMIZE_MODE_REMOVE = 2; + + private Context context; + private List items; + @Setter + private TagClickListener clickListener; + @Getter + @Setter + private int customizeMode; + + public TagListAdapter(List tags, Context context) { + this.context = context; + this.items = new ArrayList<>(tags); + this.customizeMode = CUSTOMIZE_MODE_NONE; + } + + public static class ViewHolder extends RecyclerView.ViewHolder { + protected ImageView iconView; + protected TextView nameView; + public ViewHolder(View v) { + super(v); + iconView = v.findViewById(R.id.tag_action); + nameView = v.findViewById(R.id.tag_name); + } + } + + public int getItemCount() { + return items != null ? items.size() : 0; + } + + public void addTag(Tag tag) { + if (!items.contains(tag)) { + items.add(tag); + } + notifyDataSetChanged(); + } + public List getTags() { + return new ArrayList<>(items); + } + + public void setTags(List tags) { + items = new ArrayList<>(tags); + notifyDataSetChanged(); + } + + public void addTags(List tags) { + for (Tag tag : tags) { + if (!items.contains(tag)) { + items.add(tag); + } + } + notifyDataSetChanged(); + } + public void removeTag(Tag tag) { + items.remove(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.iconView.setVisibility(customizeMode == CUSTOMIZE_MODE_NONE ? View.GONE : View.VISIBLE); + vh.iconView.setImageResource(customizeMode == CUSTOMIZE_MODE_REMOVE ? R.drawable.ic_close : R.drawable.ic_add); + 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, customizeMode); + } + } + }); + } + + public interface TagClickListener { + void onTagClicked(Tag tag, int customizeMode); + } +} 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..95b4a139 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/TransactionListAdapter.java @@ -0,0 +1,132 @@ +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.TextView; + +import androidx.recyclerview.widget.RecyclerView; + +import java.text.DecimalFormat; +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.model.Transaction; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryUri; +import lombok.Setter; + +public class TransactionListAdapter extends RecyclerView.Adapter { + + 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, 7)); + 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.claimView.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + LbryUri claimUrl = item.getClaimUrl(); + if (claimUrl != null && listener != null) { + listener.onClaimUrlClicked(claimUrl); + } + } + }); + + 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", Helper.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); + void onClaimUrlClicked(LbryUri uri); + } +} 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..cc2a09ea --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/UrlSuggestionListAdapter.java @@ -0,0 +1,147 @@ +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.exceptions.LbryUriException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.ui.controls.SolidIconView; +import io.lbry.browser.utils.Helper; +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(); + try { + if (thisUrl != null) { + LbryUri vanity = LbryUri.parse(thisUrl.toVanityString()); + if (thisUrl.equals(url) || vanity.equals(url)) { + items.get(i).setClaim(claim); + } + } + } catch (LbryUriException ex) { + // pass + } + } + } + + 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() : + ((item.isUseTextAsDescription() && !Helper.isNullOrEmpty(item.getText())) ? item.getText() : 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() : + ((item.isUseTextAsDescription() && !Helper.isNullOrEmpty(item.getText())) ? item.getText() : 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..8569038c --- /dev/null +++ b/app/src/main/java/io/lbry/browser/adapter/VerificationPagerAdapter.java @@ -0,0 +1,71 @@ +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.ManualVerificationFragment; +import io.lbry.browser.ui.verification.PhoneVerificationFragment; +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 { + public static final int PAGE_VERIFICATION_EMAIL = 0; + public static final int PAGE_VERIFICATION_PHONE = 1; + public static final int PAGE_VERIFICATION_WALLET = 2; + public static final int PAGE_VERIFICATION_MANUAL = 3; + + 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: + PhoneVerificationFragment pvFragment = PhoneVerificationFragment.class.newInstance(); + if (activity instanceof SignInListener) { + pvFragment.setListener((SignInListener) activity); + } + return pvFragment; + case 2: + WalletVerificationFragment wvFragment = WalletVerificationFragment.class.newInstance(); + if (activity instanceof WalletSyncListener) { + wvFragment.setListener((WalletSyncListener) activity); + } + return wvFragment; + case 3: + ManualVerificationFragment mvFragment = ManualVerificationFragment.class.newInstance(); + if (activity instanceof SignInListener) { + mvFragment.setListener((SignInListener) activity); + } + return mvFragment; + } + } + + @Override + public int getItemCount() { + return 4; + } +} 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..2087a7ff --- /dev/null +++ b/app/src/main/java/io/lbry/browser/data/DatabaseHelper.java @@ -0,0 +1,250 @@ +package io.lbry.browser.data; + +import android.content.Context; +import android.database.Cursor; +import android.database.sqlite.SQLiteDatabase; +import android.database.sqlite.SQLiteOpenHelper; +import android.opengl.Visibility; + +import java.math.BigDecimal; +import java.text.ParseException; +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.Date; +import java.util.List; + +import io.lbry.browser.exceptions.LbryUriException; +import io.lbry.browser.model.Tag; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.model.ViewHistory; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryUri; + +public class DatabaseHelper extends SQLiteOpenHelper { + public static final int DATABASE_VERSION = 2; + public static final String DATABASE_NAME = "LbryApp.db"; + private static DatabaseHelper instance; + + private static final String[] SQL_CREATE_TABLES = { + // local subscription store + "CREATE TABLE subscriptions (url TEXT PRIMARY KEY NOT NULL, channel_name TEXT NOT NULL)", + // url entry / suggestion history + "CREATE TABLE url_history (id INTEGER PRIMARY KEY NOT NULL, value TEXT NOT NULL, url TEXT, type INTEGER NOT NULL, timestamp TEXT NOT NULL)", + // tags (known and followed) + "CREATE TABLE tags (id INTEGER PRIMARY KEY NOT NULL, name TEXT NOT NULL, is_followed INTEGER NOT NULL)", + // view history (stores only stream claims that have resolved) + "CREATE TABLE view_history (" + + " id INTEGER PRIMARY KEY NOT NULL" + + ", url TEXT NOT NULL" + + ", claim_id TEXT" + + ", claim_name TEXT" + + ", cost REAL " + + ", currency TEXT " + + ", title TEXT " + + ", publisher_claim_id TEXT" + + ", publisher_name TEXT" + + ", publisher_title TEXT" + + ", thumbnail_url TEXT" + + ", release_time INTEGER " + + ", device TEXT" + + ", timestamp TEXT NOT NULL)" + }; + private static final String[] SQL_CREATE_INDEXES = { + "CREATE UNIQUE INDEX idx_subscription_url ON subscriptions (url)", + "CREATE UNIQUE INDEX idx_url_history_value ON url_history (value)", + "CREATE UNIQUE INDEX idx_url_history_url ON url_history (url)", + "CREATE UNIQUE INDEX idx_tag_name ON tags (name)", + "CREATE UNIQUE INDEX idx_view_history_url_device ON view_history (url, device)", + "CREATE INDEX idx_view_history_device ON view_history (device)" + }; + + private static final String[] SQL_V1_V2_UPGRADE = { + "ALTER TABLE view_history ADD COLUMN currency TEXT" + }; + + 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"; + + private static final String SQL_INSERT_URL_HISTORY = "REPLACE INTO url_history (value, url, type, timestamp) VALUES (?, ?, ?, ?)"; + private static final String SQL_CLEAR_URL_HISTORY = "DELETE FROM url_history"; + private static final String SQL_CLEAR_URL_HISTORY_BEFORE_TIME = "DELETE FROM url_history WHERE timestamp < ?"; + private static final String SQL_GET_RECENT_URL_HISTORY = "SELECT value, url, type FROM url_history ORDER BY timestamp DESC LIMIT 10"; + + private static final String SQL_INSERT_VIEW_HISTORY = + "REPLACE INTO view_history (url, claim_id, claim_name, cost, currency, title, publisher_claim_id, publisher_name, publisher_title, thumbnail_url, device, release_time, timestamp) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)"; + private static final String SQL_GET_VIEW_HISTORY = + "SELECT url, claim_id, claim_name, cost, currency, title, publisher_claim_id, publisher_name, publisher_title, thumbnail_url, device, release_time, timestamp " + + "FROM view_history WHERE '' = ? OR timestamp < ? ORDER BY timestamp DESC LIMIT %d"; + private static final String SQL_CLEAR_VIEW_HISTORY = "DELETE FROM view_history"; + private static final String SQL_CLEAR_VIEW_HISTORY_BY_DEVICE = "DELETE FROM view_history WHERE device = ?"; + private static final String SQL_CLEAR_VIEW_HISTORY_BEFORE_TIME = "DELETE FROM view_history WHERE timestamp < ?"; + private static final String SQL_CLEAR_VIEW_HISTORY_BY_DEVICE_BEFORE_TIME = "DELETE FROM view_history WHERE device = ? AND timestamp < ?"; + + private static final String SQL_INSERT_TAG = "REPLACE INTO tags (name, is_followed) VALUES (?, ?)"; + private static final String SQL_GET_KNOWN_TAGS = "SELECT name, is_followed FROM tags"; + private static final String SQL_UNFOLLOW_TAGS = "UPDATE tags SET is_followed = 0"; + private static final String SQL_GET_FOLLOWED_TAGS = "SELECT name FROM tags WHERE is_followed = 1"; + + + + public DatabaseHelper(Context context) { + super(context, String.format("%s/%s", context.getFilesDir().getAbsolutePath(), DATABASE_NAME), null, DATABASE_VERSION); + instance = this; + } + public static DatabaseHelper getInstance() { + return instance; + } + 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) { + if (oldVersion < 2) { + for (String sql : SQL_V1_V2_UPGRADE) { + db.execSQL(sql); + } + } + } + public void onDowngrade(SQLiteDatabase db, int oldVersion, int newVersion) { + + } + + public static void createOrUpdateUrlHistoryItem(String text, String url, int type, SQLiteDatabase db) { + db.execSQL(SQL_INSERT_URL_HISTORY, new Object[] { + text, url, type, new SimpleDateFormat(Helper.ISO_DATE_FORMAT_PATTERN).format(new Date()) + }); + } + public static void clearUrlHistory(SQLiteDatabase db) { + db.execSQL(SQL_CLEAR_URL_HISTORY); + } + public static void clearUrlHistoryBefore(Date date, SQLiteDatabase db) { + db.execSQL(SQL_CLEAR_URL_HISTORY_BEFORE_TIME, new Object[] { new SimpleDateFormat(Helper.ISO_DATE_FORMAT_PATTERN).format(new Date()) }); + } + // History items are essentially url suggestions + public static List getRecentHistory(SQLiteDatabase db) { + List suggestions = new ArrayList<>(); + Cursor cursor = null; + try { + cursor = db.rawQuery(SQL_GET_RECENT_URL_HISTORY, null); + while (cursor.moveToNext()) { + UrlSuggestion suggestion = new UrlSuggestion(); + suggestion.setText(cursor.getString(0)); + suggestion.setUri(cursor.isNull(1) ? null : LbryUri.tryParse(cursor.getString(1))); + suggestion.setType(cursor.getInt(2)); + suggestion.setTitleUrlOnly(true); + suggestions.add(suggestion); + } + } finally { + Helper.closeCursor(cursor); + } + return suggestions; + } + + // View history items are stream claims + public static void createOrUpdateViewHistoryItem(ViewHistory viewHistory, SQLiteDatabase db) { + db.execSQL(SQL_INSERT_VIEW_HISTORY, new Object[] { + viewHistory.getUri().toString(), + viewHistory.getClaimId(), + viewHistory.getClaimName(), + viewHistory.getCost() != null ? viewHistory.getCost().doubleValue() : 0, + viewHistory.getCurrency(), + viewHistory.getTitle(), + viewHistory.getPublisherClaimId(), + viewHistory.getPublisherName(), + viewHistory.getPublisherTitle(), + viewHistory.getThumbnailUrl(), + viewHistory.getDevice(), + viewHistory.getReleaseTime(), + new SimpleDateFormat(Helper.ISO_DATE_FORMAT_PATTERN).format(new Date()) + }); + } + + public static List getViewHistory(String lastTimestamp, int pageLimit, SQLiteDatabase db) { + List history = new ArrayList<>(); + Cursor cursor = null; + try { + String arg = lastTimestamp == null ? "" : lastTimestamp; + cursor = db.rawQuery(String.format(SQL_GET_VIEW_HISTORY, pageLimit), new String[] { arg, arg }); + while (cursor.moveToNext()) { + ViewHistory item = new ViewHistory(); + int cursorIndex = 0; + item.setUri(LbryUri.tryParse(cursor.getString(cursorIndex++))); + item.setClaimId(cursor.getString(cursorIndex++)); + item.setClaimName(cursor.getString(cursorIndex++)); + item.setCost(new BigDecimal(cursor.getDouble(cursorIndex++))); + item.setCurrency(cursor.getString(cursorIndex++)); + item.setTitle(cursor.getString(cursorIndex++)); + item.setPublisherClaimId(cursor.getString(cursorIndex++)); + item.setPublisherName(cursor.getString(cursorIndex++)); + item.setPublisherTitle(cursor.getString(cursorIndex++)); + item.setThumbnailUrl(cursor.getString(cursorIndex++)); + item.setDevice(cursor.getString(cursorIndex++)); + item.setReleaseTime(cursor.getLong(cursorIndex++)); + try { + item.setTimestamp(new SimpleDateFormat(Helper.ISO_DATE_FORMAT_PATTERN).parse(cursor.getString(cursorIndex))); + } catch (ParseException ex) { + // invalid timestamp (which shouldn't happen). Skip this item + continue; + } + + history.add(item); + } + } finally { + Helper.closeCursor(cursor); + } + return history; + } + + public static void createOrUpdateTag(Tag tag, SQLiteDatabase db) { + db.execSQL(SQL_INSERT_TAG, new Object[] { tag.getLowercaseName(), tag.isFollowed() ? 1 : 0 }); + } + public static void setAllTagsUnfollowed(SQLiteDatabase db) { + db.execSQL(SQL_UNFOLLOW_TAGS); + } + public static List getTags(SQLiteDatabase db) { + List tags = new ArrayList<>(); + Cursor cursor = null; + try { + cursor = db.rawQuery(SQL_GET_KNOWN_TAGS, null); + while (cursor.moveToNext()) { + Tag tag = new Tag(); + tag.setName(cursor.getString(0)); + tag.setFollowed(cursor.getInt(1) == 1); + tags.add(tag); + } + } finally { + Helper.closeCursor(cursor); + } + return tags; + } + + 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/ContentScopeDialogFragment.java b/app/src/main/java/io/lbry/browser/dialog/ContentScopeDialogFragment.java new file mode 100644 index 00000000..4db24f8d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/dialog/ContentScopeDialogFragment.java @@ -0,0 +1,96 @@ +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 ContentScopeDialogFragment extends BottomSheetDialogFragment { + public static final String TAG = "ContentScopeDialog"; + public static final int ITEM_EVERYONE = 1; + public static final int ITEM_TAGS = 2; + + @Setter + private ContentScopeListener contentScopeListener; + private int currentScopeItem; + + public static ContentScopeDialogFragment newInstance() { + return new ContentScopeDialogFragment(); + } + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View view = inflater.inflate(R.layout.dialog_content_scope, container,false); + + ContentScopeItemClickListener clickListener = new ContentScopeItemClickListener(this, contentScopeListener); + view.findViewById(R.id.content_scope_everyone_item).setOnClickListener(clickListener); + view.findViewById(R.id.content_scope_tags_item).setOnClickListener(clickListener); + checkSelectedScopeItem(currentScopeItem, view); + + return view; + } + + public static void checkSelectedScopeItem(int scope, View parent) { + int checkViewId = -1; + switch (scope) { + case ITEM_EVERYONE: checkViewId = R.id.content_scope_everyone_item_selected; break; + case ITEM_TAGS: checkViewId = R.id.content_scope_tags_item_selected; break; + } + if (parent != null && checkViewId > -1) { + parent.findViewById(checkViewId).setVisibility(View.VISIBLE); + } + } + + public void setCurrentScopeItem(int scopeItem) { + this.currentScopeItem = scopeItem; + } + + private static class ContentScopeItemClickListener implements View.OnClickListener { + + private final int[] checkViewIds = { + R.id.content_scope_everyone_item_selected, R.id.content_scope_tags_item_selected + }; + private BottomSheetDialogFragment dialog; + private ContentScopeListener listener; + + public ContentScopeItemClickListener(BottomSheetDialogFragment dialog, ContentScopeListener listener) { + this.dialog = dialog; + this.listener = listener; + } + + public void onClick(View view) { + int scopeItem = -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_scope_everyone_item: scopeItem = ITEM_EVERYONE; break; + case R.id.content_scope_tags_item: scopeItem = ITEM_TAGS; break; + } + + checkSelectedScopeItem(scopeItem, view); + if (listener != null) { + listener.onContentScopeItemSelected(scopeItem); + } + + if (dialog != null) { + dialog.dismiss(); + } + } + } + + public interface ContentScopeListener { + void onContentScopeItemSelected(int scopeItem); + } +} 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/dialog/CustomizeTagsDialogFragment.java b/app/src/main/java/io/lbry/browser/dialog/CustomizeTagsDialogFragment.java new file mode 100644 index 00000000..1a328108 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/dialog/CustomizeTagsDialogFragment.java @@ -0,0 +1,186 @@ +package io.lbry.browser.dialog; + +import android.os.AsyncTask; +import android.os.Bundle; +import android.text.Editable; +import android.text.TextWatcher; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; + +import androidx.recyclerview.widget.RecyclerView; + +import com.google.android.flexbox.FlexboxLayoutManager; +import com.google.android.material.bottomsheet.BottomSheetDialogFragment; +import com.google.android.material.button.MaterialButton; +import com.google.android.material.snackbar.Snackbar; +import com.google.android.material.textfield.TextInputEditText; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.R; +import io.lbry.browser.adapter.TagListAdapter; +import io.lbry.browser.listener.TagListener; +import io.lbry.browser.model.Tag; +import io.lbry.browser.tasks.UpdateSuggestedTagsTask; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import lombok.Setter; + +public class CustomizeTagsDialogFragment extends BottomSheetDialogFragment { + public static final String TAG = "CustomizeTagsDialog"; + private static final int SUGGESTED_LIMIT = 8; + private String currentFilter; + + private RecyclerView followedTagsList; + private RecyclerView suggestedTagsList; + private TagListAdapter followedTagsAdapter; + private TagListAdapter suggestedTagsAdapter; + private View noTagsView; + private View noResultsView; + @Setter + private TagListener listener; + + private void checkNoTags() { + Helper.setViewVisibility(noTagsView, followedTagsAdapter == null || followedTagsAdapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + private void checkNoResults() { + Helper.setViewVisibility(noResultsView, suggestedTagsAdapter == null || suggestedTagsAdapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + public void addTag(Tag tag) { + if (followedTagsAdapter.getTags().contains(tag)) { + Snackbar.make(getView(), getString(R.string.tag_already_added, tag.getName()), Snackbar.LENGTH_LONG).show(); + return; + } + + tag.setFollowed(true); + followedTagsAdapter.addTag(tag); + if (suggestedTagsAdapter != null) { + suggestedTagsAdapter.removeTag(tag); + } + updateKnownTags(currentFilter, SUGGESTED_LIMIT, false); + if (listener != null) { + listener.onTagAdded(tag); + } + checkNoTags(); + checkNoResults(); + } + public void removeTag(Tag tag) { + tag.setFollowed(false); + followedTagsAdapter.removeTag(tag); + updateKnownTags(currentFilter, SUGGESTED_LIMIT, false); + if (listener != null) { + listener.onTagRemoved(tag); + } + checkNoTags(); + checkNoResults(); + } + + public void setFilter(String filter) { + currentFilter = filter; + updateKnownTags(currentFilter, SUGGESTED_LIMIT, true); + } + + public static CustomizeTagsDialogFragment newInstance() { + return new CustomizeTagsDialogFragment(); + } + + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View view = inflater.inflate(R.layout.dialog_customize_tags, container, false); + + noResultsView = view.findViewById(R.id.customize_no_tag_results); + noTagsView = view.findViewById(R.id.customize_no_followed_tags); + + followedTagsAdapter = new TagListAdapter(Lbry.followedTags, getContext()); + followedTagsAdapter.setCustomizeMode(TagListAdapter.CUSTOMIZE_MODE_REMOVE); + followedTagsAdapter.setClickListener(customizeTagClickListener); + suggestedTagsAdapter = new TagListAdapter(new ArrayList<>(), getContext()); + suggestedTagsAdapter.setCustomizeMode(TagListAdapter.CUSTOMIZE_MODE_ADD); + suggestedTagsAdapter.setClickListener(customizeTagClickListener); + + FlexboxLayoutManager flm1 = new FlexboxLayoutManager(getContext()); + followedTagsList = view.findViewById(R.id.customize_tags_followed_list); + followedTagsList.setLayoutManager(flm1); + followedTagsList.setAdapter(followedTagsAdapter); + + FlexboxLayoutManager flm2 = new FlexboxLayoutManager(getContext()); + suggestedTagsList = view.findViewById(R.id.customize_tags_suggested_list); + suggestedTagsList.setLayoutManager(flm2); + suggestedTagsList.setAdapter(suggestedTagsAdapter); + + TextInputEditText filterInput = view.findViewById(R.id.customize_tag_filter_input); + filterInput.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) { + String value = Helper.getValue(charSequence); + setFilter(value); + } + + @Override + public void afterTextChanged(Editable editable) { + + } + }); + + + MaterialButton doneButton = view.findViewById(R.id.customize_done_button); + doneButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + dismiss(); + } + }); + + checkNoTags(); + return view; + } + + private TagListAdapter.TagClickListener customizeTagClickListener = new TagListAdapter.TagClickListener() { + @Override + public void onTagClicked(Tag tag, int customizeMode) { + if (customizeMode == TagListAdapter.CUSTOMIZE_MODE_ADD) { + addTag(tag); + } else if (customizeMode == TagListAdapter.CUSTOMIZE_MODE_REMOVE) { + removeTag(tag); + } + } + }; + + public void onResume() { + super.onResume(); + updateKnownTags(null, SUGGESTED_LIMIT, true); + } + + private void updateKnownTags(String filter, int limit, boolean clearPrevious) { + UpdateSuggestedTagsTask task = new UpdateSuggestedTagsTask( + filter, + SUGGESTED_LIMIT, + followedTagsAdapter, + suggestedTagsAdapter, + clearPrevious, + false, new UpdateSuggestedTagsTask.KnownTagsHandler() { + @Override + public void onSuccess(List tags) { + if (suggestedTagsAdapter == null) { + suggestedTagsAdapter = new TagListAdapter(tags, getContext()); + suggestedTagsAdapter.setCustomizeMode(TagListAdapter.CUSTOMIZE_MODE_ADD); + suggestedTagsAdapter.setClickListener(customizeTagClickListener); + if (suggestedTagsList != null) { + suggestedTagsList.setAdapter(suggestedTagsAdapter); + } + } else { + suggestedTagsAdapter.setTags(tags); + } + checkNoResults(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } +} diff --git a/app/src/main/java/io/lbry/browser/dialog/DiscoverDialogFragment.java b/app/src/main/java/io/lbry/browser/dialog/DiscoverDialogFragment.java new file mode 100644 index 00000000..927ebda1 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/dialog/DiscoverDialogFragment.java @@ -0,0 +1,107 @@ +package io.lbry.browser.dialog; + +import android.content.DialogInterface; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ProgressBar; + +import androidx.annotation.NonNull; +import androidx.recyclerview.widget.GridLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import com.google.android.material.bottomsheet.BottomSheetDialogFragment; +import com.google.android.material.button.MaterialButton; + +import io.lbry.browser.R; +import io.lbry.browser.adapter.SuggestedChannelGridAdapter; +import lombok.Getter; +import lombok.Setter; + +public class DiscoverDialogFragment extends BottomSheetDialogFragment { + public static final String TAG = "DiscoverDialog"; + + @Getter + private SuggestedChannelGridAdapter adapter; + @Setter + private DiscoverDialogListener dialogActionsListener; + + public static DiscoverDialogFragment newInstance() { + return new DiscoverDialogFragment(); + } + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View view = inflater.inflate(R.layout.dialog_discover, container, false); + + RecyclerView grid = view.findViewById(R.id.discover_channel_grid); + GridLayoutManager glm = new GridLayoutManager(getContext(), 3); + grid.setLayoutManager(glm); + grid.setAdapter(adapter); + grid.addOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { + GridLayoutManager lm = (GridLayoutManager) recyclerView.getLayoutManager(); + if (lm != null) { + int visibleItemCount = lm.getChildCount(); + int totalItemCount = lm.getItemCount(); + int pastVisibleItems = lm.findFirstVisibleItemPosition(); + if (pastVisibleItems + visibleItemCount >= totalItemCount) { + if (dialogActionsListener != null) { + dialogActionsListener.onScrollEndReached(); + } + } + } + } + }); + + MaterialButton doneButton = view.findViewById(R.id.discover_done_button); + doneButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + dismiss(); + } + }); + + return view; + } + public void setAdapter(SuggestedChannelGridAdapter adapter) { + this.adapter = adapter; + if (getView() != null) { + ((RecyclerView) getView().findViewById(R.id.discover_channel_grid)).setAdapter(adapter); + } + } + public void setLoading(boolean loading) { + if (getView() != null) { + getView().findViewById(R.id.discover_loading).setVisibility(loading ? View.VISIBLE : View.GONE); + } + } + + @Override + public void onResume() { + super.onResume(); + if (dialogActionsListener != null) { + dialogActionsListener.onResume(); + } + } + @Override + public void onCancel(DialogInterface dialog) { + super.onDismiss(dialog); + if (dialogActionsListener != null) { + dialogActionsListener.onCancel(); + } + } + @Override + public void onDismiss(DialogInterface dialog) { + super.onDismiss(dialog); + if (dialogActionsListener != null) { + dialogActionsListener.onCancel(); + } + } + + public interface DiscoverDialogListener { + void onResume(); + void onCancel(); + void onScrollEndReached(); + } +} diff --git a/app/src/main/java/io/lbry/browser/dialog/RepostClaimDialogFragment.java b/app/src/main/java/io/lbry/browser/dialog/RepostClaimDialogFragment.java new file mode 100644 index 00000000..f34099ce --- /dev/null +++ b/app/src/main/java/io/lbry/browser/dialog/RepostClaimDialogFragment.java @@ -0,0 +1,291 @@ +package io.lbry.browser.dialog; + +import android.app.Dialog; +import android.content.Context; +import android.graphics.Color; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.AdapterView; +import android.widget.EditText; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.appcompat.widget.AppCompatSpinner; + +import com.google.android.material.bottomsheet.BottomSheetDialogFragment; +import com.google.android.material.button.MaterialButton; +import com.google.android.material.snackbar.Snackbar; +import com.google.android.material.textfield.TextInputEditText; + +import java.math.BigDecimal; +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.InlineChannelSpinnerAdapter; +import io.lbry.browser.listener.WalletBalanceListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ClaimListTask; +import io.lbry.browser.tasks.claim.ClaimResultHandler; +import io.lbry.browser.tasks.claim.StreamRepostTask; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryUri; +import lombok.Setter; + +public class RepostClaimDialogFragment extends BottomSheetDialogFragment implements WalletBalanceListener { + public static final String TAG = "RepostClaimDialog"; + + private MaterialButton buttonRepost; + private View linkCancel; + private TextInputEditText inputDeposit; + private View inlineBalanceContainer; + private TextView inlineBalanceValue; + private ProgressBar repostProgress; + private TextView textTitle; + + private AppCompatSpinner channelSpinner; + private InlineChannelSpinnerAdapter channelSpinnerAdapter; + private TextView textNamePrefix; + private EditText inputName; + private TextView linkToggleAdvanced; + private View advancedContainer; + + @Setter + private RepostClaimListener listener; + @Setter + private Claim claim; + + public static RepostClaimDialogFragment newInstance() { + return new RepostClaimDialogFragment(); + } + + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View view = inflater.inflate(R.layout.dialog_repost_claim, container, false); + + buttonRepost = view.findViewById(R.id.repost_button); + linkCancel = view.findViewById(R.id.repost_cancel_link); + inputDeposit = view.findViewById(R.id.repost_input_deposit); + inlineBalanceContainer = view.findViewById(R.id.repost_inline_balance_container); + inlineBalanceValue = view.findViewById(R.id.repost_inline_balance_value); + repostProgress = view.findViewById(R.id.repost_progress); + textTitle = view.findViewById(R.id.repost_title); + + channelSpinner = view.findViewById(R.id.repost_channel_spinner); + textNamePrefix = view.findViewById(R.id.repost_name_prefix); + inputName = view.findViewById(R.id.repost_name_input); + linkToggleAdvanced = view.findViewById(R.id.repost_toggle_advanced); + advancedContainer = view.findViewById(R.id.repost_advanced_container); + + textTitle.setText(getString(R.string.repost_title, claim.getTitle())); + inputName.setText(claim.getName()); + inputDeposit.setText(R.string.min_deposit); + channelSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() { + @Override + public void onItemSelected(AdapterView adapterView, View view, int position, long l) { + Object item = adapterView.getItemAtPosition(position); + if (item instanceof Claim) { + Claim claim = (Claim) item; + textNamePrefix.setText(String.format("%s%s/", LbryUri.PROTO_DEFAULT, claim.getName())); + } + } + + @Override + public void onNothingSelected(AdapterView adapterView) { + + } + }); + + inputDeposit.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + inputDeposit.setHint(hasFocus ? getString(R.string.zero) : ""); + inlineBalanceContainer.setVisibility(hasFocus ? View.VISIBLE : View.INVISIBLE); + } + }); + + linkCancel.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + dismiss(); + } + }); + + linkToggleAdvanced.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (advancedContainer.getVisibility() != View.VISIBLE) { + advancedContainer.setVisibility(View.VISIBLE); + linkToggleAdvanced.setText(R.string.hide_advanced); + } else { + advancedContainer.setVisibility(View.GONE); + linkToggleAdvanced.setText(R.string.show_advanced); + } + } + }); + + buttonRepost.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + validateAndRepostClaim(); + } + }); + + onWalletBalanceUpdated(Lbry.walletBalance); + + return view; + } + + public void onResume() { + super.onResume(); + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).addWalletBalanceListener(this); + } + fetchChannels(); + } + + public void onPause() { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).removeWalletBalanceListener(this); + } + inputDeposit.clearFocus(); + super.onPause(); + } + + + private void fetchChannels() { + if (Lbry.ownChannels == null || Lbry.ownChannels.size() == 0) { + startLoading(); + ClaimListTask task = new ClaimListTask(Claim.TYPE_CHANNEL, repostProgress, new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + Lbry.ownChannels = new ArrayList<>(claims); + loadChannels(claims); + finishLoading(); + } + + @Override + public void onError(Exception error) { + // could not fetch channels + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).showError(error.getMessage()); + } + dismiss(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } else { + loadChannels(Lbry.ownChannels); + } + } + + private void loadChannels(List channels) { + if (channelSpinnerAdapter == null) { + Context context = getContext(); + channelSpinnerAdapter = new InlineChannelSpinnerAdapter(context, R.layout.spinner_item_channel, channels); + channelSpinnerAdapter.notifyDataSetChanged(); + } else { + channelSpinnerAdapter.clear(); + channelSpinnerAdapter.addAll(channels); + channelSpinnerAdapter.notifyDataSetChanged(); + } + if (channelSpinner != null) { + channelSpinner.setAdapter(channelSpinnerAdapter); + } + } + + @Override + public void onWalletBalanceUpdated(WalletBalance walletBalance) { + if (walletBalance != null && inlineBalanceValue != null) { + inlineBalanceValue.setText(Helper.shortCurrencyFormat(walletBalance.getAvailable().doubleValue())); + } + } + + private void validateAndRepostClaim() { + String name = Helper.getValue(inputName.getText()); + if (Helper.isNullOrEmpty(name) || !LbryUri.isNameValid(name)) { + showError(getString(R.string.repost_name_invalid_characters)); + return; + } + + String depositString = Helper.getValue(inputDeposit.getText()); + if (Helper.isNullOrEmpty(depositString)) { + showError(getString(R.string.invalid_amount)); + return; + } + + BigDecimal bid = new BigDecimal(depositString); + if (bid.doubleValue() > Lbry.walletBalance.getAvailable().doubleValue()) { + showError(getString(R.string.insufficient_balance)); + return; + } + + Claim channel = (Claim) channelSpinner.getSelectedItem(); + StreamRepostTask task = new StreamRepostTask(name, bid, claim.getClaimId(), channel.getClaimId(), repostProgress, new ClaimResultHandler() { + @Override + public void beforeStart() { + startLoading(); + } + + @Override + public void onSuccess(Claim claimResult) { + if (listener != null) { + listener.onClaimReposted(claimResult); + } + finishLoading(); + dismiss(); + } + + @Override + public void onError(Exception error) { + showError(error.getMessage()); + finishLoading(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void showError(String message) { + Snackbar.make(getView(), message, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED). + setTextColor(Color.WHITE). + show(); + } + + private void startLoading() { + Dialog dialog = getDialog(); + if (dialog != null) { + dialog.setCanceledOnTouchOutside(false); + } + linkCancel.setEnabled(false); + buttonRepost.setEnabled(false); + inputName.setEnabled(false); + channelSpinner.setEnabled(false); + linkToggleAdvanced.setVisibility(View.INVISIBLE); + } + private void finishLoading() { + Dialog dialog = getDialog(); + if (dialog != null) { + dialog.setCanceledOnTouchOutside(true); + } + linkCancel.setEnabled(true); + buttonRepost.setEnabled(true); + inputName.setEnabled(true); + channelSpinner.setEnabled(true); + linkToggleAdvanced.setVisibility(View.VISIBLE); + } + + public interface RepostClaimListener { + void onClaimReposted(Claim claim); + } +} diff --git a/app/src/main/java/io/lbry/browser/dialog/SendTipDialogFragment.java b/app/src/main/java/io/lbry/browser/dialog/SendTipDialogFragment.java new file mode 100644 index 00000000..6be2d917 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/dialog/SendTipDialogFragment.java @@ -0,0 +1,196 @@ +package io.lbry.browser.dialog; + +import android.app.Dialog; +import android.content.Context; +import android.graphics.Color; +import android.os.AsyncTask; +import android.os.Bundle; +import android.text.method.LinkMovementMethod; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.core.content.ContextCompat; +import androidx.core.text.HtmlCompat; + +import com.google.android.material.bottomsheet.BottomSheetDialogFragment; +import com.google.android.material.button.MaterialButton; +import com.google.android.material.snackbar.Snackbar; +import com.google.android.material.textfield.TextInputEditText; + +import java.math.BigDecimal; +import java.text.DecimalFormat; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.listener.WalletBalanceListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.tasks.wallet.SupportCreateTask; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import lombok.Setter; + +public class SendTipDialogFragment extends BottomSheetDialogFragment implements WalletBalanceListener { + public static final String TAG = "SendTipDialog"; + + private MaterialButton sendButton; + private View cancelLink; + private TextInputEditText inputAmount; + private View inlineBalanceContainer; + private TextView inlineBalanceValue; + private ProgressBar sendProgress; + + @Setter + private SendTipListener listener; + @Setter + private Claim claim; + + public static SendTipDialogFragment newInstance() { + return new SendTipDialogFragment(); + } + + private void disableControls() { + Dialog dialog = getDialog(); + if (dialog != null) { + dialog.setCanceledOnTouchOutside(false); + } + sendButton.setEnabled(false); + cancelLink.setEnabled(false); + } + private void enableControls() { + Dialog dialog = getDialog(); + if (dialog != null) { + dialog.setCanceledOnTouchOutside(true); + } + sendButton.setEnabled(true); + cancelLink.setEnabled(true); + } + + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View view = inflater.inflate(R.layout.dialog_send_tip, container, false); + + inputAmount = view.findViewById(R.id.tip_input_amount); + inlineBalanceContainer = view.findViewById(R.id.tip_inline_balance_container); + inlineBalanceValue = view.findViewById(R.id.tip_inline_balance_value); + sendProgress = view.findViewById(R.id.tip_send_progress); + cancelLink = view.findViewById(R.id.tip_cancel_link); + sendButton = view.findViewById(R.id.tip_send); + + inputAmount.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + inputAmount.setHint(hasFocus ? getString(R.string.zero) : ""); + inlineBalanceContainer.setVisibility(hasFocus ? View.VISIBLE : View.INVISIBLE); + } + }); + + TextView infoText = view.findViewById(R.id.tip_info); + infoText.setMovementMethod(LinkMovementMethod.getInstance()); + infoText.setText(HtmlCompat.fromHtml( + Claim.TYPE_CHANNEL.equalsIgnoreCase(claim.getValueType()) ? + getString(R.string.send_tip_info_channel, claim.getTitleOrName()) : + getString(R.string.send_tip_info_content, claim.getTitleOrName()), + HtmlCompat.FROM_HTML_MODE_LEGACY)); + + sendButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + String amountString = Helper.getValue(inputAmount.getText()); + if (Helper.isNullOrEmpty(amountString)) { + showError(getString(R.string.invalid_amount)); + return; + } + + BigDecimal amount = new BigDecimal(amountString); + if (amount.doubleValue() > Lbry.walletBalance.getAvailable().doubleValue()) { + showError(getString(R.string.insufficient_balance)); + return; + } + + SupportCreateTask task = new SupportCreateTask(claim.getClaimId(), amount, true, sendProgress, new GenericTaskHandler() { + @Override + public void beforeStart() { + disableControls(); + } + + @Override + public void onSuccess() { + enableControls(); + if (listener != null) { + listener.onTipSent(amount); + } + + dismiss(); + } + + @Override + public void onError(Exception error) { + showError(error.getMessage()); + enableControls(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + }); + + cancelLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + dismiss(); + } + }); + + String channel = null; + if (Claim.TYPE_CHANNEL.equalsIgnoreCase(claim.getValueType())) { + channel = claim.getTitleOrName(); + } else if (claim.getSigningChannel() != null) { + channel = claim.getPublisherTitle(); + } + ((TextView) view.findViewById(R.id.tip_send_title)).setText( + Helper.isNullOrEmpty(channel) ? getString(R.string.send_a_tip) : getString(R.string.send_a_tip_to, channel) + ); + + onWalletBalanceUpdated(Lbry.walletBalance); + + return view; + } + + public void onResume() { + super.onResume(); + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).addWalletBalanceListener(this); + } + } + + public void onPause() { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).removeWalletBalanceListener(this); + } + super.onPause(); + } + + @Override + public void onWalletBalanceUpdated(WalletBalance walletBalance) { + if (walletBalance != null && inlineBalanceValue != null) { + inlineBalanceValue.setText(Helper.shortCurrencyFormat(walletBalance.getAvailable().doubleValue())); + } + } + + private void showError(String message) { + Snackbar.make(getView(), message, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED). + setTextColor(Color.WHITE). + show(); + } + + public interface SendTipListener { + void onTipSent(BigDecimal amount); + } +} 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/CameraPermissionListener.java b/app/src/main/java/io/lbry/browser/listener/CameraPermissionListener.java new file mode 100644 index 00000000..ca228c18 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/CameraPermissionListener.java @@ -0,0 +1,8 @@ +package io.lbry.browser.listener; + +public interface CameraPermissionListener { + void onCameraPermissionGranted(); + void onCameraPermissionRefused(); + void onRecordAudioPermissionGranted(); + void onRecordAudioPermissionRefused(); +} 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/DownloadActionListener.java b/app/src/main/java/io/lbry/browser/listener/DownloadActionListener.java new file mode 100644 index 00000000..c93f950e --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/DownloadActionListener.java @@ -0,0 +1,5 @@ +package io.lbry.browser.listener; + +public interface DownloadActionListener { + void onDownloadAction(String downloadAction, String uri, String outpoint, String fileInfoJson, double progress); +} diff --git a/app/src/main/java/io/lbry/browser/listener/FetchChannelsListener.java b/app/src/main/java/io/lbry/browser/listener/FetchChannelsListener.java new file mode 100644 index 00000000..b2f41e21 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/FetchChannelsListener.java @@ -0,0 +1,9 @@ +package io.lbry.browser.listener; + +import java.util.List; + +import io.lbry.browser.model.Claim; + +public interface FetchChannelsListener { + void onChannelsFetched(List channels); +} diff --git a/app/src/main/java/io/lbry/browser/listener/FetchClaimsListener.java b/app/src/main/java/io/lbry/browser/listener/FetchClaimsListener.java new file mode 100644 index 00000000..8a0bbda5 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/FetchClaimsListener.java @@ -0,0 +1,9 @@ +package io.lbry.browser.listener; + +import java.util.List; + +import io.lbry.browser.model.Claim; + +public interface FetchClaimsListener { + void onClaimsFetched(List claims); +} diff --git a/app/src/main/java/io/lbry/browser/listener/FilePickerListener.java b/app/src/main/java/io/lbry/browser/listener/FilePickerListener.java new file mode 100644 index 00000000..e4dc8128 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/FilePickerListener.java @@ -0,0 +1,6 @@ +package io.lbry.browser.listener; + +public interface FilePickerListener { + void onFilePicked(String filePath); + void onFilePickerCancelled(); +} 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/SelectionModeListener.java b/app/src/main/java/io/lbry/browser/listener/SelectionModeListener.java new file mode 100644 index 00000000..01fa043d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/SelectionModeListener.java @@ -0,0 +1,7 @@ +package io.lbry.browser.listener; + +public interface SelectionModeListener { + void onEnterSelectionMode(); + void onExitSelectionMode(); + void onItemSelectionToggled(); +} 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..866e2774 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/SignInListener.java @@ -0,0 +1,10 @@ +package io.lbry.browser.listener; + +public interface SignInListener { + void onEmailAdded(String email); + void onEmailEdit(); + void onEmailVerified(); + void onPhoneAdded(String countryCode, String phoneNumber); + void onPhoneVerified(); + void onManualVerifyContinue(); +} diff --git a/app/src/main/java/io/lbry/browser/listener/StoragePermissionListener.java b/app/src/main/java/io/lbry/browser/listener/StoragePermissionListener.java new file mode 100644 index 00000000..44bcd3dc --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/StoragePermissionListener.java @@ -0,0 +1,6 @@ +package io.lbry.browser.listener; + +public interface StoragePermissionListener { + void onStoragePermissionGranted(); + void onStoragePermissionRefused(); +} diff --git a/app/src/main/java/io/lbry/browser/listener/TagListener.java b/app/src/main/java/io/lbry/browser/listener/TagListener.java new file mode 100644 index 00000000..4e5071bb --- /dev/null +++ b/app/src/main/java/io/lbry/browser/listener/TagListener.java @@ -0,0 +1,8 @@ +package io.lbry.browser.listener; + +import io.lbry.browser.model.Tag; + +public interface TagListener { + void onTagAdded(Tag tag); + void onTagRemoved(Tag tag); +} 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..31e8b5fb --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/Claim.java @@ -0,0 +1,490 @@ +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.math.BigDecimal; +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 io.lbry.browser.utils.Predefined; +import lombok.Data; +import lombok.EqualsAndHashCode; + +@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 TYPE_REPOST = "repost"; + + 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 placeholderAnonymous; + 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 | repost + private Claim repostedClaim; + private Claim signingChannel; + private String repostChannelUrl; + private boolean isChannelSignatureValid; + private GenericMetadata value; + private LbryFile file; // associated file if it exists + + // device it was viewed on (for view history) + private String device; + + public static Claim claimFromOutput(JSONObject item) { + // we only need name, permanent_url, txid and nout + Claim claim = new Claim(); + claim.setClaimId(Helper.getJSONString("claim_id", null, item)); + claim.setName(Helper.getJSONString("name", null, item)); + claim.setPermanentUrl(Helper.getJSONString("permanent_url", null, item)); + claim.setTxid(Helper.getJSONString("txid", null, item)); + claim.setNout(Helper.getJSONInt("nout", -1, item)); + return claim; + } + + public String getOutpoint() { + return String.format("%s:%d", txid, nout); + } + + public boolean isFree() { + if (!(value instanceof StreamMetadata)) { + return true; + } + + Fee fee = ((StreamMetadata) value).getFee(); + return fee == null || Helper.parseDouble(fee.getAmount(), 0) == 0; + } + + public BigDecimal getActualCost(double usdRate) { + if (!(value instanceof StreamMetadata)) { + return new BigDecimal(0); + } + + Fee fee = ((StreamMetadata) value).getFee(); + if (fee != null) { + double amount = Helper.parseDouble(fee.getAmount(), 0); + if ("usd".equalsIgnoreCase(fee.getCurrency())) { + return new BigDecimal(String.valueOf(amount / usdRate)); + } + + return new BigDecimal(String.valueOf(amount)); // deweys + } + + return new BigDecimal(0); + } + + public String getMediaType() { + if (value instanceof StreamMetadata) { + StreamMetadata metadata = (StreamMetadata) value; + String mediaType = metadata.getSource() != null ? metadata.getSource().getMediaType() : null; + return mediaType; + } + return null; + } + + public boolean isPlayable() { + if (value instanceof StreamMetadata) { + StreamMetadata metadata = (StreamMetadata) value; + String mediaType = metadata.getSource() != null ? metadata.getSource().getMediaType() : null; + if (mediaType != null) { + return mediaType.startsWith("video") || mediaType.startsWith("audio"); + } + } + return false; + } + public boolean isViewable() { + if (value instanceof StreamMetadata) { + StreamMetadata metadata = (StreamMetadata) value; + String mediaType = metadata.getSource() != null ? metadata.getSource().getMediaType() : null; + if (mediaType != null) { + return mediaType.startsWith("image") || mediaType.startsWith("text"); + } + } + return false; + } + public boolean isMature() { + List tags = getTags(); + if (tags != null && tags.size() > 0) { + for (String tag : tags) { + if (Predefined.MATURE_TAGS.contains(tag.toLowerCase())) { + return true; + } + } + } + return false; + } + + 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 getWebsiteUrl() { + return (value instanceof ChannelMetadata) ? ((ChannelMetadata) value).getWebsiteUrl() : null; + } + + public String getEmail() { + return (value instanceof ChannelMetadata) ? ((ChannelMetadata) value).getEmail() : 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 String getTitleOrName() { + return (value != null) ? value.getTitle() : getName(); + } + + 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 fromViewHistory(ViewHistory viewHistory) { + // only for stream claims + Claim claim = new Claim(); + claim.setClaimId(viewHistory.getClaimId()); + claim.setName(viewHistory.getClaimName()); + claim.setValueType(TYPE_STREAM); + claim.setPermanentUrl(viewHistory.getUri().toString()); + claim.setDevice(viewHistory.getDevice()); + claim.setConfirmations(1); + + StreamMetadata value = new StreamMetadata(); + value.setTitle(viewHistory.getTitle()); + value.setReleaseTime(viewHistory.getReleaseTime()); + if (!Helper.isNullOrEmpty(viewHistory.getThumbnailUrl())) { + Resource thumbnail = new Resource(); + thumbnail.setUrl(viewHistory.getThumbnailUrl()); + value.setThumbnail(thumbnail); + } + if (viewHistory.getCost() != null && viewHistory.getCost().doubleValue() > 0) { + Fee fee = new Fee(); + fee.setAmount(String.valueOf(viewHistory.getCost().doubleValue())); + fee.setCurrency(viewHistory.getCurrency()); + value.setFee(fee); + } + + claim.setValue(value); + + if (!Helper.isNullOrEmpty(viewHistory.getPublisherClaimId())) { + Claim signingChannel = new Claim(); + signingChannel.setClaimId(viewHistory.getPublisherClaimId()); + signingChannel.setName(viewHistory.getPublisherName()); + if (!Helper.isNullOrEmpty(viewHistory.getPublisherTitle())) { + GenericMetadata channelValue = new GenericMetadata(); + channelValue.setTitle(viewHistory.getPublisherTitle()); + signingChannel.setValue(channelValue); + } + claim.setSigningChannel(signingChannel); + } + + return claim; + } + + public static Claim fromJSONObject(JSONObject claimObject) { + Claim claim = null; + 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 = gson.fromJson(claimJson, type); + + try { + String valueType = claim.getValueType(); + // Specific value type parsing + if (TYPE_REPOST.equalsIgnoreCase(valueType)) { + JSONObject repostedClaimObject = claimObject.getJSONObject("reposted_claim"); + claim.setRepostedClaim(Claim.fromJSONObject(repostedClaimObject)); + } else { + JSONObject value = claimObject.getJSONObject("value"); + 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")); + claim.setConfirmations(1); + + 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(new BigDecimal(String.valueOf(feeAmount)).divide(new BigDecimal(100000000)))); + fee.setCurrency("LBC"); + } + + ((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")); + LbryUri channelUri = new LbryUri(); + channelUri.setChannelClaimId(signingChannel.getClaimId()); + channelUri.setChannelName(signingChannel.getName()); + signingChannel.setPermanentUrl(channelUri.toString()); + + 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..6f8c868b --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/ClaimCacheKey.java @@ -0,0 +1,68 @@ +package io.lbry.browser.model; + +import androidx.annotation.Nullable; + +import io.lbry.browser.utils.Helper; +import lombok.Getter; +import lombok.Setter; +import lombok.ToString; + +/** + * Class to represent a key to check equality with another object + */ +@ToString +public class ClaimCacheKey { + @Getter + @Setter + private String claimId; + @Getter + @Setter + private String url; + + public static ClaimCacheKey fromClaimShortUrl(Claim claim) { + ClaimCacheKey key = new ClaimCacheKey(); + key.setUrl(claim.getShortUrl()); + return key; + } + + public static ClaimCacheKey fromClaimPermanentUrl(Claim claim) { + ClaimCacheKey key = new ClaimCacheKey(); + key.setUrl(claim.getPermanentUrl()); + return key; + } + + public static ClaimCacheKey fromClaim(Claim claim) { + ClaimCacheKey key = new ClaimCacheKey(); + key.setClaimId(claim.getClaimId()); + key.setUrl(!Helper.isNullOrEmpty(claim.getShortUrl()) ? claim.getShortUrl() : claim.getPermanentUrl()); + 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(url) && !Helper.isNullOrEmpty(key.getUrl())) { + return url.equalsIgnoreCase(key.getUrl()); + } + return false; + } + + @Override + public int hashCode() { + if (!Helper.isNullOrEmpty(url)) { + return url.hashCode(); + } + if (!Helper.isNullOrEmpty(claimId)) { + return claimId.hashCode(); + } + + return super.hashCode(); + } +} + diff --git a/app/src/main/java/io/lbry/browser/model/ClaimSearchCacheValue.java b/app/src/main/java/io/lbry/browser/model/ClaimSearchCacheValue.java new file mode 100644 index 00000000..60260add --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/ClaimSearchCacheValue.java @@ -0,0 +1,26 @@ +package io.lbry.browser.model; + +import java.util.ArrayList; +import java.util.Date; +import java.util.List; + +import lombok.Getter; +import lombok.Setter; + +public class ClaimSearchCacheValue { + @Getter + @Setter + private List claims; + @Getter + @Setter + private long timestamp; + + public ClaimSearchCacheValue(List claims, long timestamp) { + this.claims = new ArrayList<>(claims); + this.timestamp = timestamp; + } + + public boolean isExpired(long ttl) { + return System.currentTimeMillis() - timestamp > ttl; + } +} diff --git a/app/src/main/java/io/lbry/browser/model/EditorsChoiceItem.java b/app/src/main/java/io/lbry/browser/model/EditorsChoiceItem.java new file mode 100644 index 00000000..49565d7a --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/EditorsChoiceItem.java @@ -0,0 +1,23 @@ +package io.lbry.browser.model; + +import lombok.Data; + +@Data +public class EditorsChoiceItem { + private boolean header; + private String title; + private String parent; + private String description; + private String thumbnailUrl; + private String permanentUrl; + + public static EditorsChoiceItem fromClaim(Claim claim) { + EditorsChoiceItem item = new EditorsChoiceItem(); + item.setTitle(claim.getTitle()); + item.setDescription(claim.getDescription()); + item.setThumbnailUrl(claim.getThumbnailUrl()); + item.setPermanentUrl(claim.getPermanentUrl()); + + return item; + } +} 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/GalleryItem.java b/app/src/main/java/io/lbry/browser/model/GalleryItem.java new file mode 100644 index 00000000..43feee58 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/GalleryItem.java @@ -0,0 +1,13 @@ +package io.lbry.browser.model; + +import lombok.Data; + +@Data +public class GalleryItem { + private String id; + private String name; + private String filePath; + private String type; + private String thumbnailPath; + private long duration; +} diff --git a/app/src/main/java/io/lbry/browser/model/Language.java b/app/src/main/java/io/lbry/browser/model/Language.java new file mode 100644 index 00000000..e3f4567b --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/Language.java @@ -0,0 +1,16 @@ +package io.lbry.browser.model; + +import lombok.Data; + +@Data +public class Language { + private String code; + private String name; + private int stringResourceId; + + public Language(String code, String name, int stringResourceId) { + this.code = code; + this.name = name; + this.stringResourceId = stringResourceId; + } +} diff --git a/app/src/main/java/io/lbry/browser/model/LbryFile.java b/app/src/main/java/io/lbry/browser/model/LbryFile.java new file mode 100644 index 00000000..ed9cb1fb --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/LbryFile.java @@ -0,0 +1,92 @@ +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 io.lbry.browser.utils.LbryUri; +import lombok.Data; +import lombok.EqualsAndHashCode; + +@Data +@EqualsAndHashCode(onlyExplicitlyIncluded = true) +public class LbryFile { + private Claim.StreamMetadata metadata; + private long addedOn; + private int blobsCompleted; + private int blobsInStream; + private int blobsRemaining; + private String channelClaimId; + private String channelName; + @EqualsAndHashCode.Include + 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 timestamp; + private long totalBytes; + private long totalBytesLowerBound; + private String txid; + private long writtenBytes; + + private Claim generatedClaim; + + public Claim getClaim() { + if (generatedClaim != null) { + return generatedClaim; + } + + generatedClaim = new Claim(); + generatedClaim.setValueType(Claim.TYPE_STREAM); + generatedClaim.setPermanentUrl(LbryUri.tryParse(String.format("%s#%s", claimName, claimId)).toString()); + generatedClaim.setClaimId(claimId); + generatedClaim.setName(claimName); + generatedClaim.setValue(metadata); + generatedClaim.setConfirmations(1); + generatedClaim.setTxid(txid); + generatedClaim.setNout(nout); + generatedClaim.setFile(this); + + if (channelClaimId != null) { + Claim signingChannel = new Claim(); + signingChannel.setClaimId(channelClaimId); + signingChannel.setName(channelName); + signingChannel.setPermanentUrl(LbryUri.tryParse(String.format("%s#%s", claimName, claimId)).toString()); + generatedClaim.setSigningChannel(signingChannel); + } + + return generatedClaim; + } + + public static LbryFile fromJSONObject(JSONObject fileObject) { + String fileJson = fileObject.toString(); + Type type = new TypeToken(){}.getType(); + Gson gson = new GsonBuilder().setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES).create(); + LbryFile file = gson.fromJson(fileJson, type); + + if (file.getMetadata() != null && file.getMetadata().getReleaseTime() == 0) { + file.getMetadata().setReleaseTime(file.getTimestamp()); + } + return file; + } +} diff --git a/app/src/main/java/io/lbry/browser/model/License.java b/app/src/main/java/io/lbry/browser/model/License.java new file mode 100644 index 00000000..b278eafa --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/License.java @@ -0,0 +1,20 @@ +package io.lbry.browser.model; + +import lombok.Data; + +@Data +public class License { + private String name; + private String url; + private int stringResourceId; + + public License(String name, int stringResourceId) { + this.name = name; + this.stringResourceId = stringResourceId; + } + public License(String name, String url, int stringResourceId) { + this.name = name; + this.url = url; + this.stringResourceId = stringResourceId; + } +} 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..53468b09 --- /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_ALL_CONTENT = 103; + + // 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 extraLabel; + 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..6f898bea --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/Tag.java @@ -0,0 +1,44 @@ +package io.lbry.browser.model; + +import java.util.Comparator; + +import io.lbry.browser.utils.Predefined; +import lombok.Getter; +import lombok.Setter; + +public class Tag implements Comparator { + @Getter + @Setter + private String name; + @Getter + @Setter + private boolean followed; + + public Tag() { + + } + public Tag(String name) { + this.name = name; + } + + public String getLowercaseName() { + return name.toLowerCase(); + } + + public boolean isMature() { + return Predefined.MATURE_TAGS.contains(name.toLowerCase()); + } + + public String toString() { + return getLowercaseName(); + } + public boolean equals(Object o) { + return (o instanceof Tag) && ((Tag) o).getName().equalsIgnoreCase(name); + } + public int hashCode() { + return name.toLowerCase().hashCode(); + } + public int compare(Tag a, Tag b) { + return a.getLowercaseName().compareToIgnoreCase(b.getLowercaseName()); + } +} 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..e48f3913 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/Transaction.java @@ -0,0 +1,133 @@ +package io.lbry.browser.model; + +import org.json.JSONArray; +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.exceptions.LbryUriException; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryUri; +import lombok.Data; + +@Data +public class Transaction { + private int confirmations; + private Date txDate; + private String date; + private String claim; + private String claimId; + private String txid; + private BigDecimal value; + private BigDecimal fee; + private long timestamp; + private int descriptionStringId; + private TransactionInfo abandonInfo; + private TransactionInfo claimInfo; + private TransactionInfo supportInfo; + private TransactionInfo updateInfo; + + public LbryUri getClaimUrl() { + if (!Helper.isNullOrEmpty(claim) && !Helper.isNullOrEmpty(claimId)) { + try { + return LbryUri.parse(LbryUri.normalize(String.format("%s#%s", claim, claimId))); + } catch (LbryUriException ex) { + // pass + } + } + return null; + } + + 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")) { + JSONArray array = jsonObject.getJSONArray("abandon_info"); + if (array.length() > 0) { + info = TransactionInfo.fromJSONObject(array.getJSONObject(0)); + descStringId = R.string.abandon; + transaction.setAbandonInfo(info); + } + } + if (info == null && jsonObject.has("claim_info")) { + JSONArray array = jsonObject.getJSONArray("claim_info"); + if (array.length() > 0) { + info = TransactionInfo.fromJSONObject(array.getJSONObject(0)); + descStringId = info.getClaimName().startsWith("@") ? R.string.channel : R.string.publish; + transaction.setClaimInfo(info); + } + } + if (info == null && jsonObject.has("support_info")) { + JSONArray array = jsonObject.getJSONArray("support_info"); + if (array.length() > 0) { + info = TransactionInfo.fromJSONObject(array.getJSONObject(0)); + descStringId = info.isTip() ? R.string.tip : R.string.support; + transaction.setSupportInfo(info); + } + } + if (info == null && jsonObject.has("update_info")) { + JSONArray array = jsonObject.getJSONArray("update_info"); + if (array.length() > 0) { + info = TransactionInfo.fromJSONObject(array.getJSONObject(0)); + descStringId = info.getClaimName().startsWith("@") ? R.string.channel_update : R.string.publish_update; + transaction.setUpdateInfo(info); + } + } + if (info != null) { + transaction.setClaim(info.getClaimName()); + transaction.setClaimId(info.getClaimId()); + } + } catch (JSONException ex) { + // pass + } + + if (transaction.getValue().doubleValue() == 0 && info != null && info.getBalanceDelta().doubleValue() != 0) { + transaction.setValue(info.getBalanceDelta()); + } + + 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 BigDecimal balanceDelta; + 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.setBalanceDelta(new BigDecimal(Helper.getJSONString("balance_delta", "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..b5ed912a --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/UrlSuggestion.java @@ -0,0 +1,55 @@ +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 + private boolean titleTextOnly; + private boolean titleUrlOnly; + private boolean useTextAsDescription; + + public UrlSuggestion() { + + } + public UrlSuggestion(int type, String text) { + this.type = type; + this.text = text; + } + public UrlSuggestion(int type, String text, LbryUri uri) { + this(type, text); + this.uri = uri; + } + public UrlSuggestion(int type, String text, LbryUri uri, boolean titleTextOnly) { + this(type, text, uri); + this.titleTextOnly = titleTextOnly; + } + + public String getTitle() { + if (titleUrlOnly && (type == TYPE_CHANNEL || type == TYPE_FILE)) { + return uri.toString(); + } + + if (!titleTextOnly) { + switch (type) { + case TYPE_CHANNEL: + return String.format("%s - %s", text.startsWith("@") ? text.substring(1) : text, uri.toVanityString()); + case TYPE_FILE: + return String.format("%s - %s", text, uri.toVanityString()); + case TYPE_TAG: + return String.format("%s - #%s", text, text); + } + } + + return text; + } +} diff --git a/app/src/main/java/io/lbry/browser/model/ViewHistory.java b/app/src/main/java/io/lbry/browser/model/ViewHistory.java new file mode 100644 index 00000000..ea668615 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/ViewHistory.java @@ -0,0 +1,64 @@ +package io.lbry.browser.model; + +import java.math.BigDecimal; +import java.util.Date; + +import io.lbry.browser.exceptions.LbryUriException; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; +import lombok.Data; + +@Data +public class ViewHistory { + private LbryUri uri; + private String claimId; + private String claimName; + private BigDecimal cost; + private String currency; + private String title; + private String publisherClaimId; + private String publisherName; + private String publisherTitle; + private String thumbnailUrl; + private String device; + private long releaseTime; + private Date timestamp; + + public static ViewHistory fromClaimWithUrlAndDeviceName(Claim claim, String url, String deviceName) { + ViewHistory history = new ViewHistory(); + LbryUri uri = LbryUri.tryParse(url); + if (uri == null) { + uri = LbryUri.tryParse(claim.getPermanentUrl()); + } + history.setUri(uri); + history.setClaimId(claim.getClaimId()); + history.setClaimName(claim.getName()); + history.setTitle(claim.getTitle()); + history.setThumbnailUrl(claim.getThumbnailUrl()); + + Claim.GenericMetadata metadata = claim.getValue(); + if (metadata instanceof Claim.StreamMetadata) { + Claim.StreamMetadata value = (Claim.StreamMetadata) metadata; + history.setReleaseTime(value.getReleaseTime()); + if (value.getFee() != null) { + Fee fee = value.getFee(); + history.setCost(new BigDecimal(fee.getAmount())); + history.setCurrency(fee.getCurrency()); + } + } + if (history.getReleaseTime() == 0) { + history.setReleaseTime(claim.getTimestamp()); + } + + Claim signingChannel = claim.getSigningChannel(); + if (signingChannel != null) { + history.setPublisherClaimId(signingChannel.getClaimId()); + history.setPublisherName(signingChannel.getName()); + history.setPublisherTitle(signingChannel.getTitle()); + } + + history.setDevice(deviceName); + + return history; + } +} 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..05857ff0 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/WalletSync.java @@ -0,0 +1,20 @@ +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) { + this.hash = hash; + this.data = data; + } + + public WalletSync(String hash, String data, boolean changed) { + this(hash, data); + this.changed = changed; + } +} diff --git a/app/src/main/java/io/lbry/browser/model/lbryinc/Invitee.java b/app/src/main/java/io/lbry/browser/model/lbryinc/Invitee.java new file mode 100644 index 00000000..2d84072d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/lbryinc/Invitee.java @@ -0,0 +1,11 @@ +package io.lbry.browser.model.lbryinc; + +import lombok.Data; + +@Data +public class Invitee { + private boolean header; + private String email; + private boolean inviteRewardClaimed; + private boolean inviteRewardClaimable; +} diff --git a/app/src/main/java/io/lbry/browser/model/lbryinc/Reward.java b/app/src/main/java/io/lbry/browser/model/lbryinc/Reward.java new file mode 100644 index 00000000..67f46db6 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/lbryinc/Reward.java @@ -0,0 +1,69 @@ +package io.lbry.browser.model.lbryinc; + +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 io.lbry.browser.model.Claim; +import io.lbry.browser.utils.Helper; +import lombok.Data; + +@Data +public class Reward { + public static final String TYPE_NEW_DEVELOPER = "new_developer"; + public static final String TYPE_NEW_USER = "new_user"; + public static final String TYPE_CONFIRM_EMAIL = "email_provided"; + public static final String TYPE_FIRST_CHANNEL = "new_channel"; + public static final String TYPE_FIRST_STREAM = "first_stream"; + public static final String TYPE_MANY_DOWNLOADS = "many_downloads"; + public static final String TYPE_FIRST_PUBLISH = "first_publish"; + public static final String TYPE_REFERRAL = "referrer"; + public static final String TYPE_REFEREE = "referee"; + public static final String TYPE_REWARD_CODE = "reward_code"; + public static final String TYPE_SUBSCRIPTION = "subscription"; + public static final String YOUTUBE_CREATOR = "youtube_creator"; + public static final String TYPE_DAILY_VIEW = "daily_view"; + public static final String TYPE_NEW_ANDROID = "new_android"; + + private boolean custom; + private long id; + private String rewardType; + private double rewardAmount; + private String transactionId; + private String createdAt; + private String updatedAt; + private String rewardTitle; + private String rewardDescription; + private String rewardNotification; + private String rewardRange; + + public String getDisplayAmount() { + if (shouldDisplayRange()) { + return rewardRange.split("-")[1]; + } + if (rewardAmount > 0) { + return String.valueOf(rewardAmount); + } + return "?"; + } + + public boolean isClaimed() { + return !Helper.isNullOrEmpty(transactionId); + } + + public boolean shouldDisplayRange() { + return (!isClaimed() && !Helper.isNullOrEmpty(rewardRange) && rewardRange.indexOf('-') > -1); + } + + public static Reward fromJSONObject(JSONObject rewardObject) { + String rewardJson = rewardObject.toString(); + Type type = new TypeToken(){}.getType(); + Gson gson = new GsonBuilder().setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES).create(); + return gson.fromJson(rewardJson, type); + } +} 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..30f2412a --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/lbryinc/Subscription.java @@ -0,0 +1,36 @@ +package io.lbry.browser.model.lbryinc; + +import io.lbry.browser.model.Claim; +import lombok.Getter; +import lombok.Setter; + +public class Subscription { + @Getter + @Setter + private String channelName; + @Getter + @Setter + private String url; + + public Subscription() { + + } + public Subscription(String channelName, String url) { + this.channelName = channelName; + this.url = url; + } + + public static Subscription fromClaim(Claim claim) { + return new Subscription(claim.getName(), claim.getPermanentUrl()); + } + public String toString() { + return url; + } + + public boolean equals(Object o) { + return (o instanceof Subscription) && url != null && url.equalsIgnoreCase(((Subscription) o).getUrl()); + } + public int hashCode() { + return url.toLowerCase().hashCode(); + } +} 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..15c5bec2 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/model/lbryinc/User.java @@ -0,0 +1,42 @@ +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; + + @Data + public static class YoutubeChannel { + String ytChannelName; + String lbryChannelName; + String channelClaimId; + String syncStatus; + String statusToken; + boolean transferable; + String transferState; + List publishToAddress; + String publicKey; + } +} 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/FollowUnfollowTagTask.java b/app/src/main/java/io/lbry/browser/tasks/FollowUnfollowTagTask.java new file mode 100644 index 00000000..3c7f4dca --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/FollowUnfollowTagTask.java @@ -0,0 +1,65 @@ +package io.lbry.browser.tasks; + +import android.content.Context; +import android.database.sqlite.SQLiteDatabase; +import android.os.AsyncTask; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.model.Tag; +import io.lbry.browser.utils.Lbry; + +public class FollowUnfollowTagTask extends AsyncTask { + private Tag tag; + private boolean unfollowing; + private Context context; + private FollowUnfollowTagHandler handler; + private Exception error; + + public FollowUnfollowTagTask(Tag tag, boolean unfollowing, Context context, FollowUnfollowTagHandler handler) { + this.tag = tag; + this.context = context; + this.unfollowing = unfollowing; + this.handler = handler; + } + public Boolean doInBackground(Void... params) { + try { + SQLiteDatabase db = null; + if (context instanceof MainActivity) { + db = ((MainActivity) context).getDbHelper().getWritableDatabase(); + if (db != null) { + if (!Lbry.knownTags.contains(tag)) { + DatabaseHelper.createOrUpdateTag(tag, db); + Lbry.addKnownTag(tag); + } + + tag.setFollowed(!unfollowing); + DatabaseHelper.createOrUpdateTag(tag, db); + if (unfollowing) { + Lbry.removeFollowedTag(tag); + } else { + Lbry.addFollowedTag(tag); + } + return true; + } + } + } catch (Exception ex) { + error = ex; + } + return false; + } + protected void onPostExecute(Boolean result) { + if (handler != null) { + if (result) { + handler.onSuccess(tag, unfollowing); + } else { + handler.onError(error); + } + } + } + + public interface FollowUnfollowTagHandler { + void onSuccess(Tag tag, boolean unfollowing); + 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..6da1d943 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/LighthouseSearchTask.java @@ -0,0 +1,56 @@ +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.tasks.claim.ClaimSearchResultHandler; +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 ClaimSearchResultHandler handler; + private ProgressBar progressBar; + private Exception error; + + public LighthouseSearchTask(String rawQuery, int size, int from, boolean nsfw, String relatedTo, ProgressBar progressBar, 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/LoadTagsTask.java b/app/src/main/java/io/lbry/browser/tasks/LoadTagsTask.java new file mode 100644 index 00000000..4eb465e3 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/LoadTagsTask.java @@ -0,0 +1,59 @@ +package io.lbry.browser.tasks; + +import android.content.Context; +import android.database.sqlite.SQLiteDatabase; +import android.database.sqlite.SQLiteException; +import android.os.AsyncTask; +import android.view.View; +import android.widget.ProgressBar; + +import java.util.List; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.exceptions.LbryRequestException; +import io.lbry.browser.exceptions.LbryResponseException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.Tag; +import io.lbry.browser.utils.Helper; + +public class LoadTagsTask extends AsyncTask> { + private Context context; + private LoadTagsHandler handler; + private Exception error; + + public LoadTagsTask(Context context, LoadTagsHandler handler) { + this.context = context; + this.handler = handler; + } + protected List doInBackground(Void... params) { + List tags = null; + SQLiteDatabase db = null; + try { + if (context instanceof MainActivity) { + db = ((MainActivity) context).getDbHelper().getReadableDatabase(); + if (db != null) { + tags = DatabaseHelper.getTags(db); + } + } + } catch (SQLiteException ex) { + error = ex; + } + + return tags; + } + protected void onPostExecute(List tags) { + if (handler != null) { + if (tags != null) { + handler.onSuccess(tags); + } else { + handler.onError(error); + } + } + } + + public interface LoadTagsHandler { + void onSuccess(List tags); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/MergeSubscriptionsTask.java b/app/src/main/java/io/lbry/browser/tasks/MergeSubscriptionsTask.java new file mode 100644 index 00000000..892ad6ba --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/MergeSubscriptionsTask.java @@ -0,0 +1,133 @@ +package io.lbry.browser.tasks; + +import android.content.Context; +import android.database.sqlite.SQLiteDatabase; +import android.database.sqlite.SQLiteException; +import android.os.AsyncTask; +import android.util.Log; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +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.data.DatabaseHelper; +import io.lbry.browser.exceptions.LbryUriException; +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; +import okhttp3.Response; + +// background task to create a diff of local and remote subscriptions and try to merge +public class MergeSubscriptionsTask extends AsyncTask> { + private static final String TAG = "MergeSubscriptionsTask"; + private Context context; + private List base; + private List diff; + private MergeSubscriptionsHandler handler; + private Exception error; + + public MergeSubscriptionsTask(List base, Context context, MergeSubscriptionsHandler handler) { + this.base = base; + this.context = context; + this.handler = handler; + } + + protected List doInBackground(Void... params) { + List combined = new ArrayList<>(base); + List localSubs = new ArrayList<>(); + List remoteSubs = new ArrayList<>(); + diff = new ArrayList<>(); + SQLiteDatabase db = null; + try { + // fetch local subscriptions + if (context instanceof MainActivity) { + db = ((MainActivity) context).getDbHelper().getWritableDatabase(); + } + if (db != null) { + localSubs = DatabaseHelper.getSubscriptions(db); + for (Subscription sub : localSubs) { + if (!combined.contains(sub)) { + combined.add(sub); + } + } + } + + // fetch remote subscriptions + 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); + Subscription subscription = new Subscription(channelName, url.toString()); + remoteSubs.add(subscription); + } + } + + for (int i = 0; i < combined.size(); i++) { + Subscription local = combined.get(i); + if (!remoteSubs.contains(local)) { + // add to remote subscriptions + try { + LbryUri uri = LbryUri.parse(local.getUrl()); + Map options = new HashMap<>(); + options.put("claim_id", uri.getChannelClaimId()); + options.put("channel_name", Helper.normalizeChannelName(local.getChannelName())); + Lbryio.parseResponse(Lbryio.call("subscription", "new", options, context)); + } catch (LbryUriException | LbryioRequestException | LbryioResponseException ex) { + // pass + Log.e(TAG, String.format("subscription/new failed: %s", ex.getMessage()), ex); + } + } + } + + for (int i = 0; i < localSubs.size(); i++) { + Subscription local = localSubs.get(i); + if (!base.contains(local) && !diff.contains(local)) { + diff.add(local); + } + } + for (int i = 0; i < remoteSubs.size(); i++) { + Subscription remote = remoteSubs.get(i); + if (!combined.contains(remote)) { + combined.add(remote); + if (!diff.contains(remote)) { + diff.add(remote); + } + } + } + } catch (ClassCastException | LbryioRequestException | LbryioResponseException | JSONException | IllegalStateException | SQLiteException ex) { + error = ex; + return null; + } + + return combined; + } + protected void onPostExecute(List subscriptions) { + if (handler != null) { + if (subscriptions != null) { + handler.onSuccess(subscriptions, diff); + } else { + handler.onError(error); + } + } + } + + public interface MergeSubscriptionsHandler { + void onSuccess(List subscriptions, List diff); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/ReadTextFileTask.java b/app/src/main/java/io/lbry/browser/tasks/ReadTextFileTask.java new file mode 100644 index 00000000..3dfa6b93 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/ReadTextFileTask.java @@ -0,0 +1,55 @@ +package io.lbry.browser.tasks; + +import android.os.AsyncTask; + +import java.io.BufferedReader; +import java.io.FileInputStream; +import java.io.FileNotFoundException; +import java.io.FileReader; +import java.io.IOException; +import java.io.InputStreamReader; +import java.nio.Buffer; + +import io.lbry.browser.utils.Helper; + +public class ReadTextFileTask extends AsyncTask { + private String filePath; + private Exception error; + private ReadTextFileHandler handler; + public ReadTextFileTask(String filePath, ReadTextFileHandler handler) { + this.filePath = filePath; + this.handler = handler; + } + protected String doInBackground(Void... params) { + StringBuilder sb = new StringBuilder(); + BufferedReader reader = null; + try { + reader = new BufferedReader(new InputStreamReader(new FileInputStream(filePath))); + String line = null; + while ((line = reader.readLine()) != null) { + sb.append(line).append("\n"); + } + } catch (IOException ex) { + error = ex; + return null; + } finally { + Helper.closeCloseable(reader); + } + + return sb.toString(); + } + protected void onPostExecute(String text) { + if (handler != null) { + if (!Helper.isNull(text)) { + handler.onSuccess(text); + } else { + handler.onError(error); + } + } + } + + public interface ReadTextFileHandler { + void onSuccess(String text); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/SetSdkSettingTask.java b/app/src/main/java/io/lbry/browser/tasks/SetSdkSettingTask.java new file mode 100644 index 00000000..42dafdc7 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/SetSdkSettingTask.java @@ -0,0 +1,45 @@ +package io.lbry.browser.tasks; + +import android.os.AsyncTask; + +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.utils.Lbry; + +public class SetSdkSettingTask extends AsyncTask { + private String key; + private String value; + private GenericTaskHandler handler; + private Exception error; + public SetSdkSettingTask(String key, String value, GenericTaskHandler handler) { + this.key = key; + this.value = value; + this.handler = handler; + } + + @Override + protected Boolean doInBackground(Void... params) { + try { + Map options = new HashMap<>(); + options.put("key", key); + options.put("value", value); + Lbry.genericApiCall("setting_set", options); + return true; + } catch (ApiCallException ex) { + error = ex; + return false; + } + } + + protected void onPostExecute(Boolean result) { + if (handler != null) { + if (result) { + handler.onSuccess(); + } else { + handler.onError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/UpdateSuggestedTagsTask.java b/app/src/main/java/io/lbry/browser/tasks/UpdateSuggestedTagsTask.java new file mode 100644 index 00000000..aee8dc1a --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/UpdateSuggestedTagsTask.java @@ -0,0 +1,85 @@ +package io.lbry.browser.tasks; + +import android.os.AsyncTask; + +import java.util.ArrayList; +import java.util.List; +import java.util.Random; + +import io.lbry.browser.adapter.TagListAdapter; +import io.lbry.browser.model.Tag; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class UpdateSuggestedTagsTask extends AsyncTask> { + + private boolean clearPrevious; + private boolean excludeMature; + private int limit; + private String filter; + private TagListAdapter addedTagsAdapter; + private TagListAdapter suggestedTagsAdapter; + private KnownTagsHandler handler; + + public UpdateSuggestedTagsTask( + String filter, + int limit, + TagListAdapter addedTagsAdapter, + TagListAdapter suggestedTagsAdapter, + boolean clearPrevious, + boolean excludeMature, + KnownTagsHandler handler) { + this.filter = filter; + this.limit = limit; + this.addedTagsAdapter = addedTagsAdapter; + this.suggestedTagsAdapter = suggestedTagsAdapter; + this.clearPrevious = clearPrevious; + this.excludeMature = excludeMature; + this.handler = handler; + } + + protected List doInBackground(Void... params) { + List tags = new ArrayList<>(); + if (Helper.isNullOrEmpty(filter)) { + Random random = new Random(); + if (suggestedTagsAdapter != null && !clearPrevious) { + tags = new ArrayList<>(suggestedTagsAdapter.getTags()); + } + while (tags.size() < limit) { + Tag randomTag = Lbry.knownTags.get(random.nextInt(Lbry.knownTags.size())); + if (excludeMature && randomTag.isMature()) { + continue; + } + if (!Lbry.followedTags.contains(randomTag) && (addedTagsAdapter == null || !addedTagsAdapter.getTags().contains(randomTag))) { + tags.add(randomTag); + } + } + } else { + Tag filterTag = new Tag(filter); + if (addedTagsAdapter == null || !addedTagsAdapter.getTags().contains(filterTag)) { + tags.add(new Tag(filter)); + } + for (int i = 0; i < Lbry.knownTags.size() && tags.size() < limit - 1; i++) { + Tag knownTag = Lbry.knownTags.get(i); + if (excludeMature && knownTag.isMature()) { + continue; + } + if ((knownTag.getLowercaseName().startsWith(filter) || knownTag.getLowercaseName().matches(filter)) && + (!tags.contains(knownTag) && + !Lbry.followedTags.contains(knownTag) && (addedTagsAdapter == null || !addedTagsAdapter.getTags().contains(knownTag)))) { + tags.add(knownTag); + } + } + } + return tags; + } + protected void onPostExecute(List tags) { + if (handler != null) { + handler.onSuccess(tags); + } + } + + public interface KnownTagsHandler { + void onSuccess(List tags); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/UploadImageTask.java b/app/src/main/java/io/lbry/browser/tasks/UploadImageTask.java new file mode 100644 index 00000000..2dcd8d65 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/UploadImageTask.java @@ -0,0 +1,100 @@ +package io.lbry.browser.tasks; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONException; +import org.json.JSONObject; + +import java.io.IOException; + +import java.io.File; +import java.util.concurrent.TimeUnit; + +import io.lbry.browser.exceptions.LbryResponseException; +import io.lbry.browser.utils.Helper; +import okhttp3.MediaType; +import okhttp3.MultipartBody; +import okhttp3.OkHttpClient; +import okhttp3.Request; +import okhttp3.RequestBody; +import okhttp3.Response; + +public class UploadImageTask extends AsyncTask { + private String filePath; + private View progressView; + private UploadThumbnailHandler handler; + private Exception error; + + public UploadImageTask(String filePath, View progressView, UploadThumbnailHandler handler) { + this.filePath = filePath; + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + protected String doInBackground(Void... params) { + String thumbnailUrl = null; + try { + File file = new File(filePath); + String fileName = file.getName(); + int dotIndex = fileName.lastIndexOf('.'); + String extension = "jpg"; + if (dotIndex > -1) { + extension = fileName.substring(dotIndex + 1); + } + String fileType = String.format("image/%s", extension); + RequestBody body = new MultipartBody.Builder().setType(MultipartBody.FORM). + addFormDataPart("name", Helper.makeid(24)). + addFormDataPart("file", fileName, RequestBody.create(file, MediaType.parse(fileType))). + build(); + Request request = new Request.Builder().url("https://spee.ch/api/claim/publish").post(body).build(); + OkHttpClient client = new OkHttpClient.Builder(). + writeTimeout(300, TimeUnit.SECONDS). + readTimeout(300, TimeUnit.SECONDS). + build(); + Response response = client.newCall(request).execute(); + JSONObject json = new JSONObject(response.body().string()); + if (json.has("success") && Helper.getJSONBoolean("success", false, json)) { + JSONObject data = json.getJSONObject("data"); + String url = Helper.getJSONString("url", null, data); + if (Helper.isNullOrEmpty(url)) { + throw new LbryResponseException("Invalid thumbnail url returned after upload."); + } + + thumbnailUrl = String.format("%s.%s", url, extension); + } else if (json.has("error") || json.has("message")) { + JSONObject error = Helper.getJSONObject("error", json); + String message = null; + if (error != null) { + message = Helper.getJSONString("message", null, error); + } + if (Helper.isNullOrEmpty(message)) { + message = Helper.getJSONString("message", null, json); + } + throw new LbryResponseException(Helper.isNullOrEmpty(message) ? "The image failed to upload." : message); + } + } catch (IOException | JSONException | LbryResponseException ex) { + error = ex; + } + + return thumbnailUrl; + } + protected void onPostExecute(String thumbnailUrl) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (!Helper.isNullOrEmpty(thumbnailUrl)) { + handler.onSuccess(thumbnailUrl); + } else { + handler.onError(error); + } + } + } + + public interface UploadThumbnailHandler { + void onSuccess(String url); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/AbandonChannelTask.java b/app/src/main/java/io/lbry/browser/tasks/claim/AbandonChannelTask.java new file mode 100644 index 00000000..3b71a45b --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/AbandonChannelTask.java @@ -0,0 +1,63 @@ +package io.lbry.browser.tasks.claim; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class AbandonChannelTask extends AsyncTask { + private List claimIds; + private List successfulClaimIds; + private List failedClaimIds; + private List failedExceptions; + private View progressView; + private AbandonHandler handler; + + public AbandonChannelTask(List claimIds, View progressView, AbandonHandler handler) { + this.claimIds = claimIds; + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + + public Boolean doInBackground(Void... params) { + successfulClaimIds = new ArrayList<>(); + failedClaimIds = new ArrayList<>(); + failedExceptions = new ArrayList<>(); + + for (String claimId : claimIds) { + try { + Map options = new HashMap<>(); + options.put("claim_id", claimId); + options.put("blocking", true); + JSONObject result = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_CHANNEL_ABANDON, options); + successfulClaimIds.add(claimId); + } catch (ApiCallException ex) { + failedClaimIds.add(claimId); + failedExceptions.add(ex); + } + } + + return true; + } + + protected void onPostExecute(Boolean result) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + handler.onComplete(successfulClaimIds, failedClaimIds, failedExceptions); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/AbandonHandler.java b/app/src/main/java/io/lbry/browser/tasks/claim/AbandonHandler.java new file mode 100644 index 00000000..706eaeb4 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/AbandonHandler.java @@ -0,0 +1,7 @@ +package io.lbry.browser.tasks.claim; + +import java.util.List; + +public interface AbandonHandler { + void onComplete(List successfulClaimIds, List failedClaimIds, List errors); +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/AbandonStreamTask.java b/app/src/main/java/io/lbry/browser/tasks/claim/AbandonStreamTask.java new file mode 100644 index 00000000..a5c68d93 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/AbandonStreamTask.java @@ -0,0 +1,63 @@ +package io.lbry.browser.tasks.claim; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class AbandonStreamTask extends AsyncTask { + private List claimIds; + private List successfulClaimIds; + private List failedClaimIds; + private List failedExceptions; + private View progressView; + private AbandonHandler handler; + + public AbandonStreamTask(List claimIds, View progressView, AbandonHandler handler) { + this.claimIds = claimIds; + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + + public Boolean doInBackground(Void... params) { + successfulClaimIds = new ArrayList<>(); + failedClaimIds = new ArrayList<>(); + failedExceptions = new ArrayList<>(); + + for (String claimId : claimIds) { + try { + Map options = new HashMap<>(); + options.put("claim_id", claimId); + options.put("blocking", true); + JSONObject result = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_STREAM_ABANDON, options); + successfulClaimIds.add(claimId); + } catch (ApiCallException ex) { + failedClaimIds.add(claimId); + failedExceptions.add(ex); + } + } + + return true; + } + + protected void onPostExecute(Boolean result) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + handler.onComplete(successfulClaimIds, failedClaimIds, failedExceptions); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/ChannelCreateUpdateTask.java b/app/src/main/java/io/lbry/browser/tasks/claim/ChannelCreateUpdateTask.java new file mode 100644 index 00000000..6611572d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/ChannelCreateUpdateTask.java @@ -0,0 +1,107 @@ +package io.lbry.browser.tasks.claim; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.math.BigDecimal; +import java.text.DecimalFormat; +import java.text.DecimalFormatSymbols; +import java.util.HashMap; +import java.util.Locale; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.tasks.claim.ClaimResultHandler; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class ChannelCreateUpdateTask extends AsyncTask { + private Claim claim; + private BigDecimal deposit; + private boolean update; + private Exception error; + private ClaimResultHandler handler; + private View progressView; + + public ChannelCreateUpdateTask(Claim claim, BigDecimal deposit, boolean update, View progressView, ClaimResultHandler handler) { + this.claim = claim; + this.deposit = deposit; + this.update = update; + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + if (handler != null) { + handler.beforeStart(); + } + } + protected Claim doInBackground(Void... params) { + Map options = new HashMap<>(); + if (!update) { + options.put("name", claim.getName()); + } else { + options.put("claim_id", claim.getClaimId()); + } + options.put("bid", new DecimalFormat(Helper.SDK_AMOUNT_FORMAT, new DecimalFormatSymbols(Locale.US)).format(deposit.doubleValue())); + if (!Helper.isNullOrEmpty(claim.getTitle())) { + options.put("title", claim.getTitle()); + } + if (!Helper.isNullOrEmpty(claim.getCoverUrl())) { + options.put("cover_url", claim.getCoverUrl()); + } + if (!Helper.isNullOrEmpty(claim.getThumbnailUrl())) { + options.put("thumbnail_url", claim.getThumbnailUrl()); + } + if (!Helper.isNullOrEmpty(claim.getDescription())) { + options.put("description", claim.getDescription()); + } + if (!Helper.isNullOrEmpty(claim.getWebsiteUrl())) { + options.put("website_url", claim.getWebsiteUrl()); + } + if (!Helper.isNullOrEmpty(claim.getEmail())) { + options.put("email", claim.getEmail()); + } + if (claim.getTags() != null && claim.getTags().size() > 0) { + options.put("tags", claim.getTags()); + } + options.put("blocking", true); + + Claim claimResult = null; + String method = !update ? Lbry.METHOD_CHANNEL_CREATE : Lbry.METHOD_CHANNEL_UPDATE; + try { + JSONObject result = (JSONObject) Lbry.genericApiCall(method, options); + if (result.has("outputs")) { + JSONArray outputs = result.getJSONArray("outputs"); + for (int i = 0; i < outputs.length(); i++) { + JSONObject output = outputs.getJSONObject(i); + if (output.has("claim_id") && output.has("claim_op")) { + claimResult = Claim.claimFromOutput(output); + break; + } + } + } + } catch (ApiCallException | ClassCastException | JSONException ex) { + error = ex; + } + + return claimResult; + } + + protected void onPostExecute(Claim result) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (result != null) { + handler.onSuccess(result); + } else { + handler.onError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/ClaimListResultHandler.java b/app/src/main/java/io/lbry/browser/tasks/claim/ClaimListResultHandler.java new file mode 100644 index 00000000..2d427e88 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/ClaimListResultHandler.java @@ -0,0 +1,10 @@ +package io.lbry.browser.tasks.claim; + +import java.util.List; + +import io.lbry.browser.model.Claim; + +public interface ClaimListResultHandler { + void onSuccess(List claims); + void onError(Exception error); +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/ClaimListTask.java b/app/src/main/java/io/lbry/browser/tasks/claim/ClaimListTask.java new file mode 100644 index 00000000..b017a0ba --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/ClaimListTask.java @@ -0,0 +1,71 @@ +package io.lbry.browser.tasks.claim; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +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 ClaimListTask extends AsyncTask> { + private List types; + private View progressView; + private ClaimListResultHandler handler; + private Exception error; + + public ClaimListTask(String type, View progressView, ClaimListResultHandler handler) { + this(Arrays.asList(type), progressView, handler); + } + public ClaimListTask(List types, View progressView, ClaimListResultHandler handler) { + this.types = types; + this.progressView = progressView; + this.handler = handler; + } + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + protected List doInBackground(Void... params) { + List claims = null; + + try { + Map options = new HashMap<>(); + if (types != null && types.size() > 0) { + options.put("claim_type", types); + } + options.put("page", 1); + options.put("page_size", 999); + options.put("resolve", true); + + JSONObject result = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_CLAIM_LIST, options); + JSONArray items = result.getJSONArray("items"); + claims = new ArrayList<>(); + for (int i = 0; i < items.length(); i++) { + claims.add(Claim.fromJSONObject(items.getJSONObject(i))); + } + } catch (ApiCallException | JSONException ex) { + error = ex; + } + return claims; + } + protected void onPostExecute(List claims) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (claims != null) { + handler.onSuccess(claims); + } else { + handler.onError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/ClaimResultHandler.java b/app/src/main/java/io/lbry/browser/tasks/claim/ClaimResultHandler.java new file mode 100644 index 00000000..5171486b --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/ClaimResultHandler.java @@ -0,0 +1,9 @@ +package io.lbry.browser.tasks.claim; + +import io.lbry.browser.model.Claim; + +public interface ClaimResultHandler { + void beforeStart(); + void onSuccess(Claim claimResult); + void onError(Exception error); +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/ClaimSearchResultHandler.java b/app/src/main/java/io/lbry/browser/tasks/claim/ClaimSearchResultHandler.java new file mode 100644 index 00000000..2f4bdd8c --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/ClaimSearchResultHandler.java @@ -0,0 +1,10 @@ +package io.lbry.browser.tasks.claim; + +import java.util.List; + +import io.lbry.browser.model.Claim; + +public interface ClaimSearchResultHandler { + void onSuccess(List claims, boolean hasReachedEnd); + void onError(Exception error); +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/ClaimSearchTask.java b/app/src/main/java/io/lbry/browser/tasks/claim/ClaimSearchTask.java new file mode 100644 index 00000000..696eb60c --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/ClaimSearchTask.java @@ -0,0 +1,49 @@ +package io.lbry.browser.tasks.claim; + +import android.os.AsyncTask; +import android.view.View; + +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 Helper.filterInvalidReposts(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); + } + } + } + +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/PublishClaimTask.java b/app/src/main/java/io/lbry/browser/tasks/claim/PublishClaimTask.java new file mode 100644 index 00000000..418f114b --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/PublishClaimTask.java @@ -0,0 +1,106 @@ +package io.lbry.browser.tasks.claim; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.math.BigDecimal; +import java.text.DecimalFormat; +import java.text.DecimalFormatSymbols; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.Locale; +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 PublishClaimTask extends AsyncTask { + private Claim claim; + private String filePath; + private View progressView; + private ClaimResultHandler handler; + private Exception error; + public PublishClaimTask(Claim claim, String filePath, View progressView, ClaimResultHandler handler) { + this.claim = claim; + this.filePath = filePath; + this.progressView = progressView; + this.handler = handler; + } + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + if (handler != null) { + handler.beforeStart(); + } + } + protected Claim doInBackground(Void... params) { + Claim.StreamMetadata metadata = (Claim.StreamMetadata) claim.getValue(); + DecimalFormat amountFormat = new DecimalFormat(Helper.SDK_AMOUNT_FORMAT, new DecimalFormatSymbols(Locale.US)); + + Map options = new HashMap<>(); + options.put("blocking", true); + options.put("name", claim.getName()); + options.put("bid", amountFormat.format(new BigDecimal(claim.getAmount()).doubleValue())); + options.put("title", Helper.isNullOrEmpty(claim.getTitle()) ? "" : claim.getTitle()); + options.put("description", Helper.isNullOrEmpty(claim.getDescription()) ? "" : claim.getDescription()); + options.put("thumbnail_url", Helper.isNullOrEmpty(claim.getThumbnailUrl()) ? "" : claim.getThumbnailUrl()); + + if (!Helper.isNullOrEmpty(filePath)) { + options.put("file_path", filePath); + } + if (claim.getTags() != null && claim.getTags().size() > 0) { + options.put("tags", new ArrayList<>(claim.getTags())); + } + if (metadata.getFee() != null) { + options.put("fee_currency", metadata.getFee().getCurrency()); + options.put("fee_amount", amountFormat.format(new BigDecimal(metadata.getFee().getAmount()).doubleValue())); + } + if (claim.getSigningChannel() != null) { + options.put("channel_id", claim.getSigningChannel().getClaimId()); + } + if (metadata.getLanguages() != null && metadata.getLanguages().size() > 0) { + options.put("languages", metadata.getLanguages()); + } + if (!Helper.isNullOrEmpty(metadata.getLicense())) { + options.put("license", metadata.getLicense()); + } + if (!Helper.isNullOrEmpty(metadata.getLicenseUrl())) { + options.put("license_url", metadata.getLicenseUrl()); + } + + Claim claimResult = null; + try { + JSONObject result = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_PUBLISH, options); + if (result.has("outputs")) { + JSONArray outputs = result.getJSONArray("outputs"); + for (int i = 0; i < outputs.length(); i++) { + JSONObject output = outputs.getJSONObject(i); + if (output.has("claim_id") && output.has("claim_op")) { + claimResult = Claim.claimFromOutput(output); + break; + } + } + } + } catch (ApiCallException | ClassCastException | JSONException ex) { + error = ex; + } + + return claimResult; + + } + protected void onPostExecute(Claim result) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (result != null) { + handler.onSuccess(result); + } else { + handler.onError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/PurchaseListTask.java b/app/src/main/java/io/lbry/browser/tasks/claim/PurchaseListTask.java new file mode 100644 index 00000000..46037086 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/PurchaseListTask.java @@ -0,0 +1,70 @@ +package io.lbry.browser.tasks.claim; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.HashMap; +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 PurchaseListTask extends AsyncTask> { + private int page; + private int pageSize; + private ClaimSearchResultHandler handler; + private View progressView; + private Exception error; + + public PurchaseListTask(int page, int pageSize, View progressView, ClaimSearchResultHandler 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 claims = null; + try { + Map options = new HashMap<>(); + options.put("page", page); + options.put("page_size", pageSize); + options.put("resolve", true); + + JSONObject result = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_PURCHASE_LIST, options); + JSONArray items = result.getJSONArray("items"); + claims = new ArrayList<>(); + for (int i = 0; i < items.length(); i++) { + Claim claim = Claim.fromJSONObject(items.getJSONObject(i).getJSONObject("claim")); + claims.add(claim); + + Lbry.addClaimToCache(claim); + } + } catch (ApiCallException | JSONException | ClassCastException ex) { + error = ex; + } + + return claims; + } + protected void onPostExecute(List claims) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (claims != null) { + handler.onSuccess(claims, claims.size() < pageSize); + } else { + handler.onError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/ResolveTask.java b/app/src/main/java/io/lbry/browser/tasks/claim/ResolveTask.java new file mode 100644 index 00000000..120c36b7 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/ResolveTask.java @@ -0,0 +1,54 @@ +package io.lbry.browser.tasks.claim; + +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.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class ResolveTask extends AsyncTask> { + private List urls; + private String connectionString; + private ClaimListResultHandler handler; + private View progressView; + private ApiCallException error; + + public ResolveTask(String url, String connectionString, View progressView, ClaimListResultHandler handler) { + this(Arrays.asList(url), connectionString, progressView, handler); + } + + public ResolveTask(List urls, String connectionString, View progressView, ClaimListResultHandler 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 Helper.filterInvalidReposts(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); + } + } + } + +} diff --git a/app/src/main/java/io/lbry/browser/tasks/claim/StreamRepostTask.java b/app/src/main/java/io/lbry/browser/tasks/claim/StreamRepostTask.java new file mode 100644 index 00000000..28ec4dcf --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/claim/StreamRepostTask.java @@ -0,0 +1,77 @@ +package io.lbry.browser.tasks.claim; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.math.BigDecimal; +import java.text.DecimalFormat; +import java.text.DecimalFormatSymbols; +import java.util.HashMap; +import java.util.Locale; +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 StreamRepostTask extends AsyncTask { + private String name; + private BigDecimal bid; + private String claimId; + private String channelId; + private View progressView; + private ClaimResultHandler handler; + private Exception error; + + public StreamRepostTask(String name, BigDecimal bid, String claimId, String channelId, View progressView, ClaimResultHandler handler) { + this.name = name; + this.bid = bid; + this.claimId = claimId; + this.channelId = channelId; + this.progressView = progressView; + this.handler = handler; + } + + protected Claim doInBackground(Void... params) { + Claim claimResult = null; + try { + Map options = new HashMap<>(); + options.put("name", name); + options.put("bid", new DecimalFormat(Helper.SDK_AMOUNT_FORMAT, new DecimalFormatSymbols(Locale.US)).format(bid.doubleValue())); + options.put("claim_id", claimId); + options.put("channel_id", channelId); + + JSONObject result = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_STREAM_REPOST, options); + if (result.has("outputs")) { + JSONArray outputs = result.getJSONArray("outputs"); + for (int i = 0; i < outputs.length(); i++) { + JSONObject output = outputs.getJSONObject(i); + if (output.has("claim_id") && output.has("claim_op")) { + claimResult = Claim.claimFromOutput(output); + break; + } + } + } + } catch (ApiCallException | ClassCastException | JSONException ex) { + error = ex; + } + + return claimResult; + } + + protected void onPostExecute(Claim result) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (result != null) { + handler.onSuccess(result); + } else { + handler.onError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/file/BulkDeleteFilesTask.java b/app/src/main/java/io/lbry/browser/tasks/file/BulkDeleteFilesTask.java new file mode 100644 index 00000000..b05ec9d8 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/file/BulkDeleteFilesTask.java @@ -0,0 +1,32 @@ +package io.lbry.browser.tasks.file; + +import android.os.AsyncTask; + +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.utils.Lbry; + +// Just run delete on the specified claim IDs (no need for a handler) +public class BulkDeleteFilesTask extends AsyncTask { + private List claimIds; + public BulkDeleteFilesTask(List claimIds) { + this.claimIds = claimIds; + } + protected Boolean doInBackground(Void... params) { + for (String claimId : claimIds) { + try { + Map options = new HashMap<>(); + options.put("claim_id", claimId); + options.put("delete_from_download_dir", true); + Lbry.genericApiCall(Lbry.METHOD_FILE_DELETE, options); + } catch (ApiCallException ex) { + // pass + } + } + return true; + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/file/DeleteFileTask.java b/app/src/main/java/io/lbry/browser/tasks/file/DeleteFileTask.java new file mode 100644 index 00000000..bdc50bc4 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/file/DeleteFileTask.java @@ -0,0 +1,43 @@ +package io.lbry.browser.tasks.file; + +import android.os.AsyncTask; + +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.utils.Lbry; + +public class DeleteFileTask extends AsyncTask { + private String claimId; + private Exception error; + private GenericTaskHandler handler; + + public DeleteFileTask(String claimId, GenericTaskHandler handler) { + this.claimId = claimId; + this.handler = handler; + } + + protected Boolean doInBackground(Void... params) { + try { + Map options = new HashMap<>(); + options.put("claim_id", claimId); + options.put("delete_from_download_dir", true); + return (boolean) Lbry.genericApiCall(Lbry.METHOD_FILE_DELETE, options); + } catch (ApiCallException ex) { + error = ex; + return false; + } + } + + protected void onPostExecute(Boolean result) { + if (handler != null) { + if (result) { + handler.onSuccess(); + } else { + handler.onError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/file/FileListTask.java b/app/src/main/java/io/lbry/browser/tasks/file/FileListTask.java new file mode 100644 index 00000000..0b442d77 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/file/FileListTask.java @@ -0,0 +1,60 @@ +package io.lbry.browser.tasks.file; + +import android.os.AsyncTask; +import android.view.View; + +import java.util.List; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.model.LbryFile; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class FileListTask extends AsyncTask> { + private String claimId; + private boolean downloads; + private int page; + private int pageSize; + private FileListResultHandler handler; + private View progressView; + private ApiCallException error; + + public FileListTask(int page, int pageSize, boolean downloads, View progressView, FileListResultHandler handler) { + this(null, progressView, handler); + this.page = page; + this.pageSize = pageSize; + this.downloads = downloads; + } + + 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, downloads, page, pageSize); + } 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, files.size() < pageSize); + } else { + handler.onError(error); + } + } + } + + public interface FileListResultHandler { + void onSuccess(List files, boolean hasReachedEnd); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/file/GetFileTask.java b/app/src/main/java/io/lbry/browser/tasks/file/GetFileTask.java new file mode 100644 index 00000000..bf708c82 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/file/GetFileTask.java @@ -0,0 +1,72 @@ +package io.lbry.browser.tasks.file; + +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.model.LbryFile; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class GetFileTask extends AsyncTask { + private String uri; + private boolean saveFile; + private View progressView; + private GetFileHandler handler; + private Exception error; + + public GetFileTask(String uri, boolean saveFile, View progressView, GetFileHandler handler) { + this.uri = uri; + this.saveFile = saveFile; + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + if (handler != null) { + handler.beforeStart(); + } + } + + protected LbryFile doInBackground(Void... params) { + LbryFile file = null; + try { + Map options = new HashMap<>(); + options.put("uri", uri); + options.put("save_file", saveFile); + JSONObject streamInfo = (JSONObject) Lbry.genericApiCall("get", options); + if (streamInfo.has("error")) { + throw new ApiCallException(Helper.getJSONString("error", "", streamInfo)); + } + + file = LbryFile.fromJSONObject(streamInfo); + } catch (ApiCallException ex) { + error = ex; + } + + return file; + } + + protected void onPostExecute(LbryFile file) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (file != null) { + handler.onSuccess(file, saveFile); + } else { + handler.onError(error, saveFile); + } + } + } + + public interface GetFileHandler { + void beforeStart(); + void onSuccess(LbryFile file, boolean saveFile); + void onError(Exception error, boolean saveFile); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/lbryinc/ChannelSubscribeTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/ChannelSubscribeTask.java new file mode 100644 index 00000000..89933903 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/ChannelSubscribeTask.java @@ -0,0 +1,85 @@ +package io.lbry.browser.tasks.lbryinc; + +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); + + if (!isUnsubscribing) { + Lbryio.addSubscription(subscription); + } else { + Lbryio.removeSubscription(subscription); + } + } catch (LbryioRequestException | LbryioResponseException | SQLiteException ex) { + error = ex; + return false; + } + + 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/lbryinc/ClaimRewardTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/ClaimRewardTask.java new file mode 100644 index 00000000..3a48c8e8 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/ClaimRewardTask.java @@ -0,0 +1,86 @@ +package io.lbry.browser.tasks.lbryinc; + +import android.content.Context; +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONObject; + +import java.text.DecimalFormat; +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.R; +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.Lbryio; + +public class ClaimRewardTask extends AsyncTask { + + private Context context; + private String type; + private String claimCode; + private View progressView; + private double amountClaimed; + private ClaimRewardHandler handler; + private Exception error; + + public ClaimRewardTask(String type, String claimCode, View progressView, Context context, ClaimRewardHandler handler) { + this.type = type; + this.claimCode = claimCode; + this.progressView = progressView; + this.context = context; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + + public String doInBackground(Void... params) { + String message = null; + try { + // Get a new wallet address for the reward + String address = (String) Lbry.genericApiCall(Lbry.METHOD_ADDRESS_UNUSED); + Map options = new HashMap<>(); + options.put("reward_type", type); + options.put("wallet_address", address); + if (!Helper.isNullOrEmpty(claimCode)) { + options.put("claim_code", claimCode); + + } + JSONObject reward = (JSONObject) Lbryio.parseResponse( + Lbryio.call("reward", "claim", options, Helper.METHOD_POST, null)); + amountClaimed = Helper.getJSONDouble("reward_amount", 0, reward); + String defaultMessage = context != null ? + context.getResources().getQuantityString( + R.plurals.claim_reward_message, + amountClaimed == 1 ? 1 : 2, + new DecimalFormat(Helper.LBC_CURRENCY_FORMAT_PATTERN).format(amountClaimed)) : ""; + message = Helper.getJSONString("reward_notification", defaultMessage, reward); + } catch (ApiCallException | LbryioRequestException | LbryioResponseException ex) { + error = ex; + } + + return message; + } + + protected void onPostExecute(String message) { + Helper.setViewVisibility(progressView, View.INVISIBLE); + if (handler != null) { + if (message != null) { + handler.onSuccess(amountClaimed, message); + } else { + handler.onError(error); + } + } + } + + public interface ClaimRewardHandler { + void onSuccess(double amountClaimed, String message); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchCurrentUserTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchCurrentUserTask.java new file mode 100644 index 00000000..3be8e03e --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchCurrentUserTask.java @@ -0,0 +1,38 @@ +package io.lbry.browser.tasks.lbryinc; + +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/lbryinc/FetchInviteStatusTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchInviteStatusTask.java new file mode 100644 index 00000000..d56f7dbd --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchInviteStatusTask.java @@ -0,0 +1,72 @@ +package io.lbry.browser.tasks.lbryinc; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.model.lbryinc.Invitee; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbryio; + +public class FetchInviteStatusTask extends AsyncTask> { + private FetchInviteStatusHandler handler; + private View progressView; + private Exception error; + + public FetchInviteStatusTask(View progressView, FetchInviteStatusHandler handler) { + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + + protected List doInBackground(Void... params) { + List invitees = null; + try { + JSONObject status = (JSONObject) Lbryio.parseResponse(Lbryio.call("user", "invite_status", null, null)); + JSONArray inviteesArray = status.getJSONArray("invitees"); + invitees = new ArrayList<>(); + for (int i = 0; i < inviteesArray.length(); i++) { + JSONObject inviteeObject = inviteesArray.getJSONObject(i); + Invitee invitee = new Invitee(); + invitee.setEmail(Helper.getJSONString("email", null, inviteeObject)); + invitee.setInviteRewardClaimable(Helper.getJSONBoolean("invite_reward_claimable", false, inviteeObject)); + invitee.setInviteRewardClaimed(Helper.getJSONBoolean("invite_reward_claimed", false, inviteeObject)); + + if (!Helper.isNullOrEmpty(invitee.getEmail())) { + invitees.add(invitee); + } + } + } catch (ClassCastException | LbryioRequestException | LbryioResponseException | JSONException ex) { + error = ex; + } + + return invitees; + } + + protected void onPostExecute(List invitees) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (invitees != null) { + handler.onSuccess(invitees); + } else { + handler.onError(error); + } + } + } + + public interface FetchInviteStatusHandler { + void onSuccess(List invitees); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchReferralCodeTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchReferralCodeTask.java new file mode 100644 index 00000000..af15d83a --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchReferralCodeTask.java @@ -0,0 +1,57 @@ +package io.lbry.browser.tasks.lbryinc; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONArray; +import org.json.JSONException; + +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 FetchReferralCodeTask extends AsyncTask { + private FetchReferralCodeHandler handler; + private View progressView; + private Exception error; + + public FetchReferralCodeTask(View progressView, FetchReferralCodeHandler handler) { + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + + protected String doInBackground(Void... params) { + String referralCode = null; + try { + JSONArray results = (JSONArray) Lbryio.parseResponse(Lbryio.call("user_referral_code", "list", null, null)); + if (results.length() > 0) { + referralCode = results.getString(0); + } + } catch (ClassCastException | LbryioRequestException | LbryioResponseException | JSONException ex) { + error = ex; + } + + return referralCode; + } + + protected void onPostExecute(String referralCode) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (!Helper.isNullOrEmpty(referralCode)) { + handler.onSuccess(referralCode); + } else { + handler.onError(error); + } + } + } + + public interface FetchReferralCodeHandler { + void onSuccess(String referralCode); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchRewardsTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchRewardsTask.java new file mode 100644 index 00000000..b9f18067 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchRewardsTask.java @@ -0,0 +1,66 @@ +package io.lbry.browser.tasks.lbryinc; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONArray; +import org.json.JSONException; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.model.lbryinc.Reward; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbryio; + +public class FetchRewardsTask extends AsyncTask> { + private FetchRewardsHandler handler; + private View progressView; + private Exception error; + + public FetchRewardsTask(View progressView, FetchRewardsHandler handler) { + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + + protected List doInBackground(Void... params) { + List rewards = null; + try { + Map options = new HashMap<>(); + options.put("multiple_rewards_per_type", "true"); + JSONArray results = (JSONArray) Lbryio.parseResponse(Lbryio.call("reward", "list", null, null)); + rewards = new ArrayList<>(); + for (int i = 0; i < results.length(); i++) { + rewards.add(Reward.fromJSONObject(results.getJSONObject(i))); + } + } catch (ClassCastException | LbryioRequestException | LbryioResponseException | JSONException ex) { + error = ex; + } + + return rewards; + } + + protected void onPostExecute(List rewards) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (rewards != null) { + handler.onSuccess(rewards); + } else { + handler.onError(error); + } + } + } + + public interface FetchRewardsHandler { + void onSuccess(List rewards); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchStatCountTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchStatCountTask.java new file mode 100644 index 00000000..1f8bb465 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchStatCountTask.java @@ -0,0 +1,73 @@ +package io.lbry.browser.tasks.lbryinc; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONArray; +import org.json.JSONException; + +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 FetchStatCountTask extends AsyncTask { + public static final int STAT_VIEW_COUNT = 1; + public static final int STAT_SUB_COUNT = 2; + + private String claimId; + private int stat; + private FetchStatCountHandler handler; + private View progressView; + private Exception error; + + public FetchStatCountTask(int stat, String claimId, View progressView, FetchStatCountHandler handler) { + this.stat = stat; + this.claimId = claimId; + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + + protected Integer doInBackground(Void... params) { + int count = -1; + try { + if (stat != STAT_VIEW_COUNT && stat != STAT_SUB_COUNT) { + throw new LbryioRequestException("Invalid stat count specified."); + } + + JSONArray results = (JSONArray) + Lbryio.parseResponse(Lbryio.call( + stat == STAT_VIEW_COUNT ? "file" : "subscription", + stat == STAT_VIEW_COUNT ? "view_count" : "sub_count", + Lbryio.buildSingleParam("claim_id", claimId), + Helper.METHOD_GET, null)); + if (results.length() > 0) { + count = results.getInt(0); + } + } catch (ClassCastException | LbryioRequestException | LbryioResponseException | JSONException ex) { + error = ex; + } + + return count; + } + + protected void onPostExecute(Integer count) { + Helper.setViewVisibility(progressView, View.GONE); + if (handler != null) { + if (count > -1) { + handler.onSuccess(count); + } else { + handler.onError(error); + } + } + } + + public interface FetchStatCountHandler { + void onSuccess(int count); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchSubscriptionsTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchSubscriptionsTask.java new file mode 100644 index 00000000..da886beb --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/FetchSubscriptionsTask.java @@ -0,0 +1,86 @@ +package io.lbry.browser.tasks.lbryinc; + +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 (ClassCastException | LbryioRequestException | LbryioResponseException | JSONException | IllegalStateException ex) { + error = ex; + return null; + } + + 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/lbryinc/InviteByEmailTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/InviteByEmailTask.java new file mode 100644 index 00000000..e2709862 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/InviteByEmailTask.java @@ -0,0 +1,55 @@ +package io.lbry.browser.tasks.lbryinc; + +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 InviteByEmailTask extends AsyncTask { + private String email; + private View progressView; + private GenericTaskHandler handler; + private Exception error; + + public InviteByEmailTask(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", "invite", 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/lbryinc/LogFileViewTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/LogFileViewTask.java new file mode 100644 index 00000000..c0055a7c --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/LogFileViewTask.java @@ -0,0 +1,55 @@ +package io.lbry.browser.tasks.lbryinc; + +import android.os.AsyncTask; + +import java.io.IOException; +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.Executor; + +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.utils.Lbryio; +import okhttp3.Response; + +public class LogFileViewTask extends AsyncTask { + private String uri; + private Claim claim; + private Exception error; + private GenericTaskHandler handler; + private long timeToStart; + + public LogFileViewTask(String uri, Claim claim, long timeToStart, GenericTaskHandler handler) { + this.uri = uri; + this.claim = claim; + this.timeToStart = timeToStart; + this.handler = handler; + } + protected Boolean doInBackground(Void... params) { + try { + Map options = new HashMap<>(); + options.put("uri", uri); + options.put("claim_id", claim.getClaimId()); + options.put("outpoint", String.format("%s:%d", claim.getTxid(), claim.getNout())); + if (timeToStart > 0) { + options.put("time_to_start", String.valueOf(timeToStart)); + } + Lbryio.call("file", "view", options, null).close(); + } catch (LbryioRequestException | LbryioResponseException ex) { + error = ex; + return false; + } + return true; + } + protected void onPostExecute(Boolean result) { + if (handler != null) { + if (result) { + handler.onSuccess(); + } else { + handler.onError(error); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/lbryinc/LogPublishTask.java b/app/src/main/java/io/lbry/browser/tasks/lbryinc/LogPublishTask.java new file mode 100644 index 00000000..f807f360 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/lbryinc/LogPublishTask.java @@ -0,0 +1,35 @@ +package io.lbry.browser.tasks.lbryinc; + +import android.os.AsyncTask; + +import java.io.IOException; +import java.util.HashMap; +import java.util.Map; + +import io.lbry.browser.exceptions.LbryioRequestException; +import io.lbry.browser.exceptions.LbryioResponseException; +import io.lbry.browser.model.Claim; +import io.lbry.browser.utils.Lbryio; +import okhttp3.Response; + +public class LogPublishTask extends AsyncTask { + private Claim claimResult; + public LogPublishTask(Claim claimResult) { + this.claimResult = claimResult; + } + protected Void doInBackground(Void... params) { + try { + Map options = new HashMap<>(); + options.put("uri", claimResult.getPermanentUrl()); + options.put("claim_id", claimResult.getClaimId()); + options.put("outpoint", String.format("%s:%d", claimResult.getTxid(), claimResult.getNout())); + if (claimResult.getSigningChannel() != null) { + options.put("channel_claim_id", claimResult.getSigningChannel().getClaimId()); + } + Lbryio.call("event", "publish", options, null).close(); + } catch (LbryioRequestException | LbryioResponseException ex) { + // pass + } + return null; + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/localdata/FetchRecentUrlHistoryTask.java b/app/src/main/java/io/lbry/browser/tasks/localdata/FetchRecentUrlHistoryTask.java new file mode 100644 index 00000000..852e87ae --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/localdata/FetchRecentUrlHistoryTask.java @@ -0,0 +1,37 @@ +package io.lbry.browser.tasks.localdata; + +import android.database.sqlite.SQLiteDatabase; +import android.database.sqlite.SQLiteException; +import android.os.AsyncTask; + +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.model.UrlSuggestion; + +public class FetchRecentUrlHistoryTask extends AsyncTask> { + private DatabaseHelper dbHelper; + private FetchRecentUrlHistoryHandler handler; + public FetchRecentUrlHistoryTask(DatabaseHelper dbHelper, FetchRecentUrlHistoryHandler handler) { + this.dbHelper = dbHelper; + this.handler = handler; + } + protected List doInBackground(Void... params) { + try { + SQLiteDatabase db = dbHelper.getReadableDatabase(); + return DatabaseHelper.getRecentHistory(db); + } catch (SQLiteException ex) { + return new ArrayList<>(); + } + } + protected void onPostExecute(List recentHistory) { + if (handler != null) { + handler.onSuccess(recentHistory); + } + } + + public interface FetchRecentUrlHistoryHandler { + void onSuccess(List recentHistory); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/localdata/FetchViewHistoryTask.java b/app/src/main/java/io/lbry/browser/tasks/localdata/FetchViewHistoryTask.java new file mode 100644 index 00000000..fd0d4b21 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/localdata/FetchViewHistoryTask.java @@ -0,0 +1,46 @@ +package io.lbry.browser.tasks.localdata; + +import android.database.sqlite.SQLiteDatabase; +import android.database.sqlite.SQLiteException; +import android.os.AsyncTask; + +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.Date; +import java.util.List; + +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.model.ViewHistory; +import io.lbry.browser.utils.Helper; + +public class FetchViewHistoryTask extends AsyncTask> { + private DatabaseHelper dbHelper; + private FetchViewHistoryHandler handler; + private int pageSize; + private Date lastDate; + public FetchViewHistoryTask(Date lastDate, int pageSize, DatabaseHelper dbHelper, FetchViewHistoryHandler handler) { + this.lastDate = lastDate; + this.pageSize = pageSize; + this.dbHelper = dbHelper; + this.handler = handler; + } + protected List doInBackground(Void... params) { + try { + SQLiteDatabase db = dbHelper.getReadableDatabase(); + return DatabaseHelper.getViewHistory( + lastDate == null ? null : new SimpleDateFormat(Helper.ISO_DATE_FORMAT_PATTERN).format(lastDate), pageSize, db); + } catch (SQLiteException ex) { + return new ArrayList<>(); + } + } + protected void onPostExecute(List history) { + if (handler != null) { + handler.onSuccess(history, history.size() < pageSize); + } + } + + public interface FetchViewHistoryHandler { + void onSuccess(List history, boolean hasReachedEnd); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/localdata/LoadGalleryItemsTask.java b/app/src/main/java/io/lbry/browser/tasks/localdata/LoadGalleryItemsTask.java new file mode 100644 index 00000000..5676f072 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/localdata/LoadGalleryItemsTask.java @@ -0,0 +1,136 @@ +package io.lbry.browser.tasks.localdata; + +import android.content.ContentResolver; +import android.content.Context; +import android.database.Cursor; +import android.database.sqlite.SQLiteException; +import android.graphics.Bitmap; +import android.graphics.BitmapFactory; +import android.net.Uri; +import android.os.AsyncTask; +import android.provider.MediaStore; +import android.util.Log; +import android.view.View; + +import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.model.GalleryItem; +import io.lbry.browser.utils.Helper; + +public class LoadGalleryItemsTask extends AsyncTask> { + private static final String TAG = "LoadGalleryItemsTask"; + private LoadGalleryHandler handler; + private View progressView; + private Context context; + + public LoadGalleryItemsTask(View progressView, Context context, LoadGalleryHandler handler) { + this.progressView = progressView; + this.context = context; + this.handler = handler; + } + + protected void onPreExecute(Void... params) { + Helper.setViewVisibility(progressView, View.VISIBLE); + } + + protected List doInBackground(Void... params) { + List items = new ArrayList<>(); + List itemsWithThumbnails = new ArrayList<>(); + Cursor cursor = null; + if (context != null) { + ContentResolver resolver = context.getContentResolver(); + try { + String[] projection = { + MediaStore.MediaColumns._ID, + MediaStore.MediaColumns.DATA, + MediaStore.MediaColumns.DISPLAY_NAME, + MediaStore.MediaColumns.MIME_TYPE, + MediaStore.Video.Media.DURATION + }; + cursor = resolver.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); + + GalleryItem item = new GalleryItem(); + item.setId(cursor.getString(idColumn)); + item.setName(cursor.getString(nameColumn)); + item.setType(cursor.getString(typeColumn)); + item.setFilePath(cursor.getString(pathColumn)); + item.setDuration(cursor.getLong(durationColumn)); + items.add(item); + } + } catch (SQLiteException ex) { + + // failed to load videos. log and pass + Log.e(TAG, ex.getMessage(), ex); + } finally { + Helper.closeCursor(cursor); + } + + // load (or generate) thumbnail for each item + for (GalleryItem item : items) { + String id = item.getId(); + 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) { + item.setThumbnailPath(file.getAbsolutePath()); + itemsWithThumbnails.add(item); + publishProgress(item); + } + } + } + + return itemsWithThumbnails; + } + + protected void onProgressUpdate(GalleryItem... items) { + if (handler != null) { + for (GalleryItem item : items) { + handler.onItemLoaded(item); + } + } + } + + protected void onPostExecute(List items) { + if (handler != null) { + handler.onAllItemsLoaded(items); + } + } + + public interface LoadGalleryHandler { + void onItemLoaded(GalleryItem item); + void onAllItemsLoaded(List items); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/localdata/SaveUrlHistoryTask.java b/app/src/main/java/io/lbry/browser/tasks/localdata/SaveUrlHistoryTask.java new file mode 100644 index 00000000..a5778984 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/localdata/SaveUrlHistoryTask.java @@ -0,0 +1,49 @@ +package io.lbry.browser.tasks.localdata; + +import android.content.Context; +import android.database.sqlite.SQLiteDatabase; +import android.os.AsyncTask; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.tasks.GenericTaskHandler; + +public class SaveUrlHistoryTask extends AsyncTask { + private DatabaseHelper dbHelper; + private UrlSuggestion suggestion; + private SaveUrlHistoryHandler handler; + private Exception error; + + public SaveUrlHistoryTask(UrlSuggestion suggestion, DatabaseHelper dbHelper, SaveUrlHistoryHandler handler) { + this.suggestion = suggestion; + this.dbHelper = dbHelper; + this.handler = handler; + + } + protected Boolean doInBackground(Void... params) { + try { + SQLiteDatabase db = dbHelper.getWritableDatabase(); + DatabaseHelper.createOrUpdateUrlHistoryItem(suggestion.getText(), suggestion.getUri().toString(), suggestion.getType(), db); + } catch (Exception ex) { + error = ex; + return false; + } + + return true; + } + protected void onPostExecute(Boolean result) { + if (handler != null) { + if (result) { + handler.onSuccess(suggestion); + } else { + handler.onError(error); + } + } + } + + public interface SaveUrlHistoryHandler { + void onSuccess(UrlSuggestion item); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/localdata/SaveViewHistoryTask.java b/app/src/main/java/io/lbry/browser/tasks/localdata/SaveViewHistoryTask.java new file mode 100644 index 00000000..bb7e0f64 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/localdata/SaveViewHistoryTask.java @@ -0,0 +1,45 @@ +package io.lbry.browser.tasks.localdata; + +import android.database.sqlite.SQLiteDatabase; +import android.os.AsyncTask; + +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.model.ViewHistory; + +public class SaveViewHistoryTask extends AsyncTask { + private DatabaseHelper dbHelper; + private ViewHistory history; + private SaveViewHistoryHandler handler; + private Exception error; + + public SaveViewHistoryTask(ViewHistory history, DatabaseHelper dbHelper, SaveViewHistoryHandler handler) { + this.history = history; + this.dbHelper = dbHelper; + this.handler = handler; + } + protected Boolean doInBackground(Void... params) { + try { + SQLiteDatabase db = dbHelper.getWritableDatabase(); + DatabaseHelper.createOrUpdateViewHistoryItem(history, db); + } catch (Exception ex) { + error = ex; + return false; + } + + return true; + } + protected void onPostExecute(Boolean result) { + if (handler != null) { + if (result) { + handler.onSuccess(history); + } else { + handler.onError(error); + } + } + } + + public interface SaveViewHistoryHandler { + void onSuccess(ViewHistory item); + 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/verification/PhoneNewVerifyTask.java b/app/src/main/java/io/lbry/browser/tasks/verification/PhoneNewVerifyTask.java new file mode 100644 index 00000000..15730464 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/verification/PhoneNewVerifyTask.java @@ -0,0 +1,65 @@ +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 PhoneNewVerifyTask extends AsyncTask { + private String countryCode; + private String phoneNumber; + private String verificationCode; + private View progressView; + private GenericTaskHandler handler; + private Exception error; + + public PhoneNewVerifyTask(String countryCode, String phoneNumber, String verificationCode, View progressView, GenericTaskHandler handler) { + this.countryCode = countryCode; + this.phoneNumber = phoneNumber; + this.verificationCode = verificationCode; + 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 { + boolean isVerify = !Helper.isNullOrEmpty(verificationCode); + Map options = new HashMap<>(); + options.put("country_code", countryCode); + options.put("phone_number", phoneNumber.replace(" ", "").replace("-", "")); + if (isVerify) { + options.put("verification_code", verificationCode); + } + + String action = isVerify ? "phone_number_confirm" : "phone_number_new"; + Lbryio.parseResponse(Lbryio.call("user", action, options, Helper.METHOD_POST, null)); + } catch (LbryioResponseException | 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); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/verification/PhoneResendTask.java b/app/src/main/java/io/lbry/browser/tasks/verification/PhoneResendTask.java new file mode 100644 index 00000000..247646b0 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/verification/PhoneResendTask.java @@ -0,0 +1,4 @@ +package io.lbry.browser.tasks.verification; + +public class PhoneResendTask { +} 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/LoadSharedUserStateTask.java b/app/src/main/java/io/lbry/browser/tasks/wallet/LoadSharedUserStateTask.java new file mode 100644 index 00000000..ac941123 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/LoadSharedUserStateTask.java @@ -0,0 +1,138 @@ +package io.lbry.browser.tasks.wallet; + +import android.content.Context; +import android.database.sqlite.SQLiteDatabase; +import android.database.sqlite.SQLiteException; +import android.os.AsyncTask; + +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.ApiCallException; +import io.lbry.browser.exceptions.LbryUriException; +import io.lbry.browser.model.Tag; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryUri; + +/* + version: '0.1', + value: { + subscriptions?: Array, + tags?: Array, + blocked?: Array, + settings?: any, + app_welcome_version?: number, + sharing_3P?: boolean, + }, + */ +public class LoadSharedUserStateTask extends AsyncTask { + private static final String KEY = "shared"; + + private Context context; + private LoadSharedUserStateHandler handler; + private Exception error; + + private List subscriptions; + private List followedTags; + + public LoadSharedUserStateTask(Context context, LoadSharedUserStateHandler handler) { + this.context = context; + this.handler = handler; + } + + protected Boolean doInBackground(Void... params) { + // data to save + // current subscriptions + // Get the previous saved state + try { + SQLiteDatabase db = null; + JSONObject result = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_PREFERENCE_GET, Lbry.buildSingleParam("key", KEY)); + if (result != null) { + if (context instanceof MainActivity) { + db = ((MainActivity) context).getDbHelper().getWritableDatabase(); + } + + JSONObject shared = result.getJSONObject("shared"); + if (shared.has("type") + && "object".equalsIgnoreCase(shared.getString("type")) + && shared.has("value")) { + JSONObject value = shared.getJSONObject("value"); + + JSONArray subscriptionUrls = + value.has("subscriptions") && !value.isNull("subscriptions") ? value.getJSONArray("subscriptions") : null; + JSONArray tags = + value.has("tags") && !value.isNull("tags") ? value.getJSONArray("tags") : null; + + if (subscriptionUrls != null) { + subscriptions = new ArrayList<>(); + for (int i = 0; i < subscriptionUrls.length(); i++) { + String url = subscriptionUrls.getString(i); + try { + LbryUri uri = LbryUri.parse(LbryUri.normalize(url)); + Subscription subscription = new Subscription(); + subscription.setChannelName(uri.getChannelName()); + subscription.setUrl(url); + subscriptions.add(subscription); + if (db != null) { + DatabaseHelper.createOrUpdateSubscription(subscription, db); + } + } catch (LbryUriException | SQLiteException | IllegalStateException ex) { + // pass + } + } + } + + if (tags != null) { + if (db != null && tags.length() > 0) { + DatabaseHelper.setAllTagsUnfollowed(db); + } + + followedTags = new ArrayList<>(); + for (int i = 0; i < tags.length(); i++) { + String tagName = tags.getString(i); + Tag tag = new Tag(tagName); + tag.setFollowed(true); + followedTags.add(tag); + + try { + if (db != null) { + DatabaseHelper.createOrUpdateTag(tag, db); + } + } catch (SQLiteException ex) { + // pass + } + } + } + } + } + + return true; + } catch (ApiCallException | JSONException ex) { + // failed + error = ex; + } + return false; + } + + protected void onPostExecute(Boolean result) { + if (handler != null) { + if (result) { + handler.onSuccess(subscriptions, followedTags); + } else { + handler.onError(error); + } + } + } + + public interface LoadSharedUserStateHandler { + void onSuccess(List subscriptions, List followedTags); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/SaveSharedUserStateTask.java b/app/src/main/java/io/lbry/browser/tasks/wallet/SaveSharedUserStateTask.java new file mode 100644 index 00000000..9a0559af --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/SaveSharedUserStateTask.java @@ -0,0 +1,108 @@ +package io.lbry.browser.tasks.wallet; + +import android.os.AsyncTask; + +import org.json.JSONException; +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.Lbryio; + +/* + version: '0.1', + value: { + subscriptions?: Array, + tags?: Array, + blocked?: Array, + settings?: any, + app_welcome_version?: number, + sharing_3P?: boolean, + }, + */ +public class SaveSharedUserStateTask extends AsyncTask { + private static final String KEY = "shared"; + private static final String VERSION = "0.1"; + private SaveSharedUserStateHandler handler; + private Exception error; + + public SaveSharedUserStateTask(SaveSharedUserStateHandler handler) { + this.handler = handler; + } + + protected Boolean doInBackground(Void... params) { + // data to save + // current subscriptions + List subscriptionUrls = new ArrayList<>(); + for (Subscription subscription : Lbryio.subscriptions) { + subscriptionUrls.add(subscription.getUrl()); + } + + // followed tags + List followedTags = Helper.getTagsForTagObjects(Lbry.followedTags); + + // Get the previous saved state + try { + boolean isExistingValid = false; + JSONObject sharedObject = null; + JSONObject result = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_PREFERENCE_GET, Lbry.buildSingleParam("key", KEY)); + if (result != null) { + JSONObject shared = result.getJSONObject("shared"); + if (shared.has("type") + && "object".equalsIgnoreCase(shared.getString("type")) + && shared.has("value")) { + isExistingValid = true; + JSONObject value = shared.getJSONObject("value"); + value.put("subscriptions", Helper.jsonArrayFromList(subscriptionUrls)); + value.put("tags", Helper.jsonArrayFromList(followedTags)); + sharedObject = shared; + } + } + + if (!isExistingValid) { + // build a new object + JSONObject value = new JSONObject(); + value.put("subscriptions", Helper.jsonArrayFromList(subscriptionUrls)); + value.put("tags", Helper.jsonArrayFromList(followedTags)); + + sharedObject = new JSONObject(); + sharedObject.put("type", "object"); + sharedObject.put("value", value); + sharedObject.put("version", VERSION); + } + + Map options = new HashMap<>(); + options.put("key", KEY); + options.put("value", sharedObject.toString()); + Lbry.genericApiCall(Lbry.METHOD_PREFERENCE_SET, options); + + return true; + } catch (ApiCallException | JSONException ex) { + // failed + error = ex; + } + return false; + } + + protected void onPostExecute(Boolean result) { + if (handler != null) { + if (result) { + handler.onSuccess(); + } else { + handler.onError(error); + } + } + } + + public interface SaveSharedUserStateHandler { + void onSuccess(); + void onError(Exception error); + } +} diff --git a/app/src/main/java/io/lbry/browser/tasks/wallet/SupportCreateTask.java b/app/src/main/java/io/lbry/browser/tasks/wallet/SupportCreateTask.java new file mode 100644 index 00000000..d571898a --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/SupportCreateTask.java @@ -0,0 +1,65 @@ +package io.lbry.browser.tasks.wallet; + +import android.os.AsyncTask; +import android.view.View; + +import java.math.BigDecimal; +import java.text.DecimalFormat; +import java.text.DecimalFormatSymbols; +import java.util.HashMap; +import java.util.Locale; +import java.util.Map; + +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; + +public class SupportCreateTask extends AsyncTask { + private String claimId; + private BigDecimal amount; + private boolean tip; + private View progressView; + private GenericTaskHandler handler; + private Exception error; + + public SupportCreateTask(String claimId, BigDecimal amount, boolean tip, View progressView, GenericTaskHandler handler) { + this.claimId = claimId; + this.amount = amount; + this.tip = tip; + this.progressView = progressView; + this.handler = handler; + } + + protected void onPreExecute() { + if (handler != null) { + handler.beforeStart(); + } + Helper.setViewVisibility(progressView, View.VISIBLE); + } + protected Boolean doInBackground(Void... params) { + try { + Map options = new HashMap<>(); + options.put("claim_id", claimId); + options.put("amount", new DecimalFormat(Helper.SDK_AMOUNT_FORMAT, new DecimalFormatSymbols(Locale.US)).format(amount.doubleValue())); + options.put("tip", tip); + Lbry.genericApiCall(Lbry.METHOD_SUPPORT_CREATE, 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); + } + } + } +} 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..51572f49 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/tasks/wallet/SyncApplyTask.java @@ -0,0 +1,75 @@ +package io.lbry.browser.tasks.wallet; + +import android.os.AsyncTask; +import android.view.View; + +import org.json.JSONException; +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, String password, SyncTaskHandler handler) { + this.fetch = fetch; + this.password = password; + 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..c6c85b0f --- /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 not 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("password", Helper.isNullOrEmpty(password) ? "" : password); + 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..1f5c888f --- /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 { + 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..a6e6bd3e --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/BaseFragment.java @@ -0,0 +1,60 @@ +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 boolean shouldHideGlobalPlayer() { + return false; + } + + public boolean shouldSuspendGlobalPlayer() { + return false; + } + + public void onStart() { + super.onStart(); + if (shouldSuspendGlobalPlayer()) { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity.suspendGlobalPlayer(context); + } + } + } + + public void onStop() { + if (shouldSuspendGlobalPlayer()) { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity.resumeGlobalPlayer(context); + } + } + super.onStop(); + } + + public void onResume() { + super.onResume(); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.setSelectedMenuItemForFragment(this); + + if (shouldHideGlobalPlayer()) { + activity.hideGlobalNowPlaying(); + } else { + activity.checkNowPlaying(); + } + } + } +} 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..3e3e804a --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/channel/ChannelContentFragment.java @@ -0,0 +1,329 @@ +package io.lbry.browser.ui.channel; + +import android.content.Context; +import android.content.SharedPreferences; +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.preference.PreferenceManager; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import org.json.JSONException; +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +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.listener.DownloadActionListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.LbryFile; +import io.lbry.browser.tasks.claim.ClaimSearchResultHandler; +import io.lbry.browser.tasks.claim.ClaimSearchTask; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.Predefined; +import lombok.Setter; + +public class ChannelContentFragment extends Fragment implements DownloadActionListener, SharedPreferences.OnSharedPreferenceChangeListener { + + @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 View noContentView; + 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); + noContentView = root.findViewById(R.id.channel_content_no_claim_search_content); + + 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(); + Context context = getContext(); + if (context != null) { + ((MainActivity) context).addDownloadActionListener(this); + } + PreferenceManager.getDefaultSharedPreferences(context).registerOnSharedPreferenceChangeListener(this); + fetchClaimSearchContent(); + } + + public void onPause() { + Context context = getContext(); + if (context != null) { + ((MainActivity) context).removeDownloadActionListener(this); + } + PreferenceManager.getDefaultSharedPreferences(context).registerOnSharedPreferenceChangeListener(this); + super.onPause(); + } + + private Map buildContentOptions() { + Context context = getContext(); + boolean canShowMatureContent = false; + if (context != null) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(context); + canShowMatureContent = sp.getBoolean(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT, false); + } + + return Lbry.buildClaimSearchOptions( + null, + null, + canShowMatureContent ? null : new ArrayList<>(Predefined.MATURE_TAGS), + 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; + Helper.setViewVisibility(noContentView, View.GONE); + Map claimSearchOptions = buildContentOptions(); + contentClaimSearchTask = new ClaimSearchTask(claimSearchOptions, Lbry.LBRY_TV_CONNECTION_STRING, getLoadingView(), new 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) { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (claim.getName().startsWith("@")) { + // channel claim + activity.openChannelClaim(claim); + } else { + activity.openFileClaim(claim); + } + } + } + }); + } else { + contentListAdapter.addItems(claims); + } + + if (contentList != null && contentList.getAdapter() == null) { + contentList.setAdapter(contentListAdapter); + } + + contentHasReachedEnd = hasReachedEnd; + contentClaimSearchLoading = false; + checkNoContent(); + } + + @Override + public void onError(Exception error) { + contentClaimSearchLoading = false; + checkNoContent(); + } + }); + contentClaimSearchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void checkNoContent() { + boolean noContent = contentListAdapter == null || contentListAdapter.getItemCount() == 0; + Helper.setViewVisibility(noContentView, noContent ? View.VISIBLE : View.GONE); + } + + public void onSharedPreferenceChanged(SharedPreferences sp, String key) { + if (key.equalsIgnoreCase(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT)) { + fetchClaimSearchContent(true); + } + } + + public void onDownloadAction(String downloadAction, String uri, String outpoint, String fileInfoJson, double progress) { + if ("abort".equals(downloadAction)) { + if (contentListAdapter != null) { + contentListAdapter.clearFileForClaimOrUrl(outpoint, uri); + } + return; + } + + try { + JSONObject fileInfo = new JSONObject(fileInfoJson); + LbryFile claimFile = LbryFile.fromJSONObject(fileInfo); + String claimId = claimFile.getClaimId(); + if (contentListAdapter != null) { + contentListAdapter.updateFileForClaimByIdOrUrl(claimFile, claimId, uri); + } + } catch (JSONException ex) { + // invalid file info for download + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/channel/ChannelFormFragment.java b/app/src/main/java/io/lbry/browser/ui/channel/ChannelFormFragment.java new file mode 100644 index 00000000..441ba9b1 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/channel/ChannelFormFragment.java @@ -0,0 +1,697 @@ +package io.lbry.browser.ui.channel; + +import android.Manifest; +import android.content.Context; +import android.content.Intent; +import android.graphics.Color; +import android.net.Uri; +import android.os.AsyncTask; +import android.os.Bundle; +import android.text.Editable; +import android.text.TextWatcher; +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.appcompat.app.ActionBar; +import androidx.core.content.ContextCompat; +import androidx.core.widget.NestedScrollView; +import androidx.recyclerview.widget.RecyclerView; + +import com.bumptech.glide.Glide; +import com.bumptech.glide.request.RequestOptions; +import com.google.android.flexbox.FlexboxLayoutManager; +import com.google.android.material.button.MaterialButton; +import com.google.android.material.snackbar.Snackbar; +import com.google.android.material.textfield.TextInputEditText; + +import java.io.File; +import java.math.BigDecimal; +import java.util.ArrayList; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.BuildConfig; +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.TagListAdapter; +import io.lbry.browser.listener.FilePickerListener; +import io.lbry.browser.listener.StoragePermissionListener; +import io.lbry.browser.listener.WalletBalanceListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.NavMenuItem; +import io.lbry.browser.model.Tag; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.tasks.UpdateSuggestedTagsTask; +import io.lbry.browser.tasks.UploadImageTask; +import io.lbry.browser.tasks.claim.ChannelCreateUpdateTask; +import io.lbry.browser.tasks.claim.ClaimResultHandler; +import io.lbry.browser.tasks.lbryinc.LogPublishTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; +import lombok.Getter; + +public class ChannelFormFragment extends BaseFragment implements + FilePickerListener, StoragePermissionListener, TagListAdapter.TagClickListener, WalletBalanceListener { + + private static final int SUGGESTED_LIMIT = 8; + + @Getter + private boolean saveInProgress; + private boolean uploading; + private Claim currentClaim; + private boolean editFieldsLoaded; + private boolean editMode; + private View linkCancel; + private TextView linkShowOptional; + private MaterialButton buttonSave; + + private NestedScrollView scrollView; + private View inlineBalanceContainer; + private TextView inlineBalanceValue; + private View uploadProgress; + private View containerOptionalFields; + private ImageView imageCover; + private ImageView imageThumbnail; + private TextInputEditText inputTitle; + private TextInputEditText inputChannelName; + private TextInputEditText inputDeposit; + private TextInputEditText inputDescription; + private TextInputEditText inputWebsite; + private TextInputEditText inputEmail; + + private TextInputEditText inputTagFilter; + private RecyclerView addedTagsList; + private RecyclerView suggestedTagsList; + private TagListAdapter addedTagsAdapter; + private TagListAdapter suggestedTagsAdapter; + private View noTagsView; + private View noTagResultsView; + + private View coverEditArea; + private View iconContainer; + private View channelSaveProgress; + + private boolean launchCoverSelectPending; + private boolean launchThumbnailSelectPending; + private boolean coverFilePickerActive; + private boolean thumbnailFilePickerActive; + + private String currentFilter; + + private String coverUrl; + private String thumbnailUrl; + private String lastSelectedCoverFile; + private String lastSelectedThumbnailFile; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_channel_form, container, false); + + scrollView = root.findViewById(R.id.channel_form_scroll_view); + linkCancel = root.findViewById(R.id.channel_form_cancel_link); + linkShowOptional = root.findViewById(R.id.channel_form_toggle_optional); + buttonSave = root.findViewById(R.id.channel_form_save_button); + + containerOptionalFields = root.findViewById(R.id.channel_form_optional_fields_container); + inputTitle = root.findViewById(R.id.channel_form_input_title); + inputChannelName = root.findViewById(R.id.channel_form_input_channel_name); + inputDeposit = root.findViewById(R.id.channel_form_input_deposit); + inputDescription = root.findViewById(R.id.channel_form_input_description); + inputWebsite = root.findViewById(R.id.channel_form_input_website); + inputEmail = root.findViewById(R.id.channel_form_input_email); + inputTagFilter = root.findViewById(R.id.form_tag_filter_input); + + coverEditArea = root.findViewById(R.id.channel_form_cover_edit_area); + iconContainer = root.findViewById(R.id.channel_form_icon_container); + imageCover = root.findViewById(R.id.channel_form_cover_image); + imageThumbnail = root.findViewById(R.id.channel_form_thumbnail); + inlineBalanceContainer = root.findViewById(R.id.channel_form_inline_balance_container); + inlineBalanceValue = root.findViewById(R.id.channel_form_inline_balance_value); + uploadProgress = root.findViewById(R.id.channel_form_upload_progress); + channelSaveProgress = root.findViewById(R.id.channel_form_save_progress); + + Context context = getContext(); + FlexboxLayoutManager flm1 = new FlexboxLayoutManager(context); + FlexboxLayoutManager flm2 = new FlexboxLayoutManager(context); + FlexboxLayoutManager flm3 = new FlexboxLayoutManager(context); + addedTagsList = root.findViewById(R.id.form_added_tags); + addedTagsList.setLayoutManager(flm1); + suggestedTagsList = root.findViewById(R.id.form_suggested_tags); + suggestedTagsList.setLayoutManager(flm2); + + addedTagsAdapter = new TagListAdapter(new ArrayList<>(), context); + addedTagsAdapter.setCustomizeMode(TagListAdapter.CUSTOMIZE_MODE_REMOVE); + addedTagsAdapter.setClickListener(this); + addedTagsList.setAdapter(addedTagsAdapter); + + suggestedTagsAdapter = new TagListAdapter(new ArrayList<>(), getContext()); + suggestedTagsAdapter.setCustomizeMode(TagListAdapter.CUSTOMIZE_MODE_ADD); + suggestedTagsAdapter.setClickListener(this); + suggestedTagsList.setAdapter(suggestedTagsAdapter); + + noTagsView = root.findViewById(R.id.form_no_added_tags); + noTagResultsView = root.findViewById(R.id.form_no_tag_results); + + buttonSave = root.findViewById(R.id.channel_form_save_button); + + inputDeposit.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + Helper.setViewVisibility(inlineBalanceContainer, hasFocus ? View.VISIBLE : View.INVISIBLE); + } + }); + linkShowOptional.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (containerOptionalFields.getVisibility() != View.VISIBLE) { + containerOptionalFields.setVisibility(View.VISIBLE); + linkShowOptional.setText(R.string.hide_optional_fields); + scrollView.post(new Runnable() { + @Override + public void run() { + scrollView.fullScroll(NestedScrollView.FOCUS_DOWN); + } + }); + } else { + containerOptionalFields.setVisibility(View.GONE); + linkShowOptional.setText(R.string.show_optional_fields); + } + } + }); + linkCancel.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + clearInputFocus(); + + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).onBackPressed(); + } + } + }); + coverEditArea.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (uploading) { + Snackbar.make(getView(), R.string.wait_for_upload, Snackbar.LENGTH_LONG).show(); + return; + } + + checkPermissionsAndLaunchFilePicker(true); + } + }); + iconContainer.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (uploading) { + Snackbar.make(getView(), R.string.wait_for_upload, Snackbar.LENGTH_LONG).show(); + return; + } + + checkPermissionsAndLaunchFilePicker(false); + } + }); + + buttonSave.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + Claim claimToSave = buildChannelClaimToSave(); + validateAndSaveClaim(claimToSave); + } + }); + inputTagFilter.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) { + String value = Helper.getValue(charSequence); + setFilter(value); + } + + @Override + public void afterTextChanged(Editable editable) { + + } + }); + + return root; + } + + private void checkParams() { + Map params = getParams(); + if (params.containsKey("claim")) { + Claim claim = (Claim) params.get("claim"); + if (claim != null && !claim.equals(this.currentClaim)) { + this.currentClaim = claim; + editFieldsLoaded = false; + } + } + } + + private void updateFieldsFromCurrentClaim() { + if (currentClaim != null && !editFieldsLoaded) { + inputTitle.setText(currentClaim.getTitle()); + inputChannelName.setText(currentClaim.getName()); + inputDeposit.setText(currentClaim.getAmount()); + inputEmail.setText(currentClaim.getEmail()); + inputWebsite.setText(currentClaim.getWebsiteUrl()); + inputDescription.setText(currentClaim.getDescription()); + if (currentClaim.getTagObjects() != null) { + addedTagsAdapter.addTags(currentClaim.getTagObjects()); + } + + Context context = getContext(); + if (context != null) { + if (!Helper.isNullOrEmpty(currentClaim.getCoverUrl())) { + Glide.with(context.getApplicationContext()).load(currentClaim.getCoverUrl()).centerCrop().into(imageCover); + coverUrl = currentClaim.getCoverUrl(); + } + if (!Helper.isNullOrEmpty(currentClaim.getThumbnailUrl())) { + Glide.with(context.getApplicationContext()).load(currentClaim.getThumbnailUrl()).apply(RequestOptions.circleCropTransform()).into(imageThumbnail); + thumbnailUrl = currentClaim.getThumbnailUrl(); + } + } + + inputChannelName.setEnabled(false); + editMode = true; + editFieldsLoaded = true; + } + } + + private void validateAndSaveClaim(Claim claim) { + if (!editMode && Helper.isNullOrEmpty(claim.getName())) { + showError(getString(R.string.please_enter_channel_name)); + return; + } + + String channelName = claim.getName().startsWith("@") ? claim.getName().substring(1) : claim.getName(); + if (!LbryUri.isNameValid(channelName)) { + showError(getString(R.string.channel_name_invalid_characters)); + return; + } + if (!editMode) { + if (Helper.channelExists(channelName)) { + showError(getString(R.string.channel_name_already_created)); + return; + } + } + + String depositString = Helper.getValue(inputDeposit.getText()); + double depositAmount = 0; + try { + depositAmount = Double.valueOf(depositString); + } catch (NumberFormatException ex) { + // pass + showError(getString(R.string.please_enter_valid_deposit)); + return; + } + if (depositAmount == 0) { + String error = getResources().getQuantityString(R.plurals.min_deposit_required, depositAmount == 1 ? 1 : 2, String.valueOf(Helper.MIN_DEPOSIT)); + showError(error); + return; + } + if (Lbry.walletBalance == null || Lbry.walletBalance.getAvailable().doubleValue() < depositAmount) { + showError(getString(R.string.deposit_more_than_balance)); + return; + } + + ChannelCreateUpdateTask task = new ChannelCreateUpdateTask(claim, new BigDecimal(depositString), editMode, channelSaveProgress, new ClaimResultHandler() { + @Override + public void beforeStart() { + preSave(); + } + + @Override + public void onSuccess(Claim claimResult) { + postSave(); + + // Run the logPublish task + if (!BuildConfig.DEBUG) { + LogPublishTask logPublish = new LogPublishTask(claimResult); + logPublish.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + if (!editMode) { + // channel created + Bundle bundle = new Bundle(); + bundle.putString("claim_id", claimResult.getClaimId()); + bundle.putString("claim_name", claimResult.getName()); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_CHANNEL_CREATE, bundle); + } + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.showMessage(R.string.channel_save_successful); + activity.onBackPressed(); + } + } + + @Override + public void onError(Exception error) { + showError(error != null ? error.getMessage() : getString(R.string.channel_save_failed)); + postSave(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void showError(String message) { + Context context = getContext(); + if (context != null) { + Snackbar.make(getView(), message, Snackbar.LENGTH_LONG).setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + } + + public void checkPermissionsAndLaunchFilePicker(boolean isCover) { + Context context = getContext(); + if (MainActivity.hasPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, context)) { + launchCoverSelectPending = false; + launchThumbnailSelectPending = false; + + coverFilePickerActive = isCover; + thumbnailFilePickerActive = !isCover; + launchFilePicker(); + } else { + launchCoverSelectPending = isCover; + launchThumbnailSelectPending = !isCover; + MainActivity.requestPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, + MainActivity.REQUEST_STORAGE_PERMISSION, + getString(R.string.storage_permission_rationale_images), + context, + true); + } + } + + private void launchFilePicker() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity.startingFilePickerActivity = true; + Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT); + intent.setType("image/*"); + ((MainActivity) context).startActivityForResult( + Intent.createChooser(intent, getString(coverFilePickerActive ? R.string.select_cover : R.string.select_thumbnail)), + MainActivity.REQUEST_FILE_PICKER); + } + } + + public void onFilePickerCancelled() { + coverFilePickerActive = false; + thumbnailFilePickerActive = false; + } + + public void onFilePicked(String filePath) { + if (Helper.isNullOrEmpty(filePath)) { + Snackbar.make(getView(), R.string.undetermined_image_filepath, Snackbar.LENGTH_LONG).setBackgroundTint( + ContextCompat.getColor(getContext(), R.color.red)).show(); + return; + } + + Context context = getContext(); + if (context != null) { + Uri fileUri = Uri.fromFile(new File(filePath)); + if (coverFilePickerActive) { + // cover selected + if (filePath.equalsIgnoreCase(lastSelectedCoverFile)) { + // previous selected cover was uploaded successfully + return; + } + Glide.with(context.getApplicationContext()).load(fileUri).centerCrop().into(imageCover); + } else if (thumbnailFilePickerActive) { + if (filePath.equalsIgnoreCase(lastSelectedThumbnailFile)) { + // previous selected thumbnail was uploaded successfully + return; + } + // thumbnail selected + Glide.with(context.getApplicationContext()).load(fileUri).apply(RequestOptions.circleCropTransform()).into(imageThumbnail); + } + + // Upload the image + uploading = true; + UploadImageTask task = new UploadImageTask(filePath, uploadProgress, new UploadImageTask.UploadThumbnailHandler() { + @Override + public void onSuccess(String url) { + if (coverFilePickerActive) { + coverUrl = url; + lastSelectedCoverFile = filePath; + } else if (thumbnailFilePickerActive) { + thumbnailUrl = url; + lastSelectedThumbnailFile = filePath; + } + + coverFilePickerActive = false; + thumbnailFilePickerActive = false; + uploading = false; + } + + @Override + public void onError(Exception error) { + if (getContext() != null) { + showError(getString(R.string.image_upload_failed)); + } + if (coverFilePickerActive) { + // cover selected + imageCover.setImageResource(R.drawable.default_channel_cover); + } else if (thumbnailFilePickerActive) { + imageThumbnail.setImageDrawable(null); + } + coverFilePickerActive = false; + thumbnailFilePickerActive = false; + uploading = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } else { + coverFilePickerActive = false; + thumbnailFilePickerActive = false; + } + } + + public void onStoragePermissionGranted() { + if (launchCoverSelectPending) { + checkPermissionsAndLaunchFilePicker(true); + } else if (launchThumbnailSelectPending) { + checkPermissionsAndLaunchFilePicker(false); + } + } + public void onStoragePermissionRefused() { + Snackbar.make(getView(), R.string.storage_permission_rationale_images, Snackbar.LENGTH_LONG).setBackgroundTint( + ContextCompat.getColor(getContext(), R.color.red) + ).show(); + } + + @Override + public void onStart() { + super.onStart(); + MainActivity activity = (MainActivity) getContext(); + if (activity != null) { + activity.hideSearchBar(); + activity.showNavigationBackIcon(); + activity.lockDrawer(); + activity.hideFloatingWalletBalance(); + + activity.addFilePickerListener(this); + activity.addWalletBalanceListener(this); + + ActionBar actionBar = activity.getSupportActionBar(); + if (actionBar != null) { + actionBar.setTitle(editMode ? R.string.edit_channel : R.string.create_a_channel); + } + } + } + + @Override + public void onPause() { + clearInputFocus(); + super.onPause(); + } + + @Override + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) getContext(); + activity.removeStoragePermissionListener(this); + activity.removeWalletBalanceListener(this); + activity.restoreToggle(); + activity.showFloatingWalletBalance(); + if (!MainActivity.startingFilePickerActivity) { + activity.removeFilePickerListener(this); + activity.removeNavFragment(ChannelFormFragment.class, NavMenuItem.ID_ITEM_CHANNELS); + } + } + super.onStop(); + } + + public void onResume() { + super.onResume(); + checkParams(); + updateFieldsFromCurrentClaim(); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Channel Form", "ChannelForm"); + activity.addStoragePermissionListener(this); + if (editMode) { + ActionBar actionBar = activity.getSupportActionBar(); + if (actionBar != null) { + actionBar.setTitle(R.string.edit_channel); + } + } + } + String filterText = Helper.getValue(inputTagFilter.getText()); + updateSuggestedTags(filterText, SUGGESTED_LIMIT, true); + } + + private void checkNoAddedTags() { + Helper.setViewVisibility(noTagsView, addedTagsAdapter == null || addedTagsAdapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + private void checkNoTagResults() { + Helper.setViewVisibility(noTagResultsView, suggestedTagsAdapter == null || suggestedTagsAdapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + public void addTag(Tag tag) { + if (addedTagsAdapter.getTags().contains(tag)) { + Snackbar.make(getView(), getString(R.string.tag_already_added, tag.getName()), Snackbar.LENGTH_LONG).show(); + return; + } + if (addedTagsAdapter.getItemCount() == 5) { + Snackbar.make(getView(), R.string.tag_limit_reached, Snackbar.LENGTH_LONG).show(); + return; + } + + addedTagsAdapter.addTag(tag); + if (suggestedTagsAdapter != null) { + suggestedTagsAdapter.removeTag(tag); + } + updateSuggestedTags(currentFilter, SUGGESTED_LIMIT, false); + + checkNoAddedTags(); + checkNoTagResults(); + } + public void removeTag(Tag tag) { + addedTagsAdapter.removeTag(tag); + updateSuggestedTags(currentFilter, SUGGESTED_LIMIT, false); + checkNoAddedTags(); + checkNoTagResults(); + } + + @Override + public void onWalletBalanceUpdated(WalletBalance walletBalance) { + if (walletBalance != null && inlineBalanceValue != null) { + inlineBalanceValue.setText(Helper.shortCurrencyFormat(walletBalance.getAvailable().doubleValue())); + } + } + + public void setFilter(String filter) { + currentFilter = filter; + updateSuggestedTags(currentFilter, SUGGESTED_LIMIT, true); + } + + private void updateSuggestedTags(String filter, int limit, boolean clearPrevious) { + UpdateSuggestedTagsTask task = new UpdateSuggestedTagsTask( + filter, + limit, + addedTagsAdapter, + suggestedTagsAdapter, + clearPrevious, + false, + new UpdateSuggestedTagsTask.KnownTagsHandler() { + @Override + public void onSuccess(List tags) { + if (suggestedTagsAdapter == null) { + suggestedTagsAdapter = new TagListAdapter(tags, getContext()); + suggestedTagsAdapter.setCustomizeMode(TagListAdapter.CUSTOMIZE_MODE_ADD); + suggestedTagsAdapter.setClickListener(ChannelFormFragment.this); + if (suggestedTagsList != null) { + suggestedTagsList.setAdapter(suggestedTagsAdapter); + } + } else { + suggestedTagsAdapter.setTags(tags); + } + + checkNoAddedTags(); + checkNoTagResults(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private Claim buildChannelClaimToSave() { + Claim claim = new Claim(); + String name = Helper.getValue(inputChannelName.getText()); + if (!name.startsWith("@")) { + name = String.format("@%s", name); + } + claim.setName(name); + + if (currentClaim != null) { + claim.setClaimId(currentClaim.getClaimId()); + } + + Claim.ChannelMetadata metadata = new Claim.ChannelMetadata(); + metadata.setTitle(Helper.getValue(inputTitle.getText())); + metadata.setDescription(Helper.getValue(inputDescription.getText())); + metadata.setWebsiteUrl(Helper.getValue(inputWebsite.getText())); + metadata.setEmail(Helper.getValue(inputEmail.getText())); + + Claim.Resource cover = new Claim.Resource(); + cover.setUrl(coverUrl == null ? "" : coverUrl); + Claim.Resource thumbnail = new Claim.Resource(); + thumbnail.setUrl(thumbnailUrl == null ? "" : thumbnailUrl); + metadata.setThumbnail(thumbnail); + metadata.setCover(cover); + + List addedTags = addedTagsAdapter != null ? new ArrayList<>(addedTagsAdapter.getTags()) : new ArrayList<>(); + metadata.setTags(Helper.getTagsForTagObjects(addedTags)); + + claim.setValue(metadata); + return claim; + } + + private void preSave() { + saveInProgress = true; + Helper.setViewVisibility(linkShowOptional, View.GONE); + Helper.setViewEnabled(linkCancel, false); + Helper.setViewEnabled(buttonSave, false); + } + + private void postSave() { + Helper.setViewVisibility(linkShowOptional, View.VISIBLE); + Helper.setViewEnabled(linkCancel, true); + Helper.setViewEnabled(buttonSave, true); + + clearInputFocus(); + + saveInProgress = false; + } + + public void clearInputFocus() { + inputChannelName.clearFocus(); + inputDeposit.clearFocus(); + inputWebsite.clearFocus(); + inputEmail.clearFocus(); + inputDescription.clearFocus(); + inputTitle.clearFocus(); + inputTagFilter.clearFocus(); + } + + @Override + public void onTagClicked(Tag tag, int customizeMode) { + if (customizeMode == TagListAdapter.CUSTOMIZE_MODE_ADD) { + addTag(tag); + } else if (customizeMode == TagListAdapter.CUSTOMIZE_MODE_REMOVE) { + removeTag(tag); + } + } +} 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..3299aec7 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/channel/ChannelFragment.java @@ -0,0 +1,531 @@ +package io.lbry.browser.ui.channel; + +import android.content.Context; +import android.content.DialogInterface; +import android.content.Intent; +import android.graphics.Color; +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 android.widget.Toast; + +import androidx.annotation.NonNull; +import androidx.appcompat.app.AlertDialog; +import androidx.core.content.ContextCompat; +import androidx.fragment.app.Fragment; +import androidx.fragment.app.FragmentActivity; +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.snackbar.Snackbar; +import com.google.android.material.tabs.TabLayout; +import com.google.android.material.tabs.TabLayoutMediator; + +import java.math.BigDecimal; +import java.text.DecimalFormat; +import java.text.NumberFormat; +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.dialog.SendTipDialogFragment; +import io.lbry.browser.exceptions.LbryUriException; +import io.lbry.browser.listener.FetchChannelsListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.ClaimCacheKey; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.tasks.claim.AbandonChannelTask; +import io.lbry.browser.tasks.claim.AbandonHandler; +import io.lbry.browser.tasks.lbryinc.ChannelSubscribeTask; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ResolveTask; +import io.lbry.browser.tasks.lbryinc.FetchStatCountTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.ui.controls.SolidIconView; +import io.lbry.browser.ui.findcontent.FollowingFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; +import lombok.SneakyThrows; + +public class ChannelFragment extends BaseFragment implements FetchChannelsListener { + private Claim claim; + private boolean subscribing; + 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; + + private View buttonEdit; + private View buttonDelete; + private View buttonShare; + private View buttonTip; + private View buttonFollowUnfollow; + private int subCount; + private SolidIconView iconFollowUnfollow; + private View layoutNothingAtLocation; + private View layoutLoadingState; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_channel, container, false); + + layoutLoadingState = root.findViewById(R.id.channel_view_loading_state); + layoutNothingAtLocation = root.findViewById(R.id.container_nothing_at_location); + 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); + + buttonEdit = root.findViewById(R.id.channel_view_edit); + buttonDelete = root.findViewById(R.id.channel_view_delete); + buttonShare = root.findViewById(R.id.channel_view_share); + buttonTip = root.findViewById(R.id.channel_view_tip); + buttonFollowUnfollow = root.findViewById(R.id.channel_view_follow_unfollow); + iconFollowUnfollow = root.findViewById(R.id.channel_view_icon_follow_unfollow); + + tabPager = root.findViewById(R.id.channel_view_pager); + tabLayout = root.findViewById(R.id.channel_view_tabs); + tabPager.setSaveEnabled(false); + + buttonEdit.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (claim != null) { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openChannelForm(claim); + } + } + } + }); + buttonDelete.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (claim != null) { + AlertDialog.Builder builder = new AlertDialog.Builder(getContext()). + setTitle(R.string.delete_channel). + setMessage(R.string.confirm_delete_channel) + .setPositiveButton(R.string.yes, new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + deleteCurrentClaim(); + } + }).setNegativeButton(R.string.no, null); + builder.show(); + } + } + }); + + buttonShare.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (claim != null) { + try { + String shareUrl = LbryUri.parse( + !Helper.isNullOrEmpty(claim.getCanonicalUrl()) ? claim.getCanonicalUrl() : + (!Helper.isNullOrEmpty(claim.getShortUrl()) ? claim.getShortUrl() : claim.getPermanentUrl())).toTvString(); + Intent shareIntent = new Intent(); + shareIntent.setAction(Intent.ACTION_SEND); + shareIntent.setType("text/plain"); + shareIntent.putExtra(Intent.EXTRA_TEXT, shareUrl); + + MainActivity.startingShareActivity = true; + Intent shareUrlIntent = Intent.createChooser(shareIntent, getString(R.string.share_lbry_content)); + shareUrlIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); + startActivity(shareUrlIntent); + } catch (LbryUriException ex) { + // pass + } + } + } + }); + + buttonTip.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (!Lbry.SDK_READY) { + Snackbar.make(getView(), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + if (claim != null) { + SendTipDialogFragment dialog = SendTipDialogFragment.newInstance(); + dialog.setClaim(claim); + dialog.setListener(new SendTipDialogFragment.SendTipListener() { + @Override + public void onTipSent(BigDecimal amount) { + double sentAmount = amount.doubleValue(); + String message = getResources().getQuantityString( + R.plurals.you_sent_a_tip, sentAmount == 1.0 ? 1 : 2, + new DecimalFormat("#,###.##").format(sentAmount)); + Snackbar.make(getView(), message, Snackbar.LENGTH_LONG).show(); + } + }); + Context context = getContext(); + if (context instanceof MainActivity) { + dialog.show(((MainActivity) context).getSupportFragmentManager(), SendTipDialogFragment.TAG); + } + } + } + }); + + buttonFollowUnfollow.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (claim != null) { + if (subscribing) { + return; + } + + subscribing = true; + boolean isFollowing = Lbryio.isFollowing(claim); + Subscription subscription = Subscription.fromClaim(claim); + buttonFollowUnfollow.setEnabled(false); + new ChannelSubscribeTask(getContext(), claim.getClaimId(), subscription, isFollowing, new ChannelSubscribeTask.ChannelSubscribeHandler() { + @Override + public void onSuccess() { + if (isFollowing) { + Lbryio.removeSubscription(subscription); + Lbryio.removeCachedResolvedSubscription(claim); + } else { + Lbryio.addSubscription(subscription); + Lbryio.addCachedResolvedSubscription(claim); + } + buttonFollowUnfollow.setEnabled(true); + subscribing = false; + checkIsFollowing(); + FollowingFragment.resetClaimSearchContent = true; + + if (Lbry.SDK_READY) { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).saveSharedUserState(); + } + } + } + + @Override + public void onError(Exception exception) { + buttonFollowUnfollow.setEnabled(true); + subscribing = false; + } + }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + }); + + return root; + } + + private void deleteCurrentClaim() { + if (claim != null) { + Helper.setViewVisibility(layoutDisplayArea, View.GONE); + Helper.setViewVisibility(layoutLoadingState, View.VISIBLE); + AbandonChannelTask task = new AbandonChannelTask(Arrays.asList(claim.getClaimId()), layoutResolving, new AbandonHandler() { + @Override + public void onComplete(List successfulClaimIds, List failedClaimIds, List errors) { + Context context = getContext(); + if (context instanceof MainActivity) { + if (failedClaimIds.size() == 0) { + MainActivity activity = (MainActivity) context; + activity.showMessage(R.string.channel_deleted); + activity.onBackPressed(); + } else { + View root = getView(); + if (root != null) { + Snackbar.make(root, R.string.channel_failed_delete, Toast.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + } + } + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + private void checkIsFollowing() { + if (claim != null) { + boolean isFollowing = Lbryio.isFollowing(claim); + if (iconFollowUnfollow != null) { + iconFollowUnfollow.setText(isFollowing ? R.string.fa_heart_broken : R.string.fa_heart); + Context context = getContext(); + if (context != null) { + iconFollowUnfollow.setTextColor(ContextCompat.getColor(context, isFollowing ? R.color.foreground : R.color.red)); + } + } + } + } + + public void onChannelsFetched(List channels) { + checkOwnChannel(); + } + + private void checkOwnChannel() { + if (claim != null) { + boolean isOwnChannel = Lbry.ownChannels.contains(claim); + Helper.setViewVisibility(buttonEdit, isOwnChannel ? View.VISIBLE : View.GONE); + Helper.setViewVisibility(buttonDelete, isOwnChannel ? View.VISIBLE : View.GONE); + } + } + + public void onResume() { + super.onResume(); + Context context = getContext(); + Map params = getParams(); + String url = params != null && params.containsKey("url") ? (String) params.get("url") : null; + Helper.setWunderbarValue(url, context); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.addFetchChannelsListener(this); + LbryAnalytics.setCurrentScreen(activity, "Channel", "Channel"); + } + + checkParams(); + checkOwnChannel(); + } + + public void onPause() { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).removeFetchChannelsListener(this); + } + super.onPause(); + } + + 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 (params.containsKey("url")) { + String newUrl = params.get("url").toString(); + if (claim == null || !newUrl.equalsIgnoreCase(url)) { + this.claim = null; + this.url = newUrl; + updateRequired = true; + } + } + + if (updateRequired) { + resetSubCount(); + if (!Helper.isNullOrEmpty(url)) { + // check if the claim is already cached + ClaimCacheKey key = new ClaimCacheKey(); + key.setUrl(url); + if (Lbry.claimCache.containsKey(key)) { + claim = Lbry.claimCache.get(key); + } else { + resolveUrl(); + } + } else if (claim == null) { + // nothing at this location + renderNothingAtLocation(); + } + } + + if (!Helper.isNullOrEmpty(url)) { + Helper.saveUrlHistory(url, claim != null ? claim.getTitle() : null, UrlSuggestion.TYPE_CHANNEL); + } + + if (claim != null) { + renderClaim(); + } + } + + private void resolveUrl() { + Helper.setViewVisibility(layoutDisplayArea, View.INVISIBLE); + Helper.setViewVisibility(layoutLoadingState, View.VISIBLE); + ResolveTask task = new ResolveTask(url, Lbry.LBRY_TV_CONNECTION_STRING, layoutResolving, new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + if (claims.size() > 0 && !Helper.isNullOrEmpty(claims.get(0).getClaimId())) { + claim = claims.get(0); + if (!Helper.isNullOrEmpty(url)) { + Helper.saveUrlHistory(url, claim.getTitle(), UrlSuggestion.TYPE_CHANNEL); + } + + renderClaim(); + checkOwnChannel(); + } else { + renderNothingAtLocation(); + } + } + + @Override + public void onError(Exception error) { + renderNothingAtLocation(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void renderNothingAtLocation() { + layoutLoadingState.setVisibility(View.VISIBLE); + layoutNothingAtLocation.setVisibility(View.VISIBLE); + layoutDisplayArea.setVisibility(View.INVISIBLE); + } + + public void setParams(Map params) { + super.setParams(params); + if (getView() != null) { + checkParams(); + } + } + + private void renderClaim() { + if (claim == null) { + renderNothingAtLocation(); + return; + } + + loadSubCount(); + checkIsFollowing(); + layoutLoadingState.setVisibility(View.GONE); + layoutDisplayArea.setVisibility(View.VISIBLE); + + String thumbnailUrl = claim.getThumbnailUrl(); + String coverUrl = claim.getCoverUrl(); + textTitle.setText(Helper.isNullOrEmpty(claim.getTitle()) ? claim.getName() : claim.getTitle()); + + Context context = getContext(); + if (context != null && !Helper.isNullOrEmpty(coverUrl)) { + Glide.with(context.getApplicationContext()).load(coverUrl).centerCrop().into(imageCover); + } + if (context != null && !Helper.isNullOrEmpty(thumbnailUrl)) { + Glide.with(context.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); + if (claim.getName() != null) { + textAlpha.setText(claim.getName().substring(1, 2)); + } + } + + try { + if (tabPager.getAdapter() == null && context instanceof MainActivity) { + tabPager.setAdapter(new ChannelPagerAdapter(claim, (MainActivity) context)); + } + } catch (IllegalStateException ex) { + // TODO: Fix why this is happening + // pass + } + 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 resetSubCount() { + subCount = -1; + Helper.setViewText(textFollowerCount, null); + Helper.setViewVisibility(textFollowerCount, View.INVISIBLE); + } + + private void loadSubCount() { + if (claim != null) { + FetchStatCountTask task = new FetchStatCountTask( + FetchStatCountTask.STAT_SUB_COUNT, claim.getClaimId(), null, new FetchStatCountTask.FetchStatCountHandler() { + @Override + public void onSuccess(int count) { + try { + String displayText = getResources().getQuantityString(R.plurals.follower_count, count, NumberFormat.getInstance().format(count)); + Helper.setViewText(textFollowerCount, displayText); + Helper.setViewVisibility(textFollowerCount, View.VISIBLE); + } catch (IllegalStateException ex) { + // pass + } + } + + @Override + public void onError(Exception error) { + // pass + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + 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(); + if (channelClaim != null) { + contentFragment.setChannelId(channelClaim.getClaimId()); + } + return contentFragment; + + case 1: + ChannelAboutFragment aboutFragment = ChannelAboutFragment.class.newInstance(); + try { + Claim.ChannelMetadata metadata = (Claim.ChannelMetadata) channelClaim.getValue(); + if (metadata != null) { + 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/channel/ChannelManagerFragment.java b/app/src/main/java/io/lbry/browser/ui/channel/ChannelManagerFragment.java new file mode 100644 index 00000000..4f2454e2 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/channel/ChannelManagerFragment.java @@ -0,0 +1,321 @@ +package io.lbry.browser.ui.channel; + +import android.content.Context; +import android.content.DialogInterface; +import android.graphics.Color; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.Menu; +import android.view.MenuItem; +import android.view.View; +import android.view.ViewGroup; +import android.widget.Button; +import android.widget.ProgressBar; + +import androidx.annotation.NonNull; +import androidx.appcompat.app.AlertDialog; +import androidx.appcompat.view.ActionMode; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import com.google.android.material.floatingactionbutton.FloatingActionButton; +import com.google.android.material.snackbar.Snackbar; + +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.R; +import io.lbry.browser.adapter.ClaimListAdapter; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.listener.SelectionModeListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.NavMenuItem; +import io.lbry.browser.tasks.claim.AbandonChannelTask; +import io.lbry.browser.tasks.claim.AbandonHandler; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ClaimListTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; + +public class ChannelManagerFragment extends BaseFragment implements ActionMode.Callback, SelectionModeListener, SdkStatusListener { + + private Button buttonNewChannel; + private FloatingActionButton fabNewChannel; + private ActionMode actionMode; + private View emptyView; + private View layoutSdkInitializing; + private ProgressBar loading; + private ProgressBar bigLoading; + private RecyclerView channelList; + private ClaimListAdapter adapter; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_channel_manager, container, false); + + buttonNewChannel = root.findViewById(R.id.channel_manager_create_button); + fabNewChannel = root.findViewById(R.id.channel_manager_fab_new_channel); + buttonNewChannel.setOnClickListener(newChannelClickListener); + fabNewChannel.setOnClickListener(newChannelClickListener); + + emptyView = root.findViewById(R.id.channel_manager_empty_container); + layoutSdkInitializing = root.findViewById(R.id.container_sdk_initializing); + channelList = root.findViewById(R.id.channel_manager_list); + LinearLayoutManager llm = new LinearLayoutManager(getContext()); + channelList.setLayoutManager(llm); + loading = root.findViewById(R.id.channel_manager_list_loading); + bigLoading = root.findViewById(R.id.channel_manager_list_big_loading); + + layoutSdkInitializing.setVisibility(Lbry.SDK_READY ? View.GONE : View.VISIBLE); + + return root; + } + + private View.OnClickListener newChannelClickListener = new View.OnClickListener() { + @Override + public void onClick(View view) { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openChannelForm(null); + } + } + }; + + @Override + public void onStart() { + super.onStart(); + Context context = getContext(); + if (context != null) { + MainActivity activity = (MainActivity) context; + activity.hideFloatingWalletBalance(); + } + } + + @Override + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.showFloatingWalletBalance(); + } + super.onStop(); + } + + public void onResume() { + super.onResume(); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.setWunderbarValue(null); + LbryAnalytics.setCurrentScreen(activity, "Channel Manager", "ChannelManager"); + } + + if (!Lbry.SDK_READY) { + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.addSdkStatusListener(this); + } + } else { + onSdkReady(); + } + } + + public void onSdkReady() { + Helper.setViewVisibility(layoutSdkInitializing, View.GONE); + Helper.setViewVisibility(fabNewChannel, View.VISIBLE); + if (adapter != null && channelList != null) { + channelList.setAdapter(adapter); + } + fetchChannels(); + } + + public View getLoading() { + return (adapter == null || adapter.getItemCount() == 0) ? bigLoading : loading; + } + + private void checkNoChannels() { + Helper.setViewVisibility(emptyView, adapter == null || adapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + + private void fetchChannels() { + Helper.setViewVisibility(emptyView, View.GONE); + ClaimListTask task = new ClaimListTask(Claim.TYPE_CHANNEL, getLoading(), new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + Lbry.ownChannels = Helper.filterDeletedClaims(new ArrayList<>(claims)); + Context context = getContext(); + if (adapter == null) { + adapter = new ClaimListAdapter(claims, context); + adapter.setCanEnterSelectionMode(true); + adapter.setSelectionModeListener(ChannelManagerFragment.this); + adapter.setListener(new ClaimListAdapter.ClaimListItemListener() { + @Override + public void onClaimClicked(Claim claim) { + if (context instanceof MainActivity) { + ((MainActivity) context).openChannelClaim(claim); + } + } + }); + if (channelList != null) { + channelList.setAdapter(adapter); + } + } else { + adapter.setItems(claims); + } + + checkNoChannels(); + } + + @Override + public void onError(Exception error) { + checkNoChannels(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void onEnterSelectionMode() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.startSupportActionMode(this); + } + } + public void onItemSelectionToggled() { + if (actionMode != null) { + actionMode.setTitle(String.valueOf(adapter.getSelectedCount())); + actionMode.invalidate(); + } + } + public void onExitSelectionMode() { + if (actionMode != null) { + actionMode.finish(); + } + } + + @Override + public boolean onCreateActionMode(ActionMode actionMode, Menu menu) { + this.actionMode = actionMode; + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (!activity.isDarkMode()) { + activity.getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_VISIBLE); + } + } + + actionMode.getMenuInflater().inflate(R.menu.menu_claim_list, menu); + return true; + } + @Override + public void onDestroyActionMode(ActionMode actionMode) { + if (adapter != null) { + adapter.clearSelectedItems(); + adapter.setInSelectionMode(false); + adapter.notifyDataSetChanged(); + } + Context context = getContext(); + if (context != null) { + MainActivity activity = (MainActivity) context; + if (!activity.isDarkMode()) { + activity.getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR); + } + } + this.actionMode = null; + } + + @Override + public boolean onPrepareActionMode(androidx.appcompat.view.ActionMode actionMode, Menu menu) { + int selectionCount = adapter != null ? adapter.getSelectedCount() : 0; + menu.findItem(R.id.action_edit).setVisible(selectionCount == 1); + return true; + } + + @Override + public boolean onActionItemClicked(androidx.appcompat.view.ActionMode actionMode, MenuItem menuItem) { + if (R.id.action_edit == menuItem.getItemId()) { + if (adapter != null && adapter.getSelectedCount() > 0) { + Claim claim = adapter.getSelectedItems().get(0); + // start channel editor with the claim + Context context = getContext(); + if (context instanceof MainActivity) { + Map params = new HashMap<>(); + params.put("claim", claim); + ((MainActivity) context).openFragment(ChannelFormFragment.class, true, NavMenuItem.ID_ITEM_CHANNELS, params); + } + + actionMode.finish(); + return true; + } + } + if (R.id.action_delete == menuItem.getItemId()) { + if (adapter != null && adapter.getSelectedCount() > 0) { + final List selectedClaims = new ArrayList<>(adapter.getSelectedItems()); + String message = getResources().getQuantityString(R.plurals.confirm_delete_channels, selectedClaims.size()); + AlertDialog.Builder builder = new AlertDialog.Builder(getContext()). + setTitle(R.string.delete_selection). + setMessage(message) + .setPositiveButton(R.string.yes, new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + handleDeleteSelectedClaims(selectedClaims); + } + }).setNegativeButton(R.string.no, null); + builder.show(); + return true; + } + } + + return false; + } + + private void handleDeleteSelectedClaims(List selectedClaims) { + List claimIds = new ArrayList<>(); + + for (Claim claim : selectedClaims) { + claimIds.add(claim.getClaimId()); + } + + if (actionMode != null) { + actionMode.finish(); + } + + Helper.setViewVisibility(channelList, View.INVISIBLE); + Helper.setViewVisibility(fabNewChannel, View.INVISIBLE); + AbandonChannelTask task = new AbandonChannelTask(claimIds, bigLoading, new AbandonHandler() { + @Override + public void onComplete(List successfulClaimIds, List failedClaimIds, List errors) { + View root = getView(); + if (root != null) { + if (failedClaimIds.size() > 0) { + Snackbar.make(root, R.string.one_or_more_channels_failed_abandon, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } else if (successfulClaimIds.size() == claimIds.size()) { + try { + String message = getResources().getQuantityString(R.plurals.channels_deleted, successfulClaimIds.size()); + Snackbar.make(root, message, Snackbar.LENGTH_LONG).show(); + } catch (IllegalStateException ex) { + // pass + } + } + } + + Lbry.abandonedClaimIds.addAll(successfulClaimIds); + if (adapter != null) { + adapter.setItems(Helper.filterDeletedClaims(adapter.getItems())); + } + + Helper.setViewVisibility(channelList, View.VISIBLE); + Helper.setViewVisibility(fabNewChannel, View.VISIBLE); + checkNoChannels(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } +} 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/findcontent/AllContentFragment.java b/app/src/main/java/io/lbry/browser/ui/findcontent/AllContentFragment.java new file mode 100644 index 00000000..9eb90191 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/findcontent/AllContentFragment.java @@ -0,0 +1,517 @@ +package io.lbry.browser.ui.findcontent; + +import android.content.Context; +import android.content.SharedPreferences; +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.preference.PreferenceManager; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import com.google.android.material.snackbar.Snackbar; + +import org.json.JSONException; +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +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.ContentScopeDialogFragment; +import io.lbry.browser.dialog.ContentSortDialogFragment; +import io.lbry.browser.dialog.CustomizeTagsDialogFragment; +import io.lbry.browser.listener.DownloadActionListener; +import io.lbry.browser.listener.TagListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.LbryFile; +import io.lbry.browser.model.Tag; +import io.lbry.browser.tasks.claim.ClaimSearchResultHandler; +import io.lbry.browser.tasks.claim.ClaimSearchTask; +import io.lbry.browser.tasks.FollowUnfollowTagTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.Predefined; +import lombok.Getter; + +// TODO: Similar code to FollowingFragment and Channel page fragment. Probably make common operations (sorting/filtering) into a control +public class AllContentFragment extends BaseFragment implements DownloadActionListener, SharedPreferences.OnSharedPreferenceChangeListener { + + @Getter + private boolean singleTagView; + private List tags; + private View layoutFilterContainer; + private View customizeLink; + 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; + @Getter + private int currentContentScope; + private String contentReleaseTime; + private List contentSortOrder; + private View fromPrefix; + private View forPrefix; + private View contentLoading; + private View bigContentLoading; + private View noContentView; + 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); + customizeLink = root.findViewById(R.id.all_content_customize_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); + noContentView = root.findViewById(R.id.all_content_no_claim_search_content); + + 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); + } + } + }); + scopeLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + ContentScopeDialogFragment dialog = ContentScopeDialogFragment.newInstance(); + dialog.setCurrentScopeItem(currentContentScope); + dialog.setContentScopeListener(new ContentScopeDialogFragment.ContentScopeListener() { + @Override + public void onContentScopeItemSelected(int scopeItem) { + onContentScopeChanged(scopeItem); + } + }); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + dialog.show(activity.getSupportFragmentManager(), ContentScopeDialogFragment.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); + } + } + }); + customizeLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + showCustomizeTagsDialog(); + } + }); + + 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)); + Helper.setViewVisibility(customizeLink, View.GONE); + } else { + singleTagView = false; + // default to followed Tags scope if any tags are followed + tags = Helper.getTagsForTagObjects(Lbry.followedTags); + if (tags.size() > 0) { + currentContentScope = ContentScopeDialogFragment.ITEM_TAGS; + Helper.setViewVisibility(customizeLink, View.VISIBLE); + } + titleView.setText(getString(R.string.all_content)); + } + + Helper.setViewVisibility(forPrefix, singleTagView ? View.GONE : View.VISIBLE); + Helper.setViewVisibility(scopeLink, 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 onContentScopeChanged(int contentScope) { + currentContentScope = contentScope; + + // rebuild options and search + updateContentScopeLinkText(); + boolean isTagScope = currentContentScope == ContentScopeDialogFragment.ITEM_TAGS; + if (isTagScope) { + tags = Helper.getTagsForTagObjects(Lbry.followedTags); + // Update tags list with the user's followed tags + if (tags == null || tags.size() == 0) { + Snackbar.make(getView(), R.string.customize_tags_hint, Snackbar.LENGTH_LONG).setAction(R.string.customize, new View.OnClickListener() { + @Override + public void onClick(View view) { + // show customize + showCustomizeTagsDialog(); + } + }).show(); + } + } + Helper.setViewVisibility(customizeLink, isTagScope ? View.VISIBLE : View.GONE); + fetchClaimSearchContent(true); + } + + private void showCustomizeTagsDialog() { + CustomizeTagsDialogFragment dialog = CustomizeTagsDialogFragment.newInstance(); + dialog.setListener(new TagListener() { + @Override + public void onTagAdded(Tag tag) { + // heavy-lifting + // save to local, save to wallet and then sync + FollowUnfollowTagTask task = new FollowUnfollowTagTask(tag, false, getContext(), followUnfollowHandler); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + @Override + public void onTagRemoved(Tag tag) { + // heavy-lifting + // save to local, save to wallet and then sync + FollowUnfollowTagTask task = new FollowUnfollowTagTask(tag, true, getContext(), followUnfollowHandler); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + }); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + dialog.show(activity.getSupportFragmentManager(), CustomizeTagsDialogFragment.TAG); + } + } + + private FollowUnfollowTagTask.FollowUnfollowTagHandler followUnfollowHandler = new FollowUnfollowTagTask.FollowUnfollowTagHandler() { + @Override + public void onSuccess(Tag tag, boolean unfollowing) { + if (tags != null) { + if (unfollowing) { + tags.remove(tag.getLowercaseName()); + } else { + tags.add(tag.getLowercaseName()); + } + fetchClaimSearchContent(true); + } + + Bundle bundle = new Bundle(); + bundle.putString("tag", tag.getLowercaseName()); + LbryAnalytics.logEvent(unfollowing ? LbryAnalytics.EVENT_TAG_UNFOLLOW : LbryAnalytics.EVENT_TAG_FOLLOW, bundle); + + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).saveSharedUserState(); + } + } + + @Override + public void onError(Exception error) { + // pass + } + }; + + 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 updateContentScopeLinkText() { + int stringResourceId = -1; + switch (currentContentScope) { + case ContentScopeDialogFragment.ITEM_EVERYONE: default: stringResourceId = R.string.everyone; break; + case ContentScopeDialogFragment.ITEM_TAGS: stringResourceId = R.string.tags; break; + } + + Helper.setViewText(scopeLinkText, 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(); + Helper.setWunderbarValue(null, getContext()); + checkParams(false); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (singleTagView) { + LbryAnalytics.setCurrentScreen(activity, "Tag", "Tag"); + } else { + LbryAnalytics.setCurrentScreen(activity, "All Content", "AllContent"); + } + activity.addDownloadActionListener(this); + } + + PreferenceManager.getDefaultSharedPreferences(getContext()).registerOnSharedPreferenceChangeListener(this); + updateContentFromLinkText(); + updateContentScopeLinkText(); + updateSortByLinkText(); + fetchClaimSearchContent(); + } + + public void onPause() { + Context context = getContext(); + if (context != null) { + ((MainActivity) context).removeDownloadActionListener(this); + } + PreferenceManager.getDefaultSharedPreferences(context).unregisterOnSharedPreferenceChangeListener(this); + super.onPause(); + } + + private Map buildContentOptions() { + Context context = getContext(); + boolean canShowMatureContent = false; + if (context != null) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + canShowMatureContent = sp.getBoolean(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT, false); + } + + return Lbry.buildClaimSearchOptions( + null, + (currentContentScope == ContentScopeDialogFragment.ITEM_EVERYONE) ? null : tags, + canShowMatureContent ? null : new ArrayList<>(Predefined.MATURE_TAGS), + 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); + } + + public void fetchClaimSearchContent(boolean reset) { + if (reset && contentListAdapter != null) { + contentListAdapter.clearItems(); + currentClaimSearchPage = 1; + } + + contentClaimSearchLoading = true; + Helper.setViewVisibility(noContentView, View.GONE); + Map claimSearchOptions = buildContentOptions(); + contentClaimSearchTask = new ClaimSearchTask(claimSearchOptions, Lbry.LBRY_TV_CONNECTION_STRING, getLoadingView(), new 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) { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (claim.getName().startsWith("@")) { + // channel claim + activity.openChannelClaim(claim); + } else { + activity.openFileClaim(claim); + } + } + } + }); + } else { + contentListAdapter.addItems(claims); + } + + if (contentList != null && contentList.getAdapter() == null) { + contentList.setAdapter(contentListAdapter); + } + + contentHasReachedEnd = hasReachedEnd; + contentClaimSearchLoading = false; + checkNoContent(); + } + + @Override + public void onError(Exception error) { + contentClaimSearchLoading = false; + checkNoContent(); + } + }); + contentClaimSearchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void checkNoContent() { + boolean noContent = contentListAdapter == null || contentListAdapter.getItemCount() == 0; + Helper.setViewVisibility(noContentView, noContent ? View.VISIBLE : View.GONE); + } + + public void onSharedPreferenceChanged(SharedPreferences sp, String key) { + if (key.equalsIgnoreCase(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT)) { + fetchClaimSearchContent(true); + } + } + + public void onDownloadAction(String downloadAction, String uri, String outpoint, String fileInfoJson, double progress) { + if ("abort".equals(downloadAction)) { + if (contentListAdapter != null) { + contentListAdapter.clearFileForClaimOrUrl(outpoint, uri); + } + return; + } + + try { + JSONObject fileInfo = new JSONObject(fileInfoJson); + LbryFile claimFile = LbryFile.fromJSONObject(fileInfo); + String claimId = claimFile.getClaimId(); + if (contentListAdapter != null) { + contentListAdapter.updateFileForClaimByIdOrUrl(claimFile, claimId, uri); + } + } catch (JSONException ex) { + // invalid file info for download + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/findcontent/EditorsChoiceFragment.java b/app/src/main/java/io/lbry/browser/ui/findcontent/EditorsChoiceFragment.java new file mode 100644 index 00000000..cd8dd19e --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/findcontent/EditorsChoiceFragment.java @@ -0,0 +1,162 @@ +package io.lbry.browser.ui.findcontent; + +import android.content.Context; +import android.content.SharedPreferences; +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 androidx.annotation.NonNull; +import androidx.preference.PreferenceManager; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.EditorsChoiceItemAdapter; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.EditorsChoiceItem; +import io.lbry.browser.tasks.claim.ClaimSearchResultHandler; +import io.lbry.browser.tasks.claim.ClaimSearchTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; + +public class EditorsChoiceFragment extends BaseFragment { + + private static final HashMap titleChannelIdsMap = new LinkedHashMap<>(); + static { + titleChannelIdsMap.put("Short Films", "7056f8267188fc49cd3f7162b4115d9e3c8216f6"); + titleChannelIdsMap.put("Feature-Length Films", "7aad6f36f61da95cb02471fae55f736b28e3bca7"); + titleChannelIdsMap.put("Documentaries", "d57c606e11462e821d5596430c336b58716193bb"); + titleChannelIdsMap.put("Episodic Content", "ea5fc1bd3e1335776fe2641a539a47850606d7db"); + } + + private boolean contentLoading; + private ProgressBar loading; + private RecyclerView contentList; + private EditorsChoiceItemAdapter contentListAdapter; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_editors_choice, container, false); + + loading = root.findViewById(R.id.editors_choice_loading); + contentList = root.findViewById(R.id.editors_choice_content_list); + + LinearLayoutManager llm = new LinearLayoutManager(getContext()); + contentList.setLayoutManager(llm); + + return root; + } + + private Map buildContentOptions() { + Context context = getContext(); + boolean canShowMatureContent = false; + if (context != null) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + canShowMatureContent = sp.getBoolean(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT, false); + } + + return Lbry.buildClaimSearchOptions( + Claim.TYPE_REPOST, + null, + null, /*canShowMatureContent ? null : new ArrayList<>(Predefined.MATURE_TAGS),*/ + new ArrayList<>(titleChannelIdsMap.values()), + null, + Arrays.asList(Claim.ORDER_BY_RELEASE_TIME), + null, + 1, + 99); + } + + public void onResume() { + super.onResume(); + Context context = getContext(); + Helper.setWunderbarValue(null, context); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Editor's Choice", "EditorsChoice"); + } + + if (contentListAdapter == null || contentListAdapter.getItemCount() == 0) { + fetchClaimSearchContent(); + } else { + if (contentList != null) { + contentList.setAdapter(contentListAdapter); + } + } + } + + public void fetchClaimSearchContent() { + if (contentLoading) { + return; + } + + contentLoading = true; + ClaimSearchTask task = new ClaimSearchTask(buildContentOptions(), Lbry.LBRY_TV_CONNECTION_STRING, loading, new ClaimSearchResultHandler() { + @Override + public void onSuccess(List items, boolean hasReachedEnd) { + List data = buildDataFromClaims(items); + if (contentListAdapter == null) { + contentListAdapter = new EditorsChoiceItemAdapter(data, getContext()); + contentListAdapter.setListener(new EditorsChoiceItemAdapter.EditorsChoiceItemListener() { + @Override + public void onEditorsChoiceItemClicked(EditorsChoiceItem item) { + String url = item.getPermanentUrl(); + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openFileUrl(url); + } + } + }); + } else { + contentListAdapter.addItems(data); + } + + if (contentList != null && contentList.getAdapter() == null) { + contentList.setAdapter(contentListAdapter); + } + contentLoading = false; + } + + @Override + public void onError(Exception error) { + contentLoading = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private List buildDataFromClaims(List claims) { + List data = new ArrayList<>(); + for (String title : titleChannelIdsMap.keySet()) { + EditorsChoiceItem titleItem = new EditorsChoiceItem(); + titleItem.setTitle(title); + titleItem.setHeader(true); + data.add(titleItem); + + String channelClaimId = titleChannelIdsMap.get(title); + for (Claim c : claims) { + if (c.getSigningChannel() != null && channelClaimId.equalsIgnoreCase(c.getSigningChannel().getClaimId())) { + EditorsChoiceItem item = EditorsChoiceItem.fromClaim( + Claim.TYPE_REPOST.equalsIgnoreCase(c.getValueType()) ? c.getRepostedClaim() : c); + data.add(item); + } + } + } + + return data; + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/findcontent/FileViewFragment.java b/app/src/main/java/io/lbry/browser/ui/findcontent/FileViewFragment.java new file mode 100644 index 00000000..3b1a7b82 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/findcontent/FileViewFragment.java @@ -0,0 +1,2144 @@ +package io.lbry.browser.ui.findcontent; + +import android.Manifest; +import android.annotation.SuppressLint; +import android.content.BroadcastReceiver; +import android.content.ComponentName; +import android.content.Context; +import android.content.DialogInterface; +import android.content.Intent; +import android.content.SharedPreferences; +import android.content.pm.ActivityInfo; +import android.graphics.Color; +import android.net.Uri; +import android.os.AsyncTask; +import android.os.Bundle; +import android.text.format.DateUtils; +import android.view.ContextMenu; +import android.view.LayoutInflater; +import android.view.MenuItem; +import android.view.View; +import android.view.ViewGroup; +import android.view.WindowManager; +import android.webkit.WebResourceRequest; +import android.webkit.WebSettings; +import android.webkit.WebView; +import android.webkit.WebViewClient; +import android.widget.ImageView; +import android.widget.ProgressBar; +import android.widget.RelativeLayout; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.appcompat.app.AlertDialog; +import androidx.constraintlayout.widget.ConstraintLayout; +import androidx.core.content.ContextCompat; +import androidx.media.session.MediaButtonReceiver; +import androidx.preference.PreferenceManager; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import com.bumptech.glide.Glide; +import com.github.chrisbanes.photoview.PhotoView; +import com.google.android.exoplayer2.C; +import com.google.android.exoplayer2.ParserException; +import com.google.android.exoplayer2.PlaybackParameters; +import com.google.android.exoplayer2.Player; +import com.google.android.exoplayer2.SimpleExoPlayer; +import com.google.android.exoplayer2.audio.AudioAttributes; +import com.google.android.exoplayer2.database.ExoDatabaseProvider; +import com.google.android.exoplayer2.ext.mediasession.MediaSessionConnector; +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.upstream.DefaultLoadErrorHandlingPolicy; +import com.google.android.exoplayer2.upstream.Loader; +import com.google.android.exoplayer2.upstream.cache.CacheDataSourceFactory; +import com.google.android.exoplayer2.upstream.cache.LeastRecentlyUsedCacheEvictor; +import com.google.android.exoplayer2.upstream.cache.SimpleCache; +import com.google.android.exoplayer2.util.Util; +import com.google.android.flexbox.FlexboxLayoutManager; +import com.google.android.material.button.MaterialButton; +import com.google.android.material.snackbar.Snackbar; + +import org.commonmark.node.Node; +import org.commonmark.parser.Parser; +import org.commonmark.renderer.html.HtmlRenderer; +import org.json.JSONException; +import org.json.JSONObject; + +import java.io.File; +import java.io.FileNotFoundException; +import java.io.IOException; +import java.math.BigDecimal; +import java.text.DecimalFormat; +import java.text.NumberFormat; +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.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.ClaimListAdapter; +import io.lbry.browser.adapter.TagListAdapter; +import io.lbry.browser.dialog.RepostClaimDialogFragment; +import io.lbry.browser.dialog.SendTipDialogFragment; +import io.lbry.browser.exceptions.LbryUriException; +import io.lbry.browser.listener.DownloadActionListener; +import io.lbry.browser.listener.FetchClaimsListener; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.listener.StoragePermissionListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.ClaimCacheKey; +import io.lbry.browser.model.Fee; +import io.lbry.browser.model.LbryFile; +import io.lbry.browser.model.NavMenuItem; +import io.lbry.browser.model.Tag; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.model.lbryinc.Reward; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.tasks.LighthouseSearchTask; +import io.lbry.browser.tasks.ReadTextFileTask; +import io.lbry.browser.tasks.SetSdkSettingTask; +import io.lbry.browser.tasks.claim.AbandonHandler; +import io.lbry.browser.tasks.claim.AbandonStreamTask; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ClaimSearchResultHandler; +import io.lbry.browser.tasks.claim.ResolveTask; +import io.lbry.browser.tasks.file.DeleteFileTask; +import io.lbry.browser.tasks.file.FileListTask; +import io.lbry.browser.tasks.file.GetFileTask; +import io.lbry.browser.tasks.lbryinc.ChannelSubscribeTask; +import io.lbry.browser.tasks.lbryinc.ClaimRewardTask; +import io.lbry.browser.tasks.lbryinc.FetchStatCountTask; +import io.lbry.browser.tasks.lbryinc.LogFileViewTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.ui.controls.SolidIconView; +import io.lbry.browser.ui.publish.PublishFormFragment; +import io.lbry.browser.ui.publish.PublishFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; +import io.lbry.lbrysdk.DownloadManager; +import io.lbry.lbrysdk.LbrynetService; +import io.lbry.lbrysdk.Utils; + +public class FileViewFragment extends BaseFragment implements + MainActivity.BackPressInterceptor, DownloadActionListener, FetchClaimsListener, SdkStatusListener, StoragePermissionListener { + private static final int RELATED_CONTENT_SIZE = 16; + private static final String DEFAULT_PLAYBACK_SPEED = "1x"; + + private PlayerControlView castControlView; + private Player currentPlayer; + private boolean loadingNewClaim; + private boolean startDownloadPending; + private boolean fileGetPending; + private boolean downloadInProgress; + private boolean downloadRequested; + private boolean loadFilePending; + private boolean resolving; + private boolean initialFileLoadDone; + private Claim claim; + private String currentUrl; + private ClaimListAdapter relatedContentAdapter; + private BroadcastReceiver sdkReceiver; + private Player.EventListener fileViewPlayerListener; + + private long elapsedDuration = 0; + private long totalDuration = 0; + private boolean elapsedPlaybackScheduled; + private ScheduledExecutorService elapsedPlaybackScheduler; + private boolean playbackStarted; + private long startTimeMillis; + private GetFileTask getFileTask; + + private View buttonPublishSomething; + private View layoutLoadingState; + private View layoutNothingAtLocation; + private View layoutDisplayArea; + private View layoutResolving; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_file_view, container, false); + + layoutLoadingState = root.findViewById(R.id.file_view_loading_state); + layoutNothingAtLocation = root.findViewById(R.id.container_nothing_at_location); + layoutResolving = root.findViewById(R.id.file_view_loading_container); + layoutDisplayArea = root.findViewById(R.id.file_view_claim_display_area); + buttonPublishSomething = root.findViewById(R.id.nothing_at_location_publish_button); + + initUi(root); + + fileViewPlayerListener = new Player.EventListener() { + @Override + public void onPlayerStateChanged(boolean playWhenReady, int playbackState) { + if (playbackState == Player.STATE_READY) { + elapsedDuration = MainActivity.appPlayer.getCurrentPosition(); + totalDuration = MainActivity.appPlayer.getDuration() < 0 ? 0 : MainActivity.appPlayer.getDuration(); + if (!playbackStarted) { + logPlay(currentUrl, startTimeMillis); + playbackStarted = true; + } + renderTotalDuration(); + scheduleElapsedPlayback(); + hideBuffering(); + + if (loadingNewClaim) { + MainActivity.appPlayer.setPlayWhenReady(true); + loadingNewClaim = false; + } + } else if (playbackState == Player.STATE_BUFFERING) { + showBuffering(); + } else { + hideBuffering(); + } + } + }; + + return root; + } + + public void onStart() { + super.onStart(); + Context context = getContext(); + + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.setBackPressInterceptor(this); + activity.addDownloadActionListener(this); + activity.addFetchClaimsListener(this); + if (!MainActivity.hasPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, context)) { + activity.addStoragePermissionListener(this); + } + } + } + + private void checkParams() { + boolean updateRequired = false; + Context context = getContext(); + Map params = getParams(); + Claim newClaim = null; + String newUrl = null; + if (params != null) { + if (params.containsKey("claim")) { + newClaim = (Claim) params.get("claim"); + if (newClaim != null && !newClaim.equals(this.claim)) { + updateRequired = true; + } + } + if (params.containsKey("url")) { + newUrl = params.get("url").toString(); + if (claim == null || !newUrl.equalsIgnoreCase(currentUrl)) { + updateRequired = true; + } + } + } else if (currentUrl != null) { + updateRequired = true; + } else if (context instanceof MainActivity) { + ((MainActivity) context).onBackPressed(); + } + + if (updateRequired) { + if (context instanceof MainActivity) { + ((MainActivity) context).clearNowPlayingClaim(); + } + if (MainActivity.appPlayer != null) { + MainActivity.appPlayer.setPlayWhenReady(false); + } + + resetViewCount(); + resetFee(); + checkNewClaimAndUrl(newClaim, newUrl); + + if (newClaim != null) { + claim = newClaim; + } + if (!Helper.isNullOrEmpty(newUrl)) { + // check if the claim is already cached + currentUrl = newUrl; + ClaimCacheKey key = new ClaimCacheKey(); + key.setUrl(currentUrl); + onNewClaim(currentUrl); + if (Lbry.claimCache.containsKey(key)) { + claim = Lbry.claimCache.get(key); + } else { + resolveUrl(currentUrl); + } + } else if (claim == null) { + // nothing at this location + renderNothingAtLocation(); + } + } else { + checkAndResetNowPlayingClaim(); + } + + if (!Helper.isNullOrEmpty(currentUrl)) { + Helper.saveUrlHistory(currentUrl, claim != null ? claim.getTitle() : null, UrlSuggestion.TYPE_FILE); + } + + if (claim != null) { + Helper.saveViewHistory(currentUrl, claim); + renderClaim(); + if (claim.getFile() == null) { + loadFile(); + } else { + initialFileLoadDone = true; + } + } + + checkIsFileComplete(); + } + + private void renderNothingAtLocation() { + Helper.setViewVisibility(layoutLoadingState, View.VISIBLE); + Helper.setViewVisibility(layoutNothingAtLocation, View.VISIBLE); + Helper.setViewVisibility(buttonPublishSomething, View.VISIBLE); + Helper.setViewVisibility(layoutResolving, View.GONE); + Helper.setViewVisibility(layoutDisplayArea, View.INVISIBLE); + } + + private void checkNewClaimAndUrl(Claim newClaim, String newUrl) { + boolean shouldResetNowPlaying = false; + if (newClaim != null && + MainActivity.nowPlayingClaim != null && + !MainActivity.nowPlayingClaim.getClaimId().equalsIgnoreCase(newClaim.getClaimId())) { + shouldResetNowPlaying = true; + } + if (!shouldResetNowPlaying && + newUrl != null && + MainActivity.nowPlayingClaim != null && + !newUrl.equalsIgnoreCase(MainActivity.nowPlayingClaim.getShortUrl()) && + !newUrl.equalsIgnoreCase(MainActivity.nowPlayingClaim.getPermanentUrl())) { + shouldResetNowPlaying = true; + } + + if (shouldResetNowPlaying) { + if (MainActivity.appPlayer != null) { + MainActivity.appPlayer.setPlayWhenReady(false); + } + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).clearNowPlayingClaim(); + resetPlayer(); + } + } + } + + @Override + public void onActivityCreated(Bundle savedInstanceState) { + super.onActivityCreated(savedInstanceState); + if (savedInstanceState != null) { + currentUrl = savedInstanceState.getString("url"); + } + } + + @Override + public void onSaveInstanceState(Bundle savedInstanceState) { + super.onSaveInstanceState(savedInstanceState); + if (savedInstanceState != null) { + savedInstanceState.putString("url", currentUrl); + } + } + + private void initWebView(View view) { + WebView webView = view.findViewById(R.id.file_view_webview); + webView.setWebViewClient(new LbryWebViewClient(getContext())); + WebSettings webSettings = webView.getSettings(); + webSettings.setAllowFileAccess(true); + webSettings.setJavaScriptEnabled(true); + } + + private void logUrlEvent(String url) { + Bundle bundle = new Bundle(); + bundle.putString("uri", url); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_OPEN_FILE_PAGE, bundle); + } + + private void checkAndResetNowPlayingClaim() { + if (MainActivity.nowPlayingClaim != null + && claim != null && + !MainActivity.nowPlayingClaim.getClaimId().equalsIgnoreCase(claim.getClaimId())) { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.clearNowPlayingClaim(); + if (claim != null && !claim.isPlayable()) { + activity.stopExoplayer(); + } + } + } + } + + private void onNewClaim(String url) { + loadingNewClaim = true; + initialFileLoadDone = false; + playbackStarted = false; + currentUrl = url; + logUrlEvent(url); + resetViewCount(); + resetFee(); + View root = getView(); + if (root != null) { + ((RecyclerView) root.findViewById(R.id.file_view_related_content_list)).setAdapter(null); + } + if (MainActivity.appPlayer != null) { + MainActivity.appPlayer.setPlayWhenReady(false); + } + resetPlayer(); + } + + public void onSdkReady() { + if (loadFilePending) { + loadFile(); + } + checkOwnClaim(); + } + + private String getStreamingUrl() { + LbryFile lbryFile = claim.getFile(); + if (lbryFile != null) { + if (!Helper.isNullOrEmpty(lbryFile.getDownloadPath()) && lbryFile.isCompleted()) { + File file = new File(lbryFile.getDownloadPath()); + if (file.exists()) { + return Uri.fromFile(file).toString(); + } + } + + if (!Helper.isNullOrEmpty(lbryFile.getStreamingUrl())) { + return lbryFile.getStreamingUrl(); + } + } + + return buildLbryTvStreamingUrl(); + } + + private String buildLbryTvStreamingUrl() { + return String.format("https://cdn.lbryplayer.xyz/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; + String claimId = claim.getClaimId(); + FileListTask task = new FileListTask(claimId, null, new FileListTask.FileListResultHandler() { + @Override + public void onSuccess(List files, boolean hasReachedEnd) { + if (files.size() > 0) { + claim.setFile(files.get(0)); + checkIsFileComplete(); + } + initialFileLoadDone = true; + } + + @Override + public void onError(Exception error) { + initialFileLoadDone = true; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void openClaimUrl(String url) { + resetViewCount(); + resetFee(); + currentUrl = url; + + ClaimCacheKey key = new ClaimCacheKey(); + key.setUrl(currentUrl); + Claim oldClaim = claim; + claim = null; + if (Lbry.claimCache.containsKey(key)) { + claim = Lbry.claimCache.get(key); + if (oldClaim != null && oldClaim.getClaimId().equalsIgnoreCase(claim.getClaimId())) { + // same claim + return; + } + } else { + resolveUrl(currentUrl); + } + + resetMedia(); + onNewClaim(currentUrl); + Helper.setWunderbarValue(currentUrl, getContext()); + + if (claim != null) { + Helper.saveViewHistory(url, claim); + renderClaim(); + } + } + + public void resetMedia() { + View root = getView(); + if (root != null) { + PlayerView view = root.findViewById(R.id.file_view_exoplayer_view); + view.setShutterBackgroundColor(Color.BLACK); + root.findViewById(R.id.file_view_exoplayer_container).setVisibility(View.GONE); + } + if (MainActivity.appPlayer != null) { + MainActivity.appPlayer.stop(); + } + resetPlayer(); + } + + public void onResume() { + super.onResume(); + checkParams(); + + Context context = getContext(); + Helper.setWunderbarValue(currentUrl, context); + if (context instanceof MainActivity) { + LbryAnalytics.setCurrentScreen((MainActivity) context, "File", "File"); + } + + if (MainActivity.appPlayer != null) { + if (MainActivity.playerReassigned) { + setPlayerForPlayerView(); + MainActivity.playerReassigned = false; + } + loadAndScheduleDurations(); + } + + if (Lbry.SDK_READY) { + if (context instanceof MainActivity) { + ((MainActivity) context).addSdkStatusListener(this); + } + } else { + onSdkReady(); + } + } + + public void onStop() { + super.onStop(); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.removeDownloadActionListener(this); + activity.removeFetchClaimsListener(this); + activity.removeSdkStatusListener(this); + activity.removeStoragePermissionListener(this); + } + } + + private void setPlayerForPlayerView() { + View root = getView(); + if (root != null) { + PlayerView view = root.findViewById(R.id.file_view_exoplayer_view); + view.setPlayer(null); + view.setPlayer(MainActivity.appPlayer); + } + } + + private void resolveUrl(String url) { + resolving = true; + Helper.setViewVisibility(layoutDisplayArea, View.INVISIBLE); + Helper.setViewVisibility(layoutLoadingState, View.VISIBLE); + Helper.setViewVisibility(layoutNothingAtLocation, View.GONE); + ResolveTask task = new ResolveTask(url, Lbry.LBRY_TV_CONNECTION_STRING, layoutResolving, new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + if (claims.size() > 0 && !Helper.isNullOrEmpty(claims.get(0).getClaimId())) { + claim = claims.get(0); + if (Claim.TYPE_REPOST.equalsIgnoreCase(claim.getValueType())) { + claim = claim.getRepostedClaim(); + // cache the reposted claim too for subsequent loads + Lbry.addClaimToCache(claim); + if (claim.getName().startsWith("@")) { + // this is a reposted channel, so finish this activity and launch the channel url + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openChannelUrl(!Helper.isNullOrEmpty(claim.getShortUrl()) ? claim.getShortUrl() : claim.getPermanentUrl()); + } + return; + } + } else { + Lbry.addClaimToCache(claim); + } + + Helper.saveUrlHistory(url, claim.getTitle(), UrlSuggestion.TYPE_FILE); + + // also save view history + Helper.saveViewHistory(url, claim); + + checkAndResetNowPlayingClaim(); + loadFile(); + renderClaim(); + } else { + // render nothing at location + renderNothingAtLocation(); + } + } + + @Override + public void onError(Exception error) { + resolving = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void initUi(View root) { + initWebView(root); + + buttonPublishSomething.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + Context context = getContext(); + if (!Helper.isNullOrEmpty(currentUrl) && context instanceof MainActivity) { + LbryUri uri = LbryUri.tryParse(currentUrl); + if (uri != null) { + Map params = new HashMap<>(); + params.put("suggestedUrl", uri.getStreamName()); + ((MainActivity) context).openFragment(PublishFragment.class, true, NavMenuItem.ID_ITEM_NEW_PUBLISH, params); + } + } + } + }); + + root.findViewById(R.id.file_view_title_area).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + ImageView descIndicator = root.findViewById(R.id.file_view_desc_toggle_arrow); + View descriptionArea = root.findViewById(R.id.file_view_description_area); + + boolean hasDescription = claim != null && !Helper.isNullOrEmpty(claim.getDescription()); + boolean hasTags = claim != null && claim.getTags() != null && claim.getTags().size() > 0; + + if (descriptionArea.getVisibility() != View.VISIBLE) { + if (hasDescription || hasTags) { + descriptionArea.setVisibility(View.VISIBLE); + } + descIndicator.setImageResource(R.drawable.ic_arrow_dropup); + } else { + descriptionArea.setVisibility(View.GONE); + descIndicator.setImageResource(R.drawable.ic_arrow_dropdown); + } + } + }); + + root.findViewById(R.id.file_view_action_share).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (claim != null) { + try { + String shareUrl = LbryUri.parse( + !Helper.isNullOrEmpty(claim.getCanonicalUrl()) ? claim.getCanonicalUrl() : + (!Helper.isNullOrEmpty(claim.getShortUrl()) ? claim.getShortUrl() : claim.getPermanentUrl())).toTvString(); + Intent shareIntent = new Intent(); + shareIntent.setAction(Intent.ACTION_SEND); + shareIntent.setType("text/plain"); + shareIntent.putExtra(Intent.EXTRA_TEXT, shareUrl); + + MainActivity.startingShareActivity = true; + Intent shareUrlIntent = Intent.createChooser(shareIntent, getString(R.string.share_lbry_content)); + shareUrlIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); + startActivity(shareUrlIntent); + } catch (LbryUriException ex) { + // pass + } + } + } + }); + + root.findViewById(R.id.file_view_action_tip).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (!Lbry.SDK_READY) { + Snackbar.make(root.findViewById(R.id.file_view_claim_display_area), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + if (claim != null) { + SendTipDialogFragment dialog = SendTipDialogFragment.newInstance(); + dialog.setClaim(claim); + dialog.setListener(new SendTipDialogFragment.SendTipListener() { + @Override + public void onTipSent(BigDecimal amount) { + double sentAmount = amount.doubleValue(); + String message = getResources().getQuantityString( + R.plurals.you_sent_a_tip, sentAmount == 1.0 ? 1 : 2, + new DecimalFormat("#,###.##").format(sentAmount)); + Snackbar.make(root.findViewById(R.id.file_view_claim_display_area), message, Snackbar.LENGTH_LONG).show(); + } + }); + Context context = getContext(); + if (context instanceof MainActivity) { + dialog.show(((MainActivity) context).getSupportFragmentManager(), SendTipDialogFragment.TAG); + } + } + } + }); + + root.findViewById(R.id.file_view_action_repost).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (!Lbry.SDK_READY) { + Snackbar.make(root.findViewById(R.id.file_view_claim_display_area), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + if (claim != null) { + RepostClaimDialogFragment dialog = RepostClaimDialogFragment.newInstance(); + dialog.setClaim(claim); + dialog.setListener(new RepostClaimDialogFragment.RepostClaimListener() { + @Override + public void onClaimReposted(Claim claim) { + Snackbar.make(root.findViewById(R.id.file_view_claim_display_area), R.string.content_successfully_reposted, Snackbar.LENGTH_LONG).show(); + } + }); + Context context = getContext(); + if (context instanceof MainActivity) { + dialog.show(((MainActivity) context).getSupportFragmentManager(), RepostClaimDialogFragment.TAG); + } + } + } + }); + + root.findViewById(R.id.file_view_action_edit).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (!Lbry.SDK_READY) { + Snackbar.make(root.findViewById(R.id.file_view_claim_display_area), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + Context context = getContext(); + if (claim != null && context instanceof MainActivity) { + ((MainActivity) context).openPublishForm(claim); + } + } + }); + + root.findViewById(R.id.file_view_action_delete).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (!Lbry.SDK_READY) { + Snackbar.make(root.findViewById(R.id.file_view_claim_display_area), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + if (claim != null) { + boolean isOwnClaim = Lbry.ownClaims.contains(claim); + if (isOwnClaim) { + AlertDialog.Builder builder = new AlertDialog.Builder(getContext()). + setTitle(R.string.delete_content). + setMessage(R.string.confirm_delete_content_message) + .setPositiveButton(R.string.yes, new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + deleteCurrentClaim(); + } + }).setNegativeButton(R.string.no, null); + builder.show(); + } else { + AlertDialog.Builder builder = new AlertDialog.Builder(getContext()). + setTitle(R.string.delete_file). + setMessage(R.string.confirm_delete_file_message) + .setPositiveButton(R.string.yes, new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + deleteClaimFile(); + } + }).setNegativeButton(R.string.no, null); + builder.show(); + } + } + } + }); + + root.findViewById(R.id.file_view_action_download).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (!Lbry.SDK_READY) { + Snackbar.make(root.findViewById(R.id.file_view_claim_display_area), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + if (claim != null) { + if (downloadInProgress) { + onDownloadAborted(); + + // file is already downloading and not completed + Intent intent = new Intent(LbrynetService.ACTION_DELETE_DOWNLOAD); + intent.putExtra("uri", claim.getPermanentUrl()); + intent.putExtra("nativeDelete", true); + Context context = getContext(); + if (context != null) { + context.sendBroadcast(intent); + } + } else { + checkStoragePermissionAndStartDownload(); + } + } + } + }); + + root.findViewById(R.id.file_view_action_report).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (claim != null) { + Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(String.format("https://lbry.com/dmca/%s", claim.getClaimId()))); + startActivity(intent); + } + } + }); + + root.findViewById(R.id.player_toggle_cast).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + toggleCast(); + } + }); + + PlayerView playerView = root.findViewById(R.id.file_view_exoplayer_view); + View playbackSpeedContainer = playerView.findViewById(R.id.player_playback_speed); + TextView textPlaybackSpeed = playerView.findViewById(R.id.player_playback_speed_label); + textPlaybackSpeed.setText(DEFAULT_PLAYBACK_SPEED); + + playbackSpeedContainer.setOnCreateContextMenuListener(new View.OnCreateContextMenuListener() { + @Override + public void onCreateContextMenu(ContextMenu contextMenu, View view, ContextMenu.ContextMenuInfo contextMenuInfo) { + Helper.buildPlaybackSpeedMenu(contextMenu); + } + }); + playbackSpeedContainer.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openContextMenu(playbackSpeedContainer); + } + } + }); + + playerView.findViewById(R.id.player_toggle_fullscreen).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + // check full screen mode + if (isInFullscreenMode()) { + disableFullScreenMode(); + } else { + enableFullScreenMode(); + } + } + }); + playerView.findViewById(R.id.player_skip_back_10).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (MainActivity.appPlayer != null) { + MainActivity.appPlayer.seekTo(Math.max(0, MainActivity.appPlayer.getCurrentPosition() - 10000)); + } + } + }); + playerView.findViewById(R.id.player_skip_forward_10).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (MainActivity.appPlayer != null) { + MainActivity.appPlayer.seekTo(MainActivity.appPlayer.getCurrentPosition() + 10000); + } + } + }); + + root.findViewById(R.id.file_view_publisher_name).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (claim != null && claim.getSigningChannel() != null) { + Claim publisher = claim.getSigningChannel(); + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openChannelUrl( + !Helper.isNullOrEmpty(publisher.getShortUrl()) ? publisher.getShortUrl() : publisher.getPermanentUrl()); + } + } + } + }); + + View buttonFollowUnfollow = root.findViewById(R.id.file_view_icon_follow_unfollow); + buttonFollowUnfollow.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (claim != null && claim.getSigningChannel() != null) { + Claim publisher = claim.getSigningChannel(); + boolean isFollowing = Lbryio.isFollowing(publisher); + Subscription subscription = Subscription.fromClaim(publisher); + buttonFollowUnfollow.setEnabled(false); + Context context = getContext(); + new ChannelSubscribeTask(context, publisher.getClaimId(), subscription, isFollowing, new ChannelSubscribeTask.ChannelSubscribeHandler() { + @Override + public void onSuccess() { + if (isFollowing) { + Lbryio.removeSubscription(subscription); + Lbryio.removeCachedResolvedSubscription(publisher); + } else { + Lbryio.addSubscription(subscription); + Lbryio.addCachedResolvedSubscription(publisher); + } + buttonFollowUnfollow.setEnabled(true); + checkIsFollowing(); + FollowingFragment.resetClaimSearchContent = true; + + // Save shared user state + if (context != null) { + context.sendBroadcast(new Intent(MainActivity.ACTION_SAVE_SHARED_USER_STATE)); + } + } + + @Override + public void onError(Exception exception) { + buttonFollowUnfollow.setEnabled(true); + } + }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + }); + + RecyclerView relatedContentList = root.findViewById(R.id.file_view_related_content_list); + relatedContentList.setNestedScrollingEnabled(false); + LinearLayoutManager llm = new LinearLayoutManager(getContext()); + relatedContentList.setLayoutManager(llm); + } + + private void deleteCurrentClaim() { + if (claim != null) { + Helper.setViewVisibility(layoutDisplayArea, View.INVISIBLE); + Helper.setViewVisibility(layoutLoadingState, View.VISIBLE); + Helper.setViewVisibility(layoutNothingAtLocation, View.GONE); + AbandonStreamTask task = new AbandonStreamTask(Arrays.asList(claim.getClaimId()), layoutResolving, new AbandonHandler() { + @Override + public void onComplete(List successfulClaimIds, List failedClaimIds, List errors) { + Context context = getContext(); + if (context instanceof MainActivity) { + if (failedClaimIds.size() == 0) { + MainActivity activity = (MainActivity) context; + activity.showMessage(R.string.content_deleted); + activity.onBackPressed(); + } else { + showError(getString(R.string.content_failed_delete)); + } + } + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + private void checkStoragePermissionAndStartDownload() { + Context context = getContext(); + if (MainActivity.hasPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, context)) { + startDownload(); + } else { + startDownloadPending = true; + MainActivity.requestPermission( + Manifest.permission.WRITE_EXTERNAL_STORAGE, + MainActivity.REQUEST_STORAGE_PERMISSION, + getString(R.string.storage_permission_rationale_download), + context, + true); + } + } + + private void checkStoragePermissionAndFileGet() { + Context context = getContext(); + if (!MainActivity.hasPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, context)) { + fileGetPending = true; + MainActivity.requestPermission( + Manifest.permission.WRITE_EXTERNAL_STORAGE, + MainActivity.REQUEST_STORAGE_PERMISSION, + getString(R.string.storage_permission_rationale_download), + context, + true); + } else { + fileGet(true); + } + } + + public void onStoragePermissionGranted() { + Context context = getContext(); + SetSdkSettingTask task = null; + if (startDownloadPending) { + startDownloadPending = false; + task = new SetSdkSettingTask("download_dir", Utils.getConfiguredDownloadDirectory(context), new GenericTaskHandler() { + @Override + public void beforeStart() { } + @Override + public void onSuccess() { startDownload(); } + @Override + public void onError(Exception error) { + // start the download anyway. Only that it will be saved in the app private folder: /sdcard/Android/io.lbry.browser/Download + startDownload(); + } + }); + } else if (fileGetPending) { + fileGetPending = false; + task = new SetSdkSettingTask("download_dir", Utils.getConfiguredDownloadDirectory(context), new GenericTaskHandler() { + @Override + public void beforeStart() { } + @Override + public void onSuccess() { fileGet(true); } + @Override + public void onError(Exception error) { + // start the file get anyway. Only that it will be saved in the app private folder: /sdcard/Android/io.lbry.browser/Download + fileGet(true); + } + }); + } + if (task != null) { + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + public void onStoragePermissionRefused() { + fileGetPending = false; + startDownloadPending = false; + onDownloadAborted(); + Snackbar.make(getView(), R.string.storage_permission_rationale_download, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + + public void startDownload() { + downloadInProgress = true; + + View root = getView(); + if (root != null) { + Helper.setViewVisibility(root.findViewById(R.id.file_view_download_progress), View.VISIBLE); + ((ImageView) root.findViewById(R.id.file_view_action_download_icon)).setImageResource(R.drawable.ic_stop); + } + + if (!claim.isFree()) { + downloadRequested = true; + onMainActionButtonClicked(); + } else { + // download the file + fileGet(true); + } + } + + private void deleteClaimFile() { + if (claim != null) { + View actionDelete = getView().findViewById(R.id.file_view_action_delete); + DeleteFileTask task = new DeleteFileTask(claim.getClaimId(), new GenericTaskHandler() { + @Override + public void beforeStart() { + actionDelete.setEnabled(false); + } + + @Override + public void onSuccess() { + Helper.setViewVisibility(actionDelete, View.GONE); + View root = getView(); + if (root != null) { + root.findViewById(R.id.file_view_action_download).setVisibility(View.VISIBLE); + root.findViewById(R.id.file_view_unsupported_container).setVisibility(View.GONE); + } + Helper.setViewEnabled(actionDelete, true); + + claim.setFile(null); + Lbry.unsetFilesForCachedClaims(Arrays.asList(claim.getClaimId())); + + restoreMainActionButton(); + } + + @Override + public void onError(Exception error) { + actionDelete.setEnabled(true); + showError(error.getMessage()); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + private void renderClaim() { + if (claim == null) { + return; + } + + if (claim.isPlayable() && MainActivity.appPlayer != null) { + MainActivity.appPlayer.setPlayWhenReady(true); + } + + Helper.setViewVisibility(layoutLoadingState, View.GONE); + Helper.setViewVisibility(layoutNothingAtLocation, View.GONE); + + loadViewCount(); + checkIsFollowing(); + + View root = getView(); + if (root != null) { + root.findViewById(R.id.file_view_scroll_view).scrollTo(0, 0); + Helper.setViewVisibility(layoutDisplayArea, View.VISIBLE); + + ImageView descIndicator = root.findViewById(R.id.file_view_desc_toggle_arrow); + descIndicator.setImageResource(R.drawable.ic_arrow_dropdown); + + boolean hasDescription = !Helper.isNullOrEmpty(claim.getDescription()); + boolean hasTags = claim.getTags() != null && claim.getTags().size() > 0; + + root.findViewById(R.id.file_view_description).setVisibility(hasDescription ? View.VISIBLE : View.GONE); + root.findViewById(R.id.file_view_tag_area).setVisibility(hasTags ? View.VISIBLE : View.GONE); + if (hasTags && !hasDescription) { + root.findViewById(R.id.file_view_tag_area).setPadding(0, 0, 0, 0); + } + + root.findViewById(R.id.file_view_description_area).setVisibility(View.GONE); + ((TextView) root.findViewById(R.id.file_view_title)).setText(claim.getTitle()); + ((TextView) root.findViewById(R.id.file_view_description)).setText(claim.getDescription()); + ((TextView) root.findViewById(R.id.file_view_publisher_name)).setText( + Helper.isNullOrEmpty(claim.getPublisherName()) ? getString(R.string.anonymous) : claim.getPublisherName()); + + Context context = getContext(); + RecyclerView descTagsList = root.findViewById(R.id.file_view_tag_list); + FlexboxLayoutManager flm = new FlexboxLayoutManager(context); + descTagsList.setLayoutManager(flm); + + List tags = claim.getTagObjects(); + TagListAdapter tagListAdapter = new TagListAdapter(tags, context); + tagListAdapter.setClickListener(new TagListAdapter.TagClickListener() { + @Override + public void onTagClicked(Tag tag, int customizeMode) { + if (customizeMode == TagListAdapter.CUSTOMIZE_MODE_NONE) { + Context ctx = getContext(); + if (ctx instanceof MainActivity) { + ((MainActivity) ctx).openAllContentFragmentWithTag(tag.getName()); + } + } + } + }); + descTagsList.setAdapter(tagListAdapter); + root.findViewById(R.id.file_view_tag_area).setVisibility(tags.size() > 0 ? View.VISIBLE : View.GONE); + + root.findViewById(R.id.file_view_exoplayer_container).setVisibility(View.GONE); + root.findViewById(R.id.file_view_unsupported_container).setVisibility(View.GONE); + root.findViewById(R.id.file_view_media_meta_container).setVisibility(View.VISIBLE); + + Claim.GenericMetadata metadata = claim.getValue(); + if (!Helper.isNullOrEmpty(claim.getThumbnailUrl())) { + ImageView thumbnailView = root.findViewById(R.id.file_view_thumbnail); + Glide.with(getContext().getApplicationContext()).load(claim.getThumbnailUrl()).centerCrop().into(thumbnailView); + } else { + // display first x letters of claim name, with random background + } + + root.findViewById(R.id.file_view_main_action_button).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + onMainActionButtonClicked(); + } + }); + root.findViewById(R.id.file_view_media_meta_container).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + onMainActionButtonClicked(); + } + }); + + if (metadata instanceof Claim.StreamMetadata) { + Claim.StreamMetadata streamMetadata = (Claim.StreamMetadata) metadata; + long publishTime = streamMetadata.getReleaseTime() > 0 ? streamMetadata.getReleaseTime() * 1000 : claim.getTimestamp() * 1000; + ((TextView) root.findViewById(R.id.file_view_publish_time)).setText(DateUtils.getRelativeTimeSpanString( + publishTime, System.currentTimeMillis(), 0, DateUtils.FORMAT_ABBREV_RELATIVE)); + + Fee fee = streamMetadata.getFee(); + if (fee != null && Helper.parseDouble(fee.getAmount(), 0) > 0) { + root.findViewById(R.id.file_view_fee_container).setVisibility(View.VISIBLE); + ((TextView) root.findViewById(R.id.file_view_fee)).setText( + Helper.shortCurrencyFormat(claim.getActualCost(Lbryio.LBCUSDRate).doubleValue())); + } + } + + root.findViewById(R.id.file_view_icon_follow_unfollow).setVisibility(claim.getSigningChannel() != null ? View.VISIBLE : View.GONE); + + MaterialButton mainActionButton = root.findViewById(R.id.file_view_main_action_button); + if (claim.isPlayable()) { + mainActionButton.setText(R.string.play); + } else if (claim.isViewable()) { + mainActionButton.setText(R.string.view); + } else { + mainActionButton.setText(R.string.download); + } + + if (claim.isFree()) { + if (claim.isPlayable()) { + if (MainActivity.nowPlayingClaim != null && MainActivity.nowPlayingClaim.getClaimId().equalsIgnoreCase(claim.getClaimId())) { + // claim already playing + showExoplayerView(); + playMedia(); + } else { + onMainActionButtonClicked(); + } + } else if (claim.isViewable() && Lbry.SDK_READY) { + onMainActionButtonClicked(); + } else if (!Lbry.SDK_READY) { + restoreMainActionButton(); + } + } else { + restoreMainActionButton(); + } + + RecyclerView relatedContentList = root.findViewById(R.id.file_view_related_content_list); + if (relatedContentList == null || relatedContentList.getAdapter() == null || relatedContentList.getAdapter().getItemCount() == 0) { + loadRelatedContent(); + } + } + checkOwnClaim(); + } + + private void showUnsupportedView() { + getView().findViewById(R.id.file_view_exoplayer_container).setVisibility(View.GONE); + getView().findViewById(R.id.file_view_unsupported_container).setVisibility(View.VISIBLE); + String fileNameString = ""; + if (claim.getFile() != null) { + LbryFile lbryFile = claim.getFile(); + File file = new File(lbryFile.getDownloadPath()); + fileNameString = String.format("\"%s\" ", file.getName()); + } + ((TextView) getView().findViewById(R.id.file_view_unsupported_text)).setText(getString(R.string.unsupported_content_desc, fileNameString)); + } + + private void showExoplayerView() { + View root = getView(); + if (root != null) { + root.findViewById(R.id.file_view_unsupported_container).setVisibility(View.GONE); + root.findViewById(R.id.file_view_exoplayer_container).setVisibility(View.VISIBLE); + } + } + + private void playMedia() { + boolean newPlayerCreated = false; + + Context context = getContext(); + if (MainActivity.appPlayer == null && context != null) { + AudioAttributes audioAttributes = new AudioAttributes.Builder() + .setUsage(C.USAGE_MEDIA) + .setContentType(C.CONTENT_TYPE_MOVIE) + .build(); + + MainActivity.appPlayer = new SimpleExoPlayer.Builder(context).build(); + MainActivity.appPlayer.setAudioAttributes(audioAttributes, true); + MainActivity.playerCache = + new SimpleCache(context.getCacheDir(), + new LeastRecentlyUsedCacheEvictor(1024 * 1024 * 256), new ExoDatabaseProvider(context)); + if (context instanceof MainActivity) { + ((MainActivity) context).initMediaSession(); + } + + newPlayerCreated = true; + } + + View root = getView(); + if (root != null) { + PlayerView view = root.findViewById(R.id.file_view_exoplayer_view); + view.setShutterBackgroundColor(Color.TRANSPARENT); + view.setPlayer(MainActivity.appPlayer); + view.setUseController(true); + if (context instanceof MainActivity) { + ((MainActivity) context).getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON); + } + + 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; + } + + if (MainActivity.appPlayer != null) { + showBuffering(); + if (fileViewPlayerListener != null) { + MainActivity.appPlayer.addListener(fileViewPlayerListener); + } + if (context instanceof MainActivity) { + ((MainActivity) context).setNowPlayingClaim(claim, currentUrl); + } + + MainActivity.appPlayer.setPlayWhenReady(true); + String userAgent = Util.getUserAgent(context, getString(R.string.app_name)); + String mediaSourceUrl = getStreamingUrl(); + MediaSource mediaSource = new ProgressiveMediaSource.Factory( + new CacheDataSourceFactory(MainActivity.playerCache, new DefaultDataSourceFactory(context, userAgent)), + new DefaultExtractorsFactory() + ).setLoadErrorHandlingPolicy(new StreamLoadErrorPolicy()).createMediaSource(Uri.parse(mediaSourceUrl)); + + MainActivity.appPlayer.prepare(mediaSource, true, true); + } + } + } + + private void setCurrentPlayer(Player currentPlayer) { + if (this.currentPlayer == currentPlayer) { + return; + } + + // View management. + if (currentPlayer == MainActivity.appPlayer) { + //localPlayerView.setVisibility(View.VISIBLE); + castControlView.hide(); + ((ImageView) getView().findViewById(R.id.player_image_cast_toggle)).setImageResource(R.drawable.ic_cast); + } else /* currentPlayer == castPlayer */ { + castControlView.show(); + ((ImageView) getView().findViewById(R.id.player_image_cast_toggle)).setImageResource(R.drawable.ic_cast_connected); + } + + // Player state management. + long playbackPositionMs = C.TIME_UNSET; + int windowIndex = C.INDEX_UNSET; + boolean playWhenReady = false; + + Player previousPlayer = this.currentPlayer; + if (previousPlayer != null) { + // Save state from the previous player. + int playbackState = previousPlayer.getPlaybackState(); + if (playbackState != Player.STATE_ENDED) { + playbackPositionMs = previousPlayer.getCurrentPosition(); + playWhenReady = previousPlayer.getPlayWhenReady(); + } + previousPlayer.stop(true); + } + + this.currentPlayer = currentPlayer; + + // Media queue management. + /*if (currentPlayer == exoPlayer) { + exoPlayer.prepare(concatenatingMediaSource); + }*/ + currentPlayer.seekTo(playbackPositionMs); + currentPlayer.setPlayWhenReady(true); + } + + private void resetViewCount() { + View root = getView(); + if (root != null) { + TextView textViewCount = root.findViewById(R.id.file_view_view_count); + Helper.setViewText(textViewCount, null); + Helper.setViewVisibility(textViewCount, View.GONE); + } + } + private void resetFee() { + View root = getView(); + if (root != null) { + TextView feeView = root.findViewById(R.id.file_view_fee); + feeView.setText(null); + Helper.setViewVisibility(root.findViewById(R.id.file_view_fee_container), View.GONE); + } + } + + private void loadViewCount() { + if (claim != null) { + FetchStatCountTask task = new FetchStatCountTask( + FetchStatCountTask.STAT_VIEW_COUNT, claim.getClaimId(), null, new FetchStatCountTask.FetchStatCountHandler() { + @Override + public void onSuccess(int count) { + try { + String displayText = getResources().getQuantityString(R.plurals.view_count, count, NumberFormat.getInstance().format(count)); + View root = getView(); + if (root != null) { + TextView textViewCount = root.findViewById(R.id.file_view_view_count); + Helper.setViewText(textViewCount, displayText); + Helper.setViewVisibility(textViewCount, View.VISIBLE); + } + } catch (IllegalStateException ex) { + // pass + } + } + + @Override + public void onError(Exception error) { + // pass + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + private void onMainActionButtonClicked() { + // Check if the claim is free + Claim.GenericMetadata metadata = claim.getValue(); + if (metadata instanceof Claim.StreamMetadata) { + Claim.StreamMetadata streamMetadata = (Claim.StreamMetadata) metadata; + if (claim.getFile() == null && !claim.isFree()) { + // not free (and the user does not own the claim yet), perform a purchase + confirmPurchaseUrl(); + } else { + if (!claim.isPlayable() && !Lbry.SDK_READY) { + Snackbar.make(getView().findViewById(R.id.file_view_global_layout), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + getView().findViewById(R.id.file_view_main_action_button).setVisibility(View.INVISIBLE); + getView().findViewById(R.id.file_view_main_action_loading).setVisibility(View.VISIBLE); + handleMainActionForClaim(); + } + } else { + showError(getString(R.string.cannot_view_claim)); + } + } + + private void confirmPurchaseUrl() { + if (claim != null) { + Fee fee = ((Claim.StreamMetadata) claim.getValue()).getFee(); + double cost = claim.getActualCost(Lbryio.LBCUSDRate).doubleValue(); + String formattedCost = Helper.LBC_CURRENCY_FORMAT.format(cost); + String message = getResources().getQuantityString( + R.plurals.confirm_purchase_message, + cost == 1 ? 1 : 2, + claim.getTitle(), + formattedCost.equals("0") ? Helper.FULL_LBC_CURRENCY_FORMAT.format(cost) : formattedCost); + AlertDialog.Builder builder = new AlertDialog.Builder(getContext()). + setTitle(R.string.confirm_purchase). + setMessage(message) + .setPositiveButton(R.string.yes, new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + Bundle bundle = new Bundle(); + bundle.putString("uri", currentUrl); + bundle.putString("paid", "true"); + bundle.putDouble("amount", Helper.parseDouble(fee.getAmount(), 0)); + bundle.putDouble("lbc_amount", cost); + bundle.putString("currency", fee.getCurrency()); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_PURCHASE_URI, bundle); + + getView().findViewById(R.id.file_view_main_action_button).setVisibility(View.INVISIBLE); + getView().findViewById(R.id.file_view_main_action_loading).setVisibility(View.VISIBLE); + handleMainActionForClaim(); + } + }).setNegativeButton(R.string.no, null); + builder.show(); + } + } + + private void tryOpenFileOrFileGet() { + if (claim != null) { + String claimId = claim.getClaimId(); + FileListTask task = new FileListTask(claimId, null, new FileListTask.FileListResultHandler() { + @Override + public void onSuccess(List files, boolean hasReachedEnd) { + if (files.size() > 0) { + claim.setFile(files.get(0)); + handleMainActionForClaim(); + checkIsFileComplete(); + } else { + checkStoragePermissionAndFileGet(); + } + } + + @Override + public void onError(Exception error) { + checkStoragePermissionAndFileGet(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + private void handleMainActionForClaim() { + if (Lbry.SDK_READY) { + // Check if the file already exists for the claim + if (claim.getFile() != null) { + playOrViewMedia(); + } else { + // check if the file exists from file list + boolean saveFile = downloadRequested || !claim.isPlayable(); + if (!saveFile) { + startTimeMillis = System.currentTimeMillis(); + fileGet(false); + return; + } else { + tryOpenFileOrFileGet(); + } + } + } else { + if (claim.isPlayable()) { + startTimeMillis = System.currentTimeMillis(); + showExoplayerView(); + playMedia(); + } else { + Snackbar.make(getView().findViewById(R.id.file_view_global_layout), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + } + } + } + + private void fileGet(boolean save) { + if (getFileTask != null && getFileTask.getStatus() != AsyncTask.Status.FINISHED) { + return; + } + getFileTask = new GetFileTask(claim.getPermanentUrl(), save, null, new GetFileTask.GetFileHandler() { + @Override + public void beforeStart() { + + } + + @Override + public void onSuccess(LbryFile file, boolean saveFile) { + // queue the download + if (claim != null) { + if (claim.isFree()) { + // paid is handled differently + Bundle bundle = new Bundle(); + bundle.putString("uri", currentUrl); + bundle.putString("paid", "false"); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_PURCHASE_URI, bundle); + } + + if (!claim.isPlayable()) { + logFileView(claim.getPermanentUrl(), 0); + } + + claim.setFile(file); + if (saveFile) { + // download + String outpoint = String.format("%s:%d", claim.getTxid(), claim.getNout()); + Intent intent = new Intent(LbrynetService.ACTION_QUEUE_DOWNLOAD); + intent.putExtra("outpoint", outpoint); + Context context = getContext(); + if (context != null) { + context.sendBroadcast(intent); + } + } else { + // streaming + playOrViewMedia(); + } + } + } + + @Override + public void onError(Exception error, boolean saveFile) { + try { + showError(getString(R.string.unable_to_view_url, currentUrl)); + if (saveFile) { + onDownloadAborted(); + } + restoreMainActionButton(); + } catch (IllegalStateException ex) { + // pass + } + } + }); + getFileTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void playOrViewMedia() { + boolean handled = false; + String mediaType = claim.getMediaType(); + if (!Helper.isNullOrEmpty(mediaType)) { + if (claim.isPlayable()) { + startTimeMillis = System.currentTimeMillis(); + showExoplayerView(); + playMedia(); + handled = true; + } else if (claim.isViewable()) { + // check type and display + boolean fileExists = false; + LbryFile claimFile = claim.getFile(); + Uri fileUri = null; + if (claimFile != null && !Helper.isNullOrEmpty(claimFile.getDownloadPath())) { + File file = new File(claimFile.getDownloadPath()); + fileUri = Uri.fromFile(file); + fileExists = file.exists(); + } + if (!fileExists) { + showError(getString(R.string.claim_file_not_found, claimFile != null ? claimFile.getDownloadPath() : "")); + } else if (fileUri != null) { + if (mediaType.startsWith("image")) { + // display the image + View container = getView().findViewById(R.id.file_view_imageviewer_container); + PhotoView photoView = getView().findViewById(R.id.file_view_imageviewer); + + Glide.with(getContext().getApplicationContext()).load(fileUri).centerInside().into(photoView); + hideFloatingWalletBalance(); + container.setVisibility(View.VISIBLE); + } else if (mediaType.startsWith("text")) { + // show web view (and parse markdown too) + View container = getView().findViewById(R.id.file_view_webview_container); + WebView webView = getView().findViewById(R.id.file_view_webview); + if (Arrays.asList("text/markdown", "text/md").contains(mediaType.toLowerCase())) { + loadMarkdownFromFile(claimFile.getDownloadPath()); + } else { + webView.loadUrl(fileUri.toString()); + } + hideFloatingWalletBalance(); + container.setVisibility(View.VISIBLE); + } + handled = true; + } + } + } + + if (!handled) { + showUnsupportedView(); + } + } + + private void loadMarkdownFromFile(String filePath) { + ReadTextFileTask task = new ReadTextFileTask(filePath, new ReadTextFileTask.ReadTextFileHandler() { + @Override + public void onSuccess(String text) { + String html = buildMarkdownHtml(text); + WebView webView = getView().findViewById(R.id.file_view_webview); + webView.loadData(html, "text/html", "utf-8"); + } + + @Override + public void onError(Exception error) { + showError(error.getMessage()); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private String buildMarkdownHtml(String markdown) { + Parser parser = Parser.builder().build(); + Node document = parser.parse(markdown); + HtmlRenderer renderer = HtmlRenderer.builder().build(); + String markdownHtml = renderer.render(document); + + return "\n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + "
\n" + + markdownHtml + + "
\n" + + " \n" + + " "; + } + + public void showError(String message) { + View root = getView(); + if (root != null) { + Snackbar.make(root, message, Snackbar.LENGTH_LONG).setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + } + + private void loadRelatedContent() { + // reset the list view + View root = getView(); + if (claim != null && root != null) { + String title = claim.getTitle(); + String claimId = claim.getClaimId(); + ProgressBar relatedLoading = root.findViewById(R.id.file_view_related_content_progress); + Context context = getContext(); + boolean canShowMatureContent = false; + if (context != null) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(context); + canShowMatureContent = sp.getBoolean(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT, false); + } + LighthouseSearchTask relatedTask = new LighthouseSearchTask( + title, RELATED_CONTENT_SIZE, 0, canShowMatureContent, claimId, relatedLoading, new 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, context); + relatedContentAdapter.setListener(new ClaimListAdapter.ClaimListItemListener() { + @Override + public void onClaimClicked(Claim claim) { + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (claim.getName().startsWith("@")) { + activity.openChannelUrl(claim.getPermanentUrl()); + } else { + activity.openFileUrl(claim.getPermanentUrl()); //openClaimUrl(claim.getPermanentUrl()); + } + } + } + }); + + View v = getView(); + if (v != null) { + RecyclerView relatedContentList = root.findViewById(R.id.file_view_related_content_list); + relatedContentList.setAdapter(relatedContentAdapter); + relatedContentAdapter.notifyDataSetChanged(); + + Helper.setViewVisibility( + v.findViewById(R.id.file_view_no_related_content), + relatedContentAdapter == null || relatedContentAdapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + } + + @Override + public void onError(Exception error) { + + } + }); + relatedTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + public boolean onBackPressed() { + if (isInFullscreenMode()) { + disableFullScreenMode(); + return true; + } + + if (isImageViewerVisible()) { + getView().findViewById(R.id.file_view_imageviewer_container).setVisibility(View.GONE); + restoreMainActionButton(); + showFloatingWalletBalance(); + return true; + } + if (isWebViewVisible()) { + getView().findViewById(R.id.file_view_webview_container).setVisibility(View.GONE); + restoreMainActionButton(); + showFloatingWalletBalance(); + return true; + } + + return false; + } + + private boolean isImageViewerVisible() { + View view = getView(); + return view != null && view.findViewById(R.id.file_view_imageviewer_container).getVisibility() == View.VISIBLE; + } + + private boolean isWebViewVisible() { + View view = getView(); + return view != null && view.findViewById(R.id.file_view_webview_container).getVisibility() == View.VISIBLE; + } + + @SuppressLint("SourceLockedOrientationActivity") + private void enableFullScreenMode() { + Context context = getContext(); + if (context instanceof MainActivity) { + View root = getView(); + ConstraintLayout globalLayout = root.findViewById(R.id.file_view_global_layout); + View exoplayerContainer = root.findViewById(R.id.file_view_exoplayer_container); + ((ViewGroup) exoplayerContainer.getParent()).removeView(exoplayerContainer); + globalLayout.addView(exoplayerContainer); + + View playerView = root.findViewById(R.id.file_view_exoplayer_view); + ((ImageView) playerView.findViewById(R.id.player_image_full_screen_toggle)).setImageResource(R.drawable.ic_fullscreen_exit); + + root.findViewById(R.id.player_image_full_screen_toggle).setVisibility(View.GONE); + + MainActivity activity = (MainActivity) context; + activity.enterFullScreenMode(); + + int statusBarHeight = activity.getStatusBarHeight(); + exoplayerContainer.setPadding(0, 0, 0, statusBarHeight); + + activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE); + } + } + + private void disableFullScreenMode() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + View root = getView(); + + RelativeLayout mediaContainer = root.findViewById(R.id.file_view_media_container); + View exoplayerContainer = root.findViewById(R.id.file_view_exoplayer_container); + ((ViewGroup) exoplayerContainer.getParent()).removeView(exoplayerContainer); + mediaContainer.addView(exoplayerContainer); + + View playerView = root.findViewById(R.id.file_view_exoplayer_view); + ((ImageView) playerView.findViewById(R.id.player_image_full_screen_toggle)).setImageResource(R.drawable.ic_fullscreen); + exoplayerContainer.setPadding(0, 0, 0, 0); + + activity.exitFullScreenMode(); + activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED); + } + } + + private boolean isInFullscreenMode() { + View view = getView(); + if (view != null) { + View exoplayerContainer = view.findViewById(R.id.file_view_exoplayer_container); + return exoplayerContainer.getParent() instanceof ConstraintLayout; + } + return false; + } + + private void scheduleElapsedPlayback() { + if (!elapsedPlaybackScheduled) { + elapsedPlaybackScheduler = Executors.newSingleThreadScheduledExecutor(); + elapsedPlaybackScheduler.scheduleAtFixedRate(new Runnable() { + @Override + public void run() { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).runOnUiThread(new Runnable() { + @Override + public void run() { + if (MainActivity.appPlayer != null) { + elapsedDuration = MainActivity.appPlayer.getCurrentPosition(); + renderElapsedDuration(); + } + } + }); + } + } + }, 0, 500, TimeUnit.MILLISECONDS); + elapsedPlaybackScheduled = true; + } + } + + private void resetPlayer() { + elapsedDuration = 0; + totalDuration = 0; + renderElapsedDuration(); + renderTotalDuration(); + + elapsedPlaybackScheduled = false; + if (elapsedPlaybackScheduler != null) { + elapsedPlaybackScheduler.shutdownNow(); + elapsedPlaybackScheduler = null; + } + + playbackStarted = false; + startTimeMillis = 0; + + if (MainActivity.appPlayer != null) { + MainActivity.appPlayer.removeListener(fileViewPlayerListener); + PlaybackParameters params = new PlaybackParameters(1.0f); + MainActivity.appPlayer.setPlaybackParameters(params); + } + } + + private void showBuffering() { + View view = getView(); + if (view != null) { + view.findViewById(R.id.player_buffering_progress).setVisibility(View.VISIBLE); + } + } + + private void hideBuffering() { + View view = getView(); + if (view != null) { + view.findViewById(R.id.player_buffering_progress).setVisibility(View.INVISIBLE); + } + } + + private void renderElapsedDuration() { + View view = getView(); + if (view != null) { + Helper.setViewText(view.findViewById(R.id.player_duration_elapsed), Helper.formatDuration(Double.valueOf(elapsedDuration / 1000.0).longValue())); + } + } + + private void renderTotalDuration() { + View view = getView(); + if (view != null) { + Helper.setViewText(view.findViewById(R.id.player_duration_total), Helper.formatDuration(Double.valueOf(totalDuration / 1000.0).longValue())); + } + } + + private void loadAndScheduleDurations() { + if (MainActivity.appPlayer != null && playbackStarted) { + elapsedDuration = MainActivity.appPlayer.getCurrentPosition() < 0 ? 0 : MainActivity.appPlayer.getCurrentPosition(); + totalDuration = MainActivity.appPlayer.getDuration() < 0 ? 0 : MainActivity.appPlayer.getDuration(); + + renderElapsedDuration(); + renderTotalDuration(); + scheduleElapsedPlayback(); + } + } + + private void logPlay(String url, long startTimeMillis) { + long timeToStartMillis = startTimeMillis > 0 ? System.currentTimeMillis() - startTimeMillis : 0; + + Bundle bundle = new Bundle(); + bundle.putString("uri", url); + bundle.putLong("time_to_start_ms", timeToStartMillis); + bundle.putLong("time_to_start_seconds", Double.valueOf(timeToStartMillis / 1000.0).longValue()); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_PLAY, bundle); + + logFileView(claim.getPermanentUrl(), timeToStartMillis); + } + + private void logFileView(String url, long timeToStart) { + if (claim != null) { + LogFileViewTask task = new LogFileViewTask(url, claim, timeToStart, new GenericTaskHandler() { + @Override + public void beforeStart() { } + + @Override + public void onSuccess() { + claimEligibleRewards(); + } + + @Override + public void onError(Exception error) { } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + private void checkIsFollowing() { + if (claim != null && claim.getSigningChannel() != null) { + boolean isFollowing = Lbryio.isFollowing(claim.getSigningChannel()); + Context context = getContext(); + View root = getView(); + if (context != null && root != null) { + SolidIconView iconFollowUnfollow = root.findViewById(R.id.file_view_icon_follow_unfollow); + if (iconFollowUnfollow != null) { + iconFollowUnfollow.setText(isFollowing ? R.string.fa_heart_broken : R.string.fa_heart); + iconFollowUnfollow.setTextColor(ContextCompat.getColor(context, isFollowing ? R.color.foreground : R.color.red)); + } + } + } + } + + private void claimEligibleRewards() { + // attempt to claim eligible rewards after viewing or playing a file (fail silently) + Context context = getContext(); + ClaimRewardTask firstStreamTask = new ClaimRewardTask(Reward.TYPE_FIRST_STREAM, null, null, context, eligibleRewardHandler); + ClaimRewardTask dailyViewTask = new ClaimRewardTask(Reward.TYPE_DAILY_VIEW, null, null, context, eligibleRewardHandler); + firstStreamTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + dailyViewTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private ClaimRewardTask.ClaimRewardHandler eligibleRewardHandler = new ClaimRewardTask.ClaimRewardHandler() { + @Override + public void onSuccess(double amountClaimed, String message) { + if (Helper.isNullOrEmpty(message)) { + message = getResources().getQuantityString( + R.plurals.claim_reward_message, + amountClaimed == 1 ? 1 : 2, + new DecimalFormat(Helper.LBC_CURRENCY_FORMAT_PATTERN).format(amountClaimed)); + } + View root = getView(); + Context context = getContext(); + if (root != null) { + Snackbar.make(root.findViewById(R.id.file_view_global_layout), message, Snackbar.LENGTH_LONG).show(); + } else if (context instanceof MainActivity) { + ((MainActivity) context).showMessage(message); + } + } + + @Override + public void onError(Exception error) { + // pass + } + }; + + private void checkIsFileComplete() { + if (claim == null) { + return; + } + View root = getView(); + if (root != null) { + if (claim.getFile() != null && claim.getFile().isCompleted()) { + Helper.setViewVisibility(root.findViewById(R.id.file_view_action_delete), View.VISIBLE); + Helper.setViewVisibility(root.findViewById(R.id.file_view_action_download), View.GONE); + } else { + Helper.setViewVisibility(root.findViewById(R.id.file_view_action_delete), View.GONE); + Helper.setViewVisibility(root.findViewById(R.id.file_view_action_download), View.VISIBLE); + } + + } + } + + private void hideFloatingWalletBalance() { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).hideFloatingWalletBalance(); + } + } + private void showFloatingWalletBalance() { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).showFloatingWalletBalance(); + } + } + + private void toggleCast() { + if (!MainActivity.castPlayer.isCastSessionAvailable()) { + showError(getString(R.string.no_cast_session_available)); + return; + } + + if (currentPlayer == MainActivity.appPlayer) { + setCurrentPlayer(MainActivity.castPlayer); + } else { + setCurrentPlayer(MainActivity.appPlayer); + } + } + + private void onDownloadAborted() { + downloadInProgress = false; + + if (claim != null) { + claim.setFile(null); + } + ((ImageView) getView().findViewById(R.id.file_view_action_download_icon)).setImageResource(R.drawable.ic_download); + Helper.setViewVisibility(getView().findViewById(R.id.file_view_download_progress), View.GONE); + Helper.setViewVisibility(getView().findViewById(R.id.file_view_unsupported_container), View.GONE); + + checkIsFileComplete(); + restoreMainActionButton(); + } + + private void restoreMainActionButton() { + getView().findViewById(R.id.file_view_main_action_loading).setVisibility(View.INVISIBLE); + getView().findViewById(R.id.file_view_main_action_button).setVisibility(View.VISIBLE); + } + + @Override + public void onDownloadAction(String downloadAction, String uri, String outpoint, String fileInfoJson, double progress) { + if (uri == null || outpoint == null || (fileInfoJson == null && !"abort".equals(downloadAction))) { + return; + } + onRelatedDownloadAction(downloadAction, uri, outpoint, fileInfoJson, progress); + if (claim == null || claim != null && !claim.getPermanentUrl().equalsIgnoreCase(uri)) { + return; + } + if ("abort".equals(downloadAction)) { + onDownloadAborted(); + return; + } + + View root = getView(); + if (root != null) { + ImageView downloadIconView = root.findViewById(R.id.file_view_action_download_icon); + ProgressBar downloadProgressView = root.findViewById(R.id.file_view_download_progress); + + try { + JSONObject fileInfo = new JSONObject(fileInfoJson); + LbryFile claimFile = LbryFile.fromJSONObject(fileInfo); + claim.setFile(claimFile); + + if (DownloadManager.ACTION_START.equals(downloadAction)) { + downloadInProgress = true; + Helper.setViewVisibility(downloadProgressView, View.VISIBLE); + downloadProgressView.setProgress(0); + downloadIconView.setImageResource(R.drawable.ic_stop); + } else if (DownloadManager.ACTION_UPDATE.equals(downloadAction)) { + // handle download updated + downloadInProgress = true; + Helper.setViewVisibility(downloadProgressView, View.VISIBLE); + downloadProgressView.setProgress(Double.valueOf(progress).intValue()); + downloadIconView.setImageResource(R.drawable.ic_stop); + } else if (DownloadManager.ACTION_COMPLETE.equals(downloadAction)) { + downloadInProgress = false; + downloadProgressView.setProgress(100); + Helper.setViewVisibility(downloadProgressView, View.GONE); + playOrViewMedia(); + } + checkIsFileComplete(); + } catch (JSONException ex) { + // invalid file info for download + } + } + } + + @Override + public void onClaimsFetched(List claims) { + checkOwnClaim(); + } + + private static class LbryWebViewClient extends WebViewClient { + private Context context; + public LbryWebViewClient(Context context) { + this.context = context; + } + @Override + public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) { + Uri url = request.getUrl(); + if (context != null) { + Intent intent = new Intent(Intent.ACTION_VIEW, url); + context.startActivity(intent); + } + return true; + } + } + + private void onRelatedDownloadAction(String downloadAction, String uri, String outpoint, String fileInfoJson, double progress) { + if ("abort".equals(downloadAction)) { + if (relatedContentAdapter != null) { + relatedContentAdapter.clearFileForClaimOrUrl(outpoint, uri); + } + return; + } + + try { + JSONObject fileInfo = new JSONObject(fileInfoJson); + LbryFile claimFile = LbryFile.fromJSONObject(fileInfo); + String claimId = claimFile.getClaimId(); + if (relatedContentAdapter != null) { + relatedContentAdapter.updateFileForClaimByIdOrUrl(claimFile, claimId, uri); + } + } catch (JSONException ex) { + // invalid file info for download + } + } + + @Override + public boolean onContextItemSelected(MenuItem item) { + View root = getView(); + if (root != null) { + float speed = item.getItemId() / 100.0f; + String speedString = String.format("%sx", new DecimalFormat("0.##").format(speed)); + PlayerView playerView = root.findViewById(R.id.file_view_exoplayer_view); + ((TextView) playerView.findViewById(R.id.player_playback_speed_label)).setText(speedString); + + if (MainActivity.appPlayer != null) { + PlaybackParameters params = new PlaybackParameters(speed); + MainActivity.appPlayer.setPlaybackParameters(params); + } + } + return true; + } + + @Override + public boolean shouldHideGlobalPlayer() { + return true; + } + + private void checkOwnClaim() { + if (claim != null) { + boolean isOwnClaim = Lbry.ownClaims.contains(claim); + View root = getView(); + if (root != null) { + Helper.setViewVisibility(root.findViewById(R.id.file_view_action_download), isOwnClaim ? View.GONE : View.VISIBLE); + Helper.setViewVisibility(root.findViewById(R.id.file_view_action_report), isOwnClaim ? View.GONE : View.VISIBLE); + Helper.setViewVisibility(root.findViewById(R.id.file_view_action_edit), isOwnClaim ? View.VISIBLE : View.GONE); + Helper.setViewVisibility(root.findViewById(R.id.file_view_action_delete), isOwnClaim ? View.VISIBLE : View.GONE); + } + } + } + + private static class StreamLoadErrorPolicy extends DefaultLoadErrorHandlingPolicy { + @Override + public long getRetryDelayMsFor(int dataType, long loadDurationMs, IOException exception, int errorCount) { + return exception instanceof ParserException + || exception instanceof FileNotFoundException + || exception instanceof Loader.UnexpectedLoaderException + ? C.TIME_UNSET + : Math.min((errorCount - 1) * 1000, 5000); + } + + @Override + public int getMinimumLoadableRetryCount(int dataType) { + return Integer.MAX_VALUE; + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/findcontent/FollowingFragment.java b/app/src/main/java/io/lbry/browser/ui/findcontent/FollowingFragment.java new file mode 100644 index 00000000..de7052ca --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/findcontent/FollowingFragment.java @@ -0,0 +1,805 @@ +package io.lbry.browser.ui.findcontent; + +import android.content.Context; +import android.content.SharedPreferences; +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.preference.PreferenceManager; +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 org.json.JSONException; +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +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.dialog.DiscoverDialogFragment; +import io.lbry.browser.exceptions.LbryUriException; +import io.lbry.browser.listener.DownloadActionListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.LbryFile; +import io.lbry.browser.model.lbryinc.Subscription; +import io.lbry.browser.tasks.claim.ClaimSearchResultHandler; +import io.lbry.browser.tasks.lbryinc.ChannelSubscribeTask; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ClaimSearchTask; +import io.lbry.browser.tasks.lbryinc.FetchSubscriptionsTask; +import io.lbry.browser.tasks.claim.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.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; +import io.lbry.browser.utils.Predefined; + +public class FollowingFragment extends BaseFragment implements + FetchSubscriptionsTask.FetchSubscriptionsHandler, + ChannelItemSelectionListener, + DownloadActionListener, + SharedPreferences.OnSharedPreferenceChangeListener { + + public static boolean resetClaimSearchContent; + private static final int SUGGESTED_PAGE_SIZE = 45; + private static final int MIN_SUGGESTED_SUBSCRIBE_COUNT = 5; + + private DiscoverDialogFragment discoverDialog; + private List excludeChannelIdsForDiscover; + 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 View discoverLink; + private int currentSortBy; + private int currentContentFrom; + private String contentReleaseTime; + private List contentSortOrder; + private boolean contentClaimSearchLoading = false; + private boolean suggestedClaimSearchLoading = false; + private View noContentView; + private boolean subscriptionsShown; + + 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); + discoverLink = root.findViewById(R.id.following_discover_link); + noContentView = root.findViewById(R.id.following_no_claim_search_content); + + Context context = getContext(); + GridLayoutManager glm = new GridLayoutManager(context, 3); + suggestedChannelGrid.setLayoutManager(glm); + suggestedChannelGrid.addOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) { + if (suggestedClaimSearchLoading) { + return; + } + + GridLayoutManager lm = (GridLayoutManager) recyclerView.getLayoutManager(); + if (lm != null) { + int visibleItemCount = lm.getChildCount(); + int totalItemCount = lm.getItemCount(); + int pastVisibleItems = lm.findFirstVisibleItemPosition(); + if (pastVisibleItems + visibleItemCount >= totalItemCount) { + if (!suggestedHasReachedEnd) { + // load more + currentSuggestedPage++; + fetchSuggestedChannels(); + } + } + } + } + }); + + 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); + } + } + }); + discoverLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + buildChannelIdsAndUrls(); + currentSuggestedPage = 1; + discoverDialog = DiscoverDialogFragment.newInstance(); + discoverDialog.setAdapter(suggestedChannelAdapter); + discoverDialog.setDialogActionsListener(new DiscoverDialogFragment.DiscoverDialogListener() { + @Override + public void onScrollEndReached() { + if (suggestedClaimSearchLoading) { + return; + } + currentSuggestedPage++; + fetchSuggestedChannels(); + } + @Override + public void onCancel() { + discoverDialog = null; + excludeChannelIdsForDiscover = null; + if (suggestedChannelAdapter != null) { + suggestedChannelAdapter.clearItems(); + } + } + @Override + public void onResume() { + if (suggestedChannelAdapter == null || suggestedChannelAdapter.getItemCount() == 0) { + discoverDialog.setLoading(true); + fetchSuggestedChannels(); + } + } + }); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + discoverDialog.show(activity.getSupportFragmentManager(), DiscoverDialogFragment.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(); + Context context = getContext(); + Helper.setWunderbarValue(null, context); + PreferenceManager.getDefaultSharedPreferences(context).registerOnSharedPreferenceChangeListener(this); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Subscriptions", "Subscriptions"); + activity.addDownloadActionListener(this); + } + + // check if subscriptions exist + if (suggestedChannelAdapter != null) { + showSuggestedChannels(); + if (suggestedChannelGrid != null) { + suggestedChannelGrid.setAdapter(suggestedChannelAdapter); + } + } + + if (Lbryio.subscriptions != null && Lbryio.subscriptions.size() > 0) { + fetchLoadedSubscriptions(true); + } else { + fetchSubscriptions(); + } + } + public void onPause() { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).removeDownloadActionListener(this); + } + PreferenceManager.getDefaultSharedPreferences(context).unregisterOnSharedPreferenceChangeListener(this); + super.onPause(); + } + public void fetchLoadedSubscriptions(boolean showSubscribed) { + subscriptionsList = new ArrayList<>(Lbryio.subscriptions); + buildChannelIdsAndUrls(); + if (Lbryio.cacheResolvedSubscriptions.size() > 0) { + updateChannelFilterListAdapter(Lbryio.cacheResolvedSubscriptions, resetClaimSearchContent); + } else { + fetchAndResolveChannelList(); + } + + fetchClaimSearchContent(resetClaimSearchContent); + resetClaimSearchContent = false; + if (showSubscribed && subscriptionsList.size() > 0) { + showSubscribedContent(); + } + } + + 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() { + Context context = getContext(); + boolean canShowMatureContent = false; + if (context != null) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + canShowMatureContent = sp.getBoolean(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT, false); + } + + return Lbry.buildClaimSearchOptions( + Claim.TYPE_CHANNEL, + null, + canShowMatureContent ? null : new ArrayList<>(Predefined.MATURE_TAGS), + null, + excludeChannelIdsForDiscover, + Arrays.asList(Claim.ORDER_BY_EFFECTIVE_AMOUNT), + null, + currentSuggestedPage == 0 ? 1 : currentSuggestedPage, + SUGGESTED_PAGE_SIZE); + } + + private Map buildContentOptions() { + Context context = getContext(); + boolean canShowMatureContent = false; + if (context != null) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + canShowMatureContent = sp.getBoolean(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT, false); + } + + return Lbry.buildClaimSearchOptions( + Claim.TYPE_STREAM, + null, + canShowMatureContent ? null : new ArrayList<>(Predefined.MATURE_TAGS), + 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() { + subscriptionsShown = true; + 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 + } + } + } + + excludeChannelIdsForDiscover = channelIds != null ? new ArrayList<>(channelIds) : null; + } + + private void fetchAndResolveChannelList() { + buildChannelIdsAndUrls(); + if (channelIds.size() > 0) { + ResolveTask resolveSubscribedTask = new ResolveTask(channelUrls, Lbry.LBRY_TV_CONNECTION_STRING, channelListLoading, new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + updateChannelFilterListAdapter(claims, true); + 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, boolean reset) { + 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); + } + if (reset) { + channelFilterListAdapter.clearClaims(); + channelFilterListAdapter.setSelectedItem(null); + } + channelFilterListAdapter.addClaims(resolvedSubs); + } + + private void fetchClaimSearchContent() { + fetchClaimSearchContent(false); + } + + private void fetchClaimSearchContent(boolean reset) { + if (reset && contentListAdapter != null) { + contentListAdapter.clearItems(); + currentClaimSearchPage = 1; + } + + contentClaimSearchLoading = true; + Helper.setViewVisibility(noContentView, View.GONE); + Map claimSearchOptions = buildContentOptions(); + contentClaimSearchTask = new ClaimSearchTask(claimSearchOptions, Lbry.LBRY_TV_CONNECTION_STRING, getLoadingView(), new 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) { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (claim.getName().startsWith("@")) { + // channel claim + activity.openChannelClaim(claim); + } else { + activity.openFileClaim(claim); + } + } + } + }); + } else { + contentListAdapter.addItems(claims); + } + + if (contentList != null && contentList.getAdapter() == null) { + contentList.setAdapter(contentListAdapter); + } + + contentHasReachedEnd = hasReachedEnd; + contentClaimSearchLoading = false; + checkNoContent(false); + } + + @Override + public void onError(Exception error) { + contentClaimSearchLoading = false; + checkNoContent(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); + } + } + + private void fetchSuggestedChannels() { + if (suggestedClaimSearchLoading) { + return; + } + + suggestedClaimSearchLoading = true; + if (discoverDialog != null) { + discoverDialog.setLoading(true); + } + + Helper.setViewVisibility(noContentView, View.GONE); + suggestedChannelClaimSearchTask = new ClaimSearchTask( + buildSuggestedOptions(), + Lbry.LBRY_TV_CONNECTION_STRING, + suggestedChannelAdapter == null || suggestedChannelAdapter.getItemCount() == 0 ? bigContentLoading : contentLoading, + new ClaimSearchResultHandler() { + @Override + public void onSuccess(List claims, boolean hasReachedEnd) { + suggestedHasReachedEnd = hasReachedEnd; + suggestedClaimSearchLoading = false; + if (discoverDialog != null) { + discoverDialog.setLoading(false); + } + + if (suggestedChannelAdapter == null) { + suggestedChannelAdapter = new SuggestedChannelGridAdapter(claims, getContext()); + suggestedChannelAdapter.setListener(FollowingFragment.this); + if (suggestedChannelGrid != null) { + suggestedChannelGrid.setAdapter(suggestedChannelAdapter); + } + if (discoverDialog != null) { + discoverDialog.setAdapter(suggestedChannelAdapter); + } + } else { + suggestedChannelAdapter.addClaims(claims); + } + + if (discoverDialog == null || !discoverDialog.isVisible()) { + checkNoContent(true); + } + } + + @Override + public void onError(Exception error) { + suggestedClaimSearchLoading = false; + if (discoverDialog != null) { + discoverDialog.setLoading(false); + } + if (discoverDialog == null || !discoverDialog.isVisible()) { + checkNoContent(true); + } + } + }); + + suggestedChannelClaimSearchTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + // 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; + + fetchSuggestedChannels(); + showSuggestedChannels(); + } else { + Lbryio.subscriptions = subscriptions; + subscriptionsList = new ArrayList<>(subscriptions); + showSubscribedContent(); + fetchAndResolveChannelList(); + } + } + + public void onError(Exception exception) { + + } + + public void onChannelItemSelected(Claim claim) { + // subscribe + Subscription subscription = Subscription.fromClaim(claim); + String channelClaimId = claim.getClaimId(); + + ChannelSubscribeTask task = new ChannelSubscribeTask(getContext(), channelClaimId, subscription, false, new ChannelSubscribeTask.ChannelSubscribeHandler() { + @Override + public void onSuccess() { + Lbryio.addSubscription(subscription); + Lbryio.addCachedResolvedSubscription(claim); + resetClaimSearchContent = true; + fetchLoadedSubscriptions(subscriptionsShown); + + saveSharedUserState(); + } + + @Override + public void onError(Exception error) { } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + updateSuggestedDoneButtonText(); + } + public void onChannelItemDeselected(Claim claim) { + // unsubscribe + Subscription subscription = Subscription.fromClaim(claim); + String channelClaimId = claim.getClaimId(); + ChannelSubscribeTask task = new ChannelSubscribeTask(getContext(), channelClaimId, subscription, true, new ChannelSubscribeTask.ChannelSubscribeHandler() { + @Override + public void onSuccess() { + Lbryio.removeSubscription(subscription); + Lbryio.removeCachedResolvedSubscription(claim); + resetClaimSearchContent = true; + fetchLoadedSubscriptions(subscriptionsShown); + + saveSharedUserState(); + } + + @Override + public void onError(Exception error) { + + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + updateSuggestedDoneButtonText(); + } + public void onChannelSelectionCleared() { + + } + + private void checkNoContent(boolean suggested) { + RecyclerView.Adapter adapter = suggested ? suggestedChannelAdapter : contentListAdapter; + boolean noContent = adapter == null || adapter.getItemCount() == 0; + Helper.setViewVisibility(noContentView, noContent ? View.VISIBLE : View.GONE); + } + + private void saveSharedUserState() { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).saveSharedUserState(); + } + } + + public void onSharedPreferenceChanged(SharedPreferences sp, String key) { + if (key.equalsIgnoreCase(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT)) { + fetchClaimSearchContent(true); + } + } + + public void onDownloadAction(String downloadAction, String uri, String outpoint, String fileInfoJson, double progress) { + if ("abort".equals(downloadAction)) { + if (contentListAdapter != null) { + contentListAdapter.clearFileForClaimOrUrl(outpoint, uri); + } + return; + } + + try { + JSONObject fileInfo = new JSONObject(fileInfoJson); + LbryFile claimFile = LbryFile.fromJSONObject(fileInfo); + String claimId = claimFile.getClaimId(); + if (contentListAdapter != null) { + contentListAdapter.updateFileForClaimByIdOrUrl(claimFile, claimId, uri); + } + } catch (JSONException ex) { + // invalid file info for download + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/findcontent/SearchFragment.java b/app/src/main/java/io/lbry/browser/ui/findcontent/SearchFragment.java new file mode 100644 index 00000000..901d57fa --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/findcontent/SearchFragment.java @@ -0,0 +1,321 @@ +package io.lbry.browser.ui.findcontent; + +import android.content.Context; +import android.content.SharedPreferences; +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.preference.PreferenceManager; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import org.json.JSONException; +import org.json.JSONObject; + +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.ClaimListAdapter; +import io.lbry.browser.listener.DownloadActionListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.ClaimCacheKey; +import io.lbry.browser.model.LbryFile; +import io.lbry.browser.model.NavMenuItem; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ClaimSearchResultHandler; +import io.lbry.browser.tasks.LighthouseSearchTask; +import io.lbry.browser.tasks.claim.ResolveTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.ui.publish.PublishFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; +import lombok.Setter; + +public class SearchFragment extends BaseFragment implements + ClaimListAdapter.ClaimListItemListener, DownloadActionListener, SharedPreferences.OnSharedPreferenceChangeListener { + private static final int PAGE_SIZE = 25; + + private ClaimListAdapter resultListAdapter; + 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(); + Context context = getContext(); + Helper.setWunderbarValue(currentQuery, context); + PreferenceManager.getDefaultSharedPreferences(context).registerOnSharedPreferenceChangeListener(this); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Search", "Search"); + activity.addDownloadActionListener(this); + } + if (!Helper.isNullOrEmpty(currentQuery)) { + logSearch(currentQuery); + search(currentQuery, currentFrom); + } else { + noQueryView.setVisibility(View.VISIBLE); + noResultsView.setVisibility(View.GONE); + } + } + + public void onPause() { + Context context = getContext(); + if (context != null) { + ((MainActivity) context).removeDownloadActionListener(this); + } + PreferenceManager.getDefaultSharedPreferences(context).unregisterOnSharedPreferenceChangeListener(this); + super.onPause(); + } + + 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); + claim.setConfirmations(1); + 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.setUrl(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 ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + if (claims.size() > 0 && !Helper.isNullOrEmpty(claims.get(0).getClaimId())) { + 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(); + + Context context = getContext(); + boolean canShowMatureContent = false; + if (context != null) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + canShowMatureContent = sp.getBoolean(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT, false); + } + if (resolved.isMature() && !canShowMatureContent) { + resultListAdapter.removeFeaturedItem(); + } else { + // 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); + unresolved.setConfirmations(resolved.getConfirmations()); + } + + resultListAdapter.notifyDataSetChanged(); + } + } + + private void logSearch(String query) { + Bundle bundle = new Bundle(); + bundle.putString("query", query); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_SEARCH, bundle); + } + + public void search(String query, int from) { + boolean queryChanged = checkQuery(query); + if (!queryChanged && from > 0) { + currentFrom = from; + } + + if (queryChanged) { + logSearch(query); + } + + searchLoading = true; + Context context = getContext(); + boolean canShowMatureContent = false; + if (context != null) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + canShowMatureContent = sp.getBoolean(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT, false); + } + + LighthouseSearchTask task = new LighthouseSearchTask( + currentQuery, PAGE_SIZE, currentFrom, canShowMatureContent, null, loadingView, new 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; + } + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (claim.isUnresolved()) { + // open the publish page + Map params = new HashMap<>(); + params.put("suggestedUrl", claim.getName()); + activity.openFragment(PublishFragment.class, true, NavMenuItem.ID_ITEM_NEW_PUBLISH, params); + } else if (claim.getName().startsWith("@")) { + activity.openChannelUrl(claim.getPermanentUrl()); + } else { + // not a channel + activity.openFileClaim(claim); + } + } + } + + public void onSharedPreferenceChanged(SharedPreferences sp, String key) { + if (key.equalsIgnoreCase(MainActivity.PREFERENCE_KEY_SHOW_MATURE_CONTENT)) { + search(currentQuery, currentFrom); + } + } + + public void onDownloadAction(String downloadAction, String uri, String outpoint, String fileInfoJson, double progress) { + if ("abort".equals(downloadAction)) { + if (resultListAdapter != null) { + resultListAdapter.clearFileForClaimOrUrl(outpoint, uri); + } + return; + } + + try { + JSONObject fileInfo = new JSONObject(fileInfoJson); + LbryFile claimFile = LbryFile.fromJSONObject(fileInfo); + String claimId = claimFile.getClaimId(); + if (resultListAdapter != null) { + resultListAdapter.updateFileForClaimByIdOrUrl(claimFile, claimId, uri); + } + } catch (JSONException ex) { + // invalid file info for download + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/library/LibraryFragment.java b/app/src/main/java/io/lbry/browser/ui/library/LibraryFragment.java new file mode 100644 index 00000000..49377b2a --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/library/LibraryFragment.java @@ -0,0 +1,741 @@ +package io.lbry.browser.ui.library; + +import android.content.Context; +import android.content.DialogInterface; +import android.graphics.Typeface; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.Menu; +import android.view.MenuItem; +import android.view.View; +import android.view.ViewGroup; +import android.widget.LinearLayout; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.appcompat.app.AlertDialog; +import androidx.appcompat.view.ActionMode; +import androidx.cardview.widget.CardView; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import com.google.android.material.snackbar.Snackbar; + +import org.json.JSONException; +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Date; +import java.util.List; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.ClaimListAdapter; +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.listener.DownloadActionListener; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.listener.SelectionModeListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.LbryFile; +import io.lbry.browser.model.ViewHistory; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ClaimListTask; +import io.lbry.browser.tasks.claim.ClaimSearchResultHandler; +import io.lbry.browser.tasks.claim.PurchaseListTask; +import io.lbry.browser.tasks.claim.ResolveTask; +import io.lbry.browser.tasks.file.BulkDeleteFilesTask; +import io.lbry.browser.tasks.file.FileListTask; +import io.lbry.browser.tasks.localdata.FetchViewHistoryTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; + +public class LibraryFragment extends BaseFragment implements + ActionMode.Callback, DownloadActionListener, SelectionModeListener, SdkStatusListener { + + private static final int FILTER_DOWNLOADS = 1; + private static final int FILTER_PURCHASES = 2; + private static final int FILTER_HISTORY = 3; + private static final int PAGE_SIZE = 50; + + private ActionMode actionMode; + private int currentFilter; + private List currentFiles; + private View layoutSdkInitializing; + private RecyclerView contentList; + private ClaimListAdapter contentListAdapter; + private ProgressBar listLoading; + private TextView linkFilterDownloads; + private TextView linkFilterPurchases; + private TextView linkFilterHistory; + private View layoutListEmpty; + private TextView textListEmpty; + private int currentPage; + private Date lastDate; + private boolean listReachedEnd; + private boolean contentListLoading; + private boolean initialOwnClaimsFetched; + + private CardView cardStats; + private TextView linkStats; + private TextView linkHide; + private View viewStatsDistribution; + private View viewVideoStatsBar; + private View viewAudioStatsBar; + private View viewImageStatsBar; + private View viewOtherStatsBar; + private TextView textStatsTotalSize; + private TextView textStatsTotalSizeUnits; + private TextView textStatsVideoSize; + private TextView textStatsAudioSize; + private TextView textStatsImageSize; + private TextView textStatsOtherSize; + private View legendVideo; + private View legendAudio; + private View legendImage; + private View legendOther; + + private long totalBytes; + private long totalVideoBytes; + private long totalAudioBytes; + private long totalImageBytes; + private long totalOtherBytes; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_library, container, false); + + layoutSdkInitializing = root.findViewById(R.id.container_library_sdk_initializing); + + LinearLayoutManager llm = new LinearLayoutManager(getContext()); + contentList = root.findViewById(R.id.library_list); + contentList.setLayoutManager(llm); + + listLoading = root.findViewById(R.id.library_list_loading); + linkFilterDownloads = root.findViewById(R.id.library_filter_link_downloads); + linkFilterPurchases = root.findViewById(R.id.library_filter_link_purchases); + linkFilterHistory = root.findViewById(R.id.library_filter_link_history); + + layoutListEmpty = root.findViewById(R.id.library_empty_container); + textListEmpty = root.findViewById(R.id.library_list_empty_text); + + currentFilter = FILTER_DOWNLOADS; + linkFilterDownloads.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + showDownloads(); + } + }); + linkFilterPurchases.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + showPurchases(); + } + }); + linkFilterHistory.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + showHistory(); + } + }); + contentList.addOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { + if (contentListLoading) { + 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 (!listReachedEnd) { + // load more + if (currentFilter == FILTER_DOWNLOADS) { + currentPage++; + fetchDownloads(); + } else if (currentFilter == FILTER_HISTORY) { + fetchHistory(); + } + } + } + } + } + }); + + // stats + linkStats = root.findViewById(R.id.library_show_stats); + linkHide = root.findViewById(R.id.library_hide_stats); + cardStats = root.findViewById(R.id.library_storage_stats_card); + viewStatsDistribution = root.findViewById(R.id.library_storage_stat_distribution); + viewVideoStatsBar = root.findViewById(R.id.library_storage_stat_video_bar); + viewAudioStatsBar = root.findViewById(R.id.library_storage_stat_audio_bar); + viewImageStatsBar = root.findViewById(R.id.library_storage_stat_image_bar); + viewOtherStatsBar = root.findViewById(R.id.library_storage_stat_other_bar); + textStatsTotalSize = root.findViewById(R.id.library_storage_stat_used); + textStatsTotalSizeUnits = root.findViewById(R.id.library_storage_stat_unit); + textStatsVideoSize = root.findViewById(R.id.library_storage_stat_video_size); + textStatsAudioSize = root.findViewById(R.id.library_storage_stat_audio_size); + textStatsImageSize = root.findViewById(R.id.library_storage_stat_image_size); + textStatsOtherSize = root.findViewById(R.id.library_storage_stat_other_size); + legendVideo = root.findViewById(R.id.library_storage_legend_video); + legendAudio = root.findViewById(R.id.library_storage_legend_audio); + legendImage = root.findViewById(R.id.library_storage_legend_image); + legendOther = root.findViewById(R.id.library_storage_legend_other); + + linkStats.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + updateStats(); + cardStats.setVisibility(View.VISIBLE); + checkStatsLink(); + } + }); + linkHide.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + cardStats.setVisibility(View.GONE); + checkStatsLink(); + } + }); + + return root; + } + + public void onResume() { + super.onResume(); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Library", "Library"); + activity.addDownloadActionListener(this); + } + + layoutSdkInitializing.setVisibility( + !Lbry.SDK_READY && currentFilter == FILTER_DOWNLOADS ? View.VISIBLE : View.GONE); + if (!Lbry.SDK_READY) { + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.addSdkStatusListener(this); + } + } else { + onSdkReady(); + } + } + + public void onPause() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.removeSdkStatusListener(this); + activity.removeDownloadActionListener(this); + } + super.onPause(); + } + + public void onSdkReady() { + layoutSdkInitializing.setVisibility(View.GONE); + if (currentFilter == FILTER_DOWNLOADS) { + showDownloads(); + } else if (currentFilter == FILTER_HISTORY) { + showHistory(); + } else if (currentFilter == FILTER_PURCHASES) { + showPurchases(); + } + } + + private void showDownloads() { + currentFilter = FILTER_DOWNLOADS; + linkFilterDownloads.setTypeface(null, Typeface.BOLD); + linkFilterPurchases.setTypeface(null, Typeface.NORMAL); + linkFilterHistory.setTypeface(null, Typeface.NORMAL); + if (contentListAdapter != null) { + contentListAdapter.setHideFee(false); + contentListAdapter.clearItems(); + contentListAdapter.setCanEnterSelectionMode(true); + } + listReachedEnd = false; + + checkStatsLink(); + layoutSdkInitializing.setVisibility(Lbry.SDK_READY ? View.GONE : View.VISIBLE); + currentPage = 1; + if (Lbry.SDK_READY) { + if (!initialOwnClaimsFetched) { + fetchOwnClaimsAndShowDownloads(); + } else { + fetchDownloads(); + } + } + } + + private void fetchOwnClaimsAndShowDownloads() { + if (Lbry.ownClaims != null && Lbry.ownClaims.size() > 0) { + initialOwnClaimsFetched = true; + fetchDownloads(); + return; + } + + linkStats.setVisibility(View.INVISIBLE); + ClaimListTask task = new ClaimListTask(Arrays.asList(Claim.TYPE_STREAM, Claim.TYPE_REPOST), listLoading, new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + Lbry.ownClaims = Helper.filterDeletedClaims(new ArrayList<>(claims)); + initialOwnClaimsFetched = true; + if (currentFilter == FILTER_DOWNLOADS) { + fetchDownloads(); + } + checkStatsLink(); + } + + @Override + public void onError(Exception error) { + initialOwnClaimsFetched = true; + checkStatsLink(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void showPurchases() { + currentFilter = FILTER_PURCHASES; + linkFilterDownloads.setTypeface(null, Typeface.NORMAL); + linkFilterPurchases.setTypeface(null, Typeface.BOLD); + linkFilterHistory.setTypeface(null, Typeface.NORMAL); + if (contentListAdapter != null) { + contentListAdapter.setHideFee(true); + contentListAdapter.clearItems(); + contentListAdapter.setCanEnterSelectionMode(true); + } + listReachedEnd = false; + + cardStats.setVisibility(View.GONE); + checkStatsLink(); + + layoutSdkInitializing.setVisibility(Lbry.SDK_READY ? View.GONE : View.VISIBLE); + currentPage = 1; + if (Lbry.SDK_READY) { + fetchPurchases(); + } + } + + private void showHistory() { + currentFilter = FILTER_HISTORY; + linkFilterDownloads.setTypeface(null, Typeface.NORMAL); + linkFilterPurchases.setTypeface(null, Typeface.NORMAL); + linkFilterHistory.setTypeface(null, Typeface.BOLD); + if (actionMode != null) { + actionMode.finish(); + } + if (contentListAdapter != null) { + contentListAdapter.setHideFee(false); + contentListAdapter.clearItems(); + contentListAdapter.setCanEnterSelectionMode(false); + } + listReachedEnd = false; + + cardStats.setVisibility(View.GONE); + checkStatsLink(); + + layoutSdkInitializing.setVisibility(View.GONE); + lastDate = null; + fetchHistory(); + } + + private void initContentListAdapter(List claims) { + contentListAdapter = new ClaimListAdapter(claims, getContext()); + contentListAdapter.setCanEnterSelectionMode(true); + contentListAdapter.setSelectionModeListener(this); + contentListAdapter.setHideFee(currentFilter != FILTER_PURCHASES); + contentListAdapter.setListener(new ClaimListAdapter.ClaimListItemListener() { + @Override + public void onClaimClicked(Claim claim) { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) getContext(); + if (claim.getName().startsWith("@")) { + activity.openChannelUrl(claim.getPermanentUrl()); + } else { + activity.openFileUrl(claim.getPermanentUrl()); + } + } + } + }); + } + + private void fetchDownloads() { + contentListLoading = true; + Helper.setViewVisibility(linkStats, View.GONE); + Helper.setViewVisibility(layoutListEmpty, View.GONE); + FileListTask task = new FileListTask(currentPage, PAGE_SIZE, true, listLoading, new FileListTask.FileListResultHandler() { + @Override + public void onSuccess(List files, boolean hasReachedEnd) { + listReachedEnd = hasReachedEnd; + List filteredFiles = Helper.filterDownloads(files); + List claims = Helper.claimsFromFiles(filteredFiles); + + addFiles(filteredFiles); + updateStats(); + checkStatsLink(); + + if (contentListAdapter == null) { + initContentListAdapter(claims); + } else { + contentListAdapter.addItems(claims); + } + if (contentList.getAdapter() == null) { + contentList.setAdapter(contentListAdapter); + } + resolveMissingChannelNames(buildUrlsToResolve(claims)); + checkListEmpty(); + contentListLoading = false; + } + + @Override + public void onError(Exception error) { + // pass + checkStatsLink(); + checkListEmpty(); + contentListLoading = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void fetchPurchases() { + contentListLoading = true; + Helper.setViewVisibility(linkStats, View.GONE); + Helper.setViewVisibility(layoutListEmpty, View.GONE); + PurchaseListTask task = new PurchaseListTask(currentPage, PAGE_SIZE, listLoading, new ClaimSearchResultHandler() { + @Override + public void onSuccess(List claims, boolean hasReachedEnd) { + listReachedEnd = hasReachedEnd; + if (contentListAdapter == null) { + initContentListAdapter(claims); + } else { + contentListAdapter.addItems(claims); + } + if (contentList.getAdapter() == null) { + contentList.setAdapter(contentListAdapter); + } + checkListEmpty(); + contentListLoading = false; + } + + @Override + public void onError(Exception error) { + checkStatsLink(); + checkListEmpty(); + contentListLoading = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void fetchHistory() { + contentListLoading = true; + Helper.setViewVisibility(layoutListEmpty, View.GONE); + DatabaseHelper dbHelper = DatabaseHelper.getInstance(); + if (dbHelper != null) { + FetchViewHistoryTask task = new FetchViewHistoryTask(lastDate, PAGE_SIZE, dbHelper, new FetchViewHistoryTask.FetchViewHistoryHandler() { + @Override + public void onSuccess(List history, boolean hasReachedEnd) { + listReachedEnd = hasReachedEnd; + if (history.size() > 0) { + lastDate = history.get(history.size() - 1).getTimestamp(); + } + + List claims = Helper.claimsFromViewHistory(history); + if (contentListAdapter == null) { + initContentListAdapter(claims); + } else { + contentListAdapter.addItems(claims); + } + if (contentList.getAdapter() == null) { + contentList.setAdapter(contentListAdapter); + } + checkListEmpty(); + contentListLoading = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } else { + checkListEmpty(); + contentListLoading = false; + } + } + + public void onDownloadAction(String downloadAction, String uri, String outpoint, String fileInfoJson, double progress) { + if ("abort".equals(downloadAction)) { + if (contentListAdapter != null) { + contentListAdapter.clearFileForClaimOrUrl(outpoint, uri, currentFilter == FILTER_DOWNLOADS); + } + return; + } + + try { + JSONObject fileInfo = new JSONObject(fileInfoJson); + LbryFile claimFile = LbryFile.fromJSONObject(fileInfo); + String claimId = claimFile.getClaimId(); + if (contentListAdapter != null) { + contentListAdapter.updateFileForClaimByIdOrUrl(claimFile, claimId, uri, true); + } + } catch (JSONException ex) { + // invalid file info for download + } + } + + private void checkListEmpty() { + layoutListEmpty.setVisibility(contentListAdapter == null || contentListAdapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + int stringResourceId; + switch (currentFilter) { + case FILTER_DOWNLOADS: default: stringResourceId = R.string.library_no_downloads; break; + case FILTER_HISTORY: stringResourceId = R.string.library_no_history; break; + case FILTER_PURCHASES: stringResourceId = R.string.library_no_purchases; break; + } + textListEmpty.setText(stringResourceId); + } + + private void addFiles(List files) { + if (currentFiles == null) { + currentFiles = new ArrayList<>(); + } + for (LbryFile file : files) { + if (!currentFiles.contains(file)) { + currentFiles.add(file); + } + } + } + + private void updateStats() { + totalBytes = 0; + totalVideoBytes = 0; + totalAudioBytes = 0; + totalImageBytes = 0; + totalOtherBytes = 0; + if (currentFiles != null) { + for (LbryFile file : currentFiles) { + long writtenBytes = file.getWrittenBytes(); + String mime = file.getMimeType(); + if (mime != null) { + if (mime.startsWith("video/")) { + totalVideoBytes += writtenBytes; + } else if (mime.startsWith("audio/")) { + totalAudioBytes += writtenBytes; + } else if (mime.startsWith("image/")) { + totalImageBytes += writtenBytes; + } else { + totalOtherBytes += writtenBytes; + } + } + + totalBytes += writtenBytes; + } + } + + renderStats(); + } + + private void renderStats() { + String[] totalSizeParts = Helper.formatBytesParts(totalBytes, false); + textStatsTotalSize.setText(totalSizeParts[0]); + textStatsTotalSizeUnits.setText(totalSizeParts[1]); + + viewStatsDistribution.setVisibility(totalBytes > 0 ? View.VISIBLE : View.GONE); + + int percentVideo = normalizePercent((double) totalVideoBytes / (double) totalBytes * 100.0); + legendVideo.setVisibility(totalVideoBytes > 0 ? View.VISIBLE : View.GONE); + textStatsVideoSize.setText(Helper.formatBytes(totalVideoBytes, false)); + applyLayoutWeight(viewVideoStatsBar, percentVideo); + + int percentAudio = normalizePercent((double) totalAudioBytes / (double) totalBytes * 100.0); + legendAudio.setVisibility(totalAudioBytes > 0 ? View.VISIBLE : View.GONE); + textStatsAudioSize.setText(Helper.formatBytes(totalAudioBytes, false)); + applyLayoutWeight(viewAudioStatsBar, percentAudio); + + int percentImage = normalizePercent((double) totalImageBytes / (double) totalBytes * 100.0); + legendImage.setVisibility(totalImageBytes > 0 ? View.VISIBLE : View.GONE); + textStatsImageSize.setText(Helper.formatBytes(totalImageBytes, false)); + applyLayoutWeight(viewImageStatsBar, percentImage); + + int percentOther = normalizePercent((double) totalOtherBytes / (double) totalBytes * 100.0); + legendOther.setVisibility(totalOtherBytes > 0 ? View.VISIBLE : View.GONE); + textStatsOtherSize.setText(Helper.formatBytes(totalOtherBytes, false)); + applyLayoutWeight(viewOtherStatsBar, percentOther); + + // We have to get to 100 (or adjust the container accordingly) + int totalPercent = percentVideo + percentAudio + percentImage + percentOther; + ((LinearLayout) viewStatsDistribution).setWeightSum(totalPercent); + } + + private void applyLayoutWeight(View view, int weight) { + LinearLayout.LayoutParams params = (LinearLayout.LayoutParams) view.getLayoutParams(); + params.weight = weight; + } + + private static int normalizePercent(double value) { + if (value > 0 && value < 1) { + return 1; + } + return Double.valueOf(Math.floor(value)).intValue(); + } + + private void checkStatsLink() { + linkStats.setVisibility(cardStats.getVisibility() == View.VISIBLE || + listLoading.getVisibility() == View.VISIBLE || + currentFilter != FILTER_DOWNLOADS || + !Lbry.SDK_READY ? + View.GONE : View.VISIBLE); + } + + @Override + public boolean onCreateActionMode(ActionMode actionMode, Menu menu) { + this.actionMode = actionMode; + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (!activity.isDarkMode()) { + activity.getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_VISIBLE); + } + } + + actionMode.getMenuInflater().inflate(R.menu.menu_claim_list, menu); + return true; + } + @Override + public void onDestroyActionMode(ActionMode actionMode) { + if (contentListAdapter != null) { + contentListAdapter.clearSelectedItems(); + contentListAdapter.setInSelectionMode(false); + contentListAdapter.notifyDataSetChanged(); + } + Context context = getContext(); + if (context != null) { + MainActivity activity = (MainActivity) context; + if (!activity.isDarkMode()) { + activity.getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR); + } + } + this.actionMode = null; + } + + @Override + public boolean onPrepareActionMode(androidx.appcompat.view.ActionMode actionMode, Menu menu) { + menu.findItem(R.id.action_edit).setVisible(false); + return true; + } + + @Override + public boolean onActionItemClicked(androidx.appcompat.view.ActionMode actionMode, MenuItem menuItem) { + if (R.id.action_delete == menuItem.getItemId()) { + if (contentListAdapter != null && contentListAdapter.getSelectedCount() > 0) { + final List selectedClaims = new ArrayList<>(contentListAdapter.getSelectedItems()); + String message = getResources().getQuantityString(R.plurals.confirm_delete_files, selectedClaims.size()); + AlertDialog.Builder builder = new AlertDialog.Builder(getContext()). + setTitle(R.string.delete_selection). + setMessage(message) + .setPositiveButton(R.string.yes, new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + handleDeleteSelectedClaims(selectedClaims); + } + }).setNegativeButton(R.string.no, null); + builder.show(); + return true; + } + } + + return false; + } + + private void handleDeleteSelectedClaims(List selectedClaims) { + List claimIds = new ArrayList<>(); + for (Claim claim : selectedClaims) { + claimIds.add(claim.getClaimId()); + } + + new BulkDeleteFilesTask(claimIds).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + Lbry.unsetFilesForCachedClaims(claimIds); + if (currentFilter == FILTER_DOWNLOADS) { + contentListAdapter.removeItems(selectedClaims); + } + if (actionMode != null) { + actionMode.finish(); + } + View root = getView(); + if (root != null) { + String message = getResources().getQuantityString(R.plurals.files_deleted, claimIds.size()); + Snackbar.make(root, message, Snackbar.LENGTH_LONG).show(); + } + } + + private List buildUrlsToResolve(List claims) { + List urls = new ArrayList<>(); + for (Claim claim : claims) { + Claim channel = claim.getSigningChannel(); + if (channel != null && Helper.isNullOrEmpty(channel.getName()) && !Helper.isNullOrEmpty(channel.getClaimId())) { + LbryUri uri = LbryUri.tryParse(String.format("%s#%s", claim.getName(), claim.getClaimId())); + if (uri != null) { + urls.add(uri.toString()); + } + } + } + return urls; + } + + private void resolveMissingChannelNames(List urls) { + if (urls.size() > 0) { + ResolveTask task = new ResolveTask(urls, Lbry.SDK_CONNECTION_STRING, null, new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + boolean updated = false; + for (Claim claim : claims) { + if (claim.getClaimId() == null) { + continue; + } + + if (contentListAdapter != null) { + contentListAdapter.updateSigningChannelForClaim(claim); + updated = true; + } + } + if (updated) { + contentListAdapter.notifyDataSetChanged(); + } + } + + @Override + public void onError(Exception error) { + + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + public void onEnterSelectionMode() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.startSupportActionMode(this); + } + } + public void onItemSelectionToggled() { + if (actionMode != null) { + actionMode.setTitle(String.valueOf(contentListAdapter.getSelectedCount())); + actionMode.invalidate(); + } + } + public void onExitSelectionMode() { + if (actionMode != null) { + actionMode.finish(); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/other/AboutFragment.java b/app/src/main/java/io/lbry/browser/ui/other/AboutFragment.java new file mode 100644 index 00000000..0923ad1b --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/other/AboutFragment.java @@ -0,0 +1,245 @@ +package io.lbry.browser.ui.other; + +import android.content.Context; +import android.content.Intent; +import android.content.pm.PackageInfo; +import android.content.pm.PackageManager; +import android.graphics.Color; +import android.graphics.Typeface; +import android.net.Uri; +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.appcompat.app.ActionBar; +import androidx.core.content.FileProvider; +import androidx.preference.PreferenceManager; + +import com.google.android.gms.tasks.OnCompleteListener; +import com.google.android.gms.tasks.Task; +import com.google.android.material.snackbar.Snackbar; +import com.google.firebase.iid.FirebaseInstanceId; +import com.google.firebase.iid.InstanceIdResult; + +import org.json.JSONObject; + +import java.io.File; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.exceptions.ApiCallException; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.Lbryio; +import io.lbry.lbrysdk.Utils; + +public class AboutFragment extends BaseFragment implements SdkStatusListener { + + private static final String FILE_PROVIDER = "io.lbry.browser.fileprovider"; + + private TextView textLinkWhatIsLBRY; + private TextView textLinkAndroidBasics; + private TextView textLinkFAQ; + private TextView textLinkDiscord; + private TextView textLinkFacebook; + private TextView textLinkInstagram; + private TextView textLinkReddit; + private TextView textLinkTelegram; + private TextView textLinkTwitter; + + private TextView textConnectedEmail; + private TextView textAppVersion; + private TextView textLbrySdkVersion; + private TextView textPlatform; + private TextView textInstallationId; + private TextView textFirebaseToken; + private View linkSendLog; + private View linkUpdateMailingPreferences; + + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_about, container, false); + + textLinkWhatIsLBRY = root.findViewById(R.id.about_link_what_is_lbry); + textLinkAndroidBasics = root.findViewById(R.id.about_link_android_basics); + textLinkFAQ = root.findViewById(R.id.about_link_faq); + textLinkDiscord = root.findViewById(R.id.about_link_discord); + textLinkFacebook = root.findViewById(R.id.about_link_facebook); + textLinkInstagram = root.findViewById(R.id.about_link_instagram); + textLinkReddit = root.findViewById(R.id.about_link_reddit); + textLinkTelegram = root.findViewById(R.id.about_link_telegram); + textLinkTwitter = root.findViewById(R.id.about_link_twitter); + + TextView[] textLinks = { + textLinkWhatIsLBRY, textLinkAndroidBasics, textLinkFAQ, textLinkDiscord, textLinkFacebook, + textLinkInstagram, textLinkReddit, textLinkTelegram, textLinkTwitter + }; + for (TextView view : textLinks) { + Helper.applyHtmlForTextView(view); + } + + textConnectedEmail = root.findViewById(R.id.about_connected_email); + textAppVersion = root.findViewById(R.id.about_app_version); + textLbrySdkVersion = root.findViewById(R.id.about_lbry_sdk); + textPlatform = root.findViewById(R.id.about_platform); + textInstallationId = root.findViewById(R.id.about_installation_id); + textFirebaseToken = root.findViewById(R.id.about_firebase_token); + linkSendLog = root.findViewById(R.id.about_send_log); + linkUpdateMailingPreferences = root.findViewById(R.id.about_update_mailing_preferences); + + if (Lbryio.isSignedIn()) { + textConnectedEmail.setText(Lbryio.getSignedInEmail()); + textConnectedEmail.setTypeface(null, Typeface.NORMAL); + linkUpdateMailingPreferences.setVisibility(View.VISIBLE); + } else { + linkUpdateMailingPreferences.setVisibility(View.GONE); + } + + Context context = getContext(); + String appVersion = getString(R.string.unknown); + if (context != null) { + try { + PackageManager manager = context.getPackageManager(); + PackageInfo info = manager.getPackageInfo(context.getPackageName(), PackageManager.GET_ACTIVITIES); + appVersion = info.versionName; + } catch (PackageManager.NameNotFoundException ex) { + // pass + } + } + textAppVersion.setText(appVersion); + textInstallationId.setText(Lbry.INSTALLATION_ID); + textPlatform.setText(String.format("Android %s (API %d)", Utils.getAndroidRelease(), Utils.getAndroidSdk())); + + linkUpdateMailingPreferences.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(String.format("http://lbry.com/list/edit/%s", Lbryio.AUTH_TOKEN))); + startActivity(intent); + } + }); + + linkSendLog.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + shareLogFile(); + } + }); + + return root; + } + + private void shareLogFile() { + Context context = getContext(); + if (context != null) { + String logFileName = "lbrynet.log"; + File logFile = new File(String.format("%s/%s", Utils.getAppInternalStorageDir(context), "lbrynet"), logFileName); + if (!logFile.exists()) { + Snackbar.make(getView(), R.string.cannot_find_lbrynet_log, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED). + setTextColor(Color.WHITE). + show(); + return; + } + + try { + Uri fileUri = FileProvider.getUriForFile(getContext(), FILE_PROVIDER, logFile); + if (fileUri != null) { + MainActivity.startingShareActivity = true; + 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); + startActivity(sendLogIntent); + } + } catch (IllegalArgumentException e) { + Snackbar.make(getView(), R.string.cannot_share_lbrynet_log, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED). + setTextColor(Color.WHITE). + show(); + } + } + } + + @Override + public void onStart() { + super.onStart(); + MainActivity activity = (MainActivity) getContext(); + if (activity != null) { + activity.hideSearchBar(); + activity.showNavigationBackIcon(); + activity.lockDrawer(); + activity.hideFloatingWalletBalance(); + + ActionBar actionBar = activity.getSupportActionBar(); + if (actionBar != null) { + actionBar.setTitle(R.string.about_lbry); + } + } + } + + @Override + public void onResume() { + super.onResume(); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "About", "About"); + + if (!Lbry.SDK_READY) { + activity.addSdkStatusListener(this); + } else { + onSdkReady(); + } + } + FirebaseInstanceId.getInstance().getInstanceId().addOnCompleteListener(new OnCompleteListener() { + @Override + public void onComplete(Task task) { + Helper.setViewText(textFirebaseToken, task.isSuccessful() ? task.getResult().getToken() : getString(R.string.unknown)); + } + }); + + } + + @Override + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) getContext(); + activity.removeSdkStatusListener(this); + activity.restoreToggle(); + activity.showFloatingWalletBalance(); + } + super.onStop(); + } + + public void onSdkReady() { + loadLbryVersion(); + } + + private void loadLbryVersion() { + (new AsyncTask() { + protected String doInBackground(Void... params) { + try { + JSONObject result = (JSONObject) Lbry.genericApiCall(Lbry.METHOD_VERSION); + return Helper.getJSONString("lbrynet_version", null, result); + } catch (ApiCallException | ClassCastException ex) { + // pass + return null; + } + } + protected void onPostExecute(String version) { + Helper.setViewText(textLbrySdkVersion, Helper.isNullOrEmpty(version) ? getString(R.string.unknown) : version); + } + }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/other/SettingsFragment.java b/app/src/main/java/io/lbry/browser/ui/other/SettingsFragment.java new file mode 100644 index 00000000..6cf4deaf --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/other/SettingsFragment.java @@ -0,0 +1,103 @@ +package io.lbry.browser.ui.other; + +import android.content.Context; +import android.content.SharedPreferences; +import android.os.AsyncTask; +import android.os.Bundle; + +import androidx.appcompat.app.ActionBar; +import androidx.appcompat.app.AppCompatDelegate; +import androidx.preference.PreferenceFragmentCompat; +import androidx.preference.PreferenceManager; + +import java.io.FileOutputStream; +import java.io.PrintStream; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.lbrysdk.Utils; + +public class SettingsFragment extends PreferenceFragmentCompat implements SharedPreferences.OnSharedPreferenceChangeListener { + @Override + public void onCreatePreferences(Bundle savedInstanceState, String rootKey) { + setPreferencesFromResource(R.xml.settings, rootKey); + } + + @Override + public void onStart() { + super.onStart(); + MainActivity activity = (MainActivity) getContext(); + if (activity != null) { + activity.hideSearchBar(); + activity.showNavigationBackIcon(); + activity.lockDrawer(); + activity.hideFloatingWalletBalance(); + + ActionBar actionBar = activity.getSupportActionBar(); + if (actionBar != null) { + actionBar.setTitle(R.string.settings); + } + } + } + + @Override + public void onResume() { + super.onResume(); + Context context = getContext(); + if (context instanceof MainActivity) { + PreferenceManager.getDefaultSharedPreferences(context).registerOnSharedPreferenceChangeListener(this); + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Settings", "Settings"); + } + } + @Override + public void onPause() { + Context context = getContext(); + if (context != null) { + PreferenceManager.getDefaultSharedPreferences(context).unregisterOnSharedPreferenceChangeListener(this); + } + super.onPause(); + } + @Override + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) getContext(); + activity.restoreToggle(); + activity.showFloatingWalletBalance(); + } + 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); + } else if (key.equalsIgnoreCase(MainActivity.PREFERENCE_KEY_PARTICIPATE_DATA_NETWORK)) { + boolean dhtEnabled = sp.getBoolean(MainActivity.PREFERENCE_KEY_PARTICIPATE_DATA_NETWORK, false); + updateDHTFileSetting(dhtEnabled); + } + } + + private void updateDHTFileSetting(final boolean enabled) { + Context context = getContext(); + (new AsyncTask() { + protected Void doInBackground(Void... params) { + PrintStream out = null; + try { + String fileContent = enabled ? "on" : "off"; + String path = String.format("%s/%s", Utils.getAppInternalStorageDir(context), "dht"); + out = new PrintStream(new FileOutputStream(path)); + out.print(fileContent); + } catch (Exception ex) { + // pass + } finally { + Helper.closeCloseable(out); + } + return null; + } + }).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/publish/PublishFormFragment.java b/app/src/main/java/io/lbry/browser/ui/publish/PublishFormFragment.java new file mode 100644 index 00000000..7e5614ac --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/publish/PublishFormFragment.java @@ -0,0 +1,1644 @@ +package io.lbry.browser.ui.publish; + +import android.Manifest; +import android.content.Context; +import android.content.Intent; +import android.graphics.Bitmap; +import android.graphics.BitmapFactory; +import android.graphics.Color; +import android.media.ThumbnailUtils; +import android.net.Uri; +import android.os.AsyncTask; +import android.os.Bundle; +import android.os.Handler; +import android.provider.MediaStore; +import android.text.Editable; +import android.text.TextWatcher; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.webkit.MimeTypeMap; +import android.widget.AdapterView; +import android.widget.CompoundButton; +import android.widget.ImageView; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.appcompat.app.ActionBar; +import androidx.appcompat.widget.AppCompatSpinner; +import androidx.cardview.widget.CardView; +import androidx.core.content.ContextCompat; +import androidx.core.widget.NestedScrollView; +import androidx.recyclerview.widget.RecyclerView; + +import com.arthenica.mobileffmpeg.Config; +import com.arthenica.mobileffmpeg.FFmpeg; +import com.arthenica.mobileffmpeg.FFprobe; + +import com.arthenica.mobileffmpeg.Statistics; +import com.arthenica.mobileffmpeg.StatisticsCallback; +import com.bumptech.glide.Glide; +import com.google.android.flexbox.FlexboxLayoutManager; +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 com.google.android.material.textfield.TextInputLayout; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.io.File; +import java.io.FileOutputStream; +import java.math.BigDecimal; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.BuildConfig; +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.InlineChannelSpinnerAdapter; +import io.lbry.browser.adapter.LanguageSpinnerAdapter; +import io.lbry.browser.adapter.LicenseSpinnerAdapter; +import io.lbry.browser.adapter.TagListAdapter; +import io.lbry.browser.listener.FilePickerListener; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.listener.StoragePermissionListener; +import io.lbry.browser.listener.WalletBalanceListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.Fee; +import io.lbry.browser.model.GalleryItem; +import io.lbry.browser.model.Language; +import io.lbry.browser.model.License; +import io.lbry.browser.model.NavMenuItem; +import io.lbry.browser.model.Tag; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.tasks.claim.ChannelCreateUpdateTask; +import io.lbry.browser.tasks.UpdateSuggestedTagsTask; +import io.lbry.browser.tasks.UploadImageTask; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ClaimListTask; +import io.lbry.browser.tasks.claim.ClaimResultHandler; +import io.lbry.browser.tasks.claim.PublishClaimTask; +import io.lbry.browser.tasks.lbryinc.LogPublishTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Predefined; +import io.lbry.lbrysdk.Utils; +import lombok.Data; +import lombok.Getter; + +public class PublishFormFragment extends BaseFragment implements + FilePickerListener, SdkStatusListener, StoragePermissionListener, TagListAdapter.TagClickListener, WalletBalanceListener { + + private static final String H264_CODEC = "h264"; + private static final int MAX_VIDEO_DIMENSION = 1920; + private static final int MAX_BITRATE = 5000000; // 5mbps + + private static final int SUGGESTED_LIMIT = 8; + + private boolean editMode; + @Getter + private boolean saveInProgress; + private String currentFilter; + private boolean publishFileChecked; + private boolean fetchingChannels; + private boolean launchPickerPending; + @Getter + private boolean transcodeInProgress; + private long transcodeStartTime; + private VideoTranscodeTask videoTranscodeTask; + + private TextInputEditText inputTagFilter; + private RecyclerView addedTagsList; + private RecyclerView suggestedTagsList; + private RecyclerView matureTagsList; + private TagListAdapter addedTagsAdapter; + private TagListAdapter suggestedTagsAdapter; + private TagListAdapter matureTagsAdapter; + private ProgressBar progressPublish; + private ProgressBar progressLoadingChannels; + private View noTagsView; + private View noTagResultsView; + + private InlineChannelSpinnerAdapter channelSpinnerAdapter; + private AppCompatSpinner channelSpinner; + private AppCompatSpinner priceCurrencySpinner; + private AppCompatSpinner languageSpinner; + private AppCompatSpinner licenseSpinner; + + private NestedScrollView scrollView; + private View layoutExtraFields; + private TextView linkShowExtraFields; + private View textNoPrice; + private View layoutPrice; + private SwitchMaterial switchPrice; + private ImageView imageThumbnail; + private TextView linkGenerateAddress; + + private TextInputEditText inputTitle; + private TextInputEditText inputDescription; + private TextInputEditText inputPrice; + private TextInputEditText inputAddress; + private TextInputEditText inputDeposit; + private TextInputEditText inputOtherLicenseDescription; + private TextInputLayout layoutOtherLicenseDescription; + private View inlineDepositBalanceContainer; + private TextView inlineDepositBalanceValue; + private TextView textInlineAddressInvalid; + + private View linkPublishCancel; + private MaterialButton buttonPublish; + + private View inlineChannelCreator; + private TextInputEditText inlineChannelCreatorInputName; + private TextInputEditText inlineChannelCreatorInputDeposit; + private View inlineChannelCreatorInlineBalance; + private TextView inlineChannelCreatorInlineBalanceValue; + private View inlineChannelCreatorCancelLink; + private View inlineChannelCreatorProgress; + private MaterialButton inlineChannelCreatorCreateButton; + + private boolean uploading; + private String lastSelectedThumbnailFile; + private String uploadedThumbnailUrl; + private boolean editFieldsLoaded; + private boolean editChannelSpinnerLoaded; + private Claim currentClaim; + private GalleryItem currentGalleryItem; + private String currentFilePath; + private String transcodedFilePath; + + private View mediaContainer; + private View uploadProgress; + private CardView cardVideoOptimization; + private ProgressBar optimizationRealProgress; + private ProgressBar optimizationProgress; + private TextView textOptimizationProgress; + private TextView textOptimizationStatus; + private TextView textOptimizationElapsed; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_publish_form, container, false); + + scrollView = root.findViewById(R.id.publish_form_scroll_view); + progressLoadingChannels = root.findViewById(R.id.publish_form_loading_channels); + progressPublish = root.findViewById(R.id.publish_form_publishing); + channelSpinner = root.findViewById(R.id.publish_form_channel_spinner); + mediaContainer = root.findViewById(R.id.publish_form_media_container); + + inputTagFilter = root.findViewById(R.id.form_tag_filter_input); + noTagsView = root.findViewById(R.id.form_no_added_tags); + noTagResultsView = root.findViewById(R.id.form_no_tag_results); + + textInlineAddressInvalid = root.findViewById(R.id.publish_form_inline_address_invalid); + inlineDepositBalanceContainer = root.findViewById(R.id.publish_form_inline_balance_container); + inlineDepositBalanceValue = root.findViewById(R.id.publish_form_inline_balance_value); + + cardVideoOptimization = root.findViewById(R.id.publish_form_video_opt_card); + optimizationProgress = root.findViewById(R.id.publish_form_video_opt_progress); + optimizationRealProgress = root.findViewById(R.id.publish_form_video_opt_real_progress); + textOptimizationProgress = root.findViewById(R.id.publish_form_video_opt_progress_text); + textOptimizationStatus = root.findViewById(R.id.publish_form_video_opt_status); + textOptimizationElapsed = root.findViewById(R.id.publish_form_video_opt_elapsed); + + layoutExtraFields = root.findViewById(R.id.publish_form_extra_options_container); + linkShowExtraFields = root.findViewById(R.id.publish_form_toggle_extra); + layoutPrice = root.findViewById(R.id.publish_form_price_container); + textNoPrice = root.findViewById(R.id.publish_form_no_price); + switchPrice = root.findViewById(R.id.publish_form_price_switch); + uploadProgress = root.findViewById(R.id.publish_form_thumbnail_upload_progress); + imageThumbnail = root.findViewById(R.id.publish_form_thumbnail_preview); + linkGenerateAddress = root.findViewById(R.id.publish_form_generate_address); + + inputTitle = root.findViewById(R.id.publish_form_input_title); + inputDescription = root.findViewById(R.id.publish_form_input_description); + inputPrice = root.findViewById(R.id.publish_form_input_price); + inputAddress = root.findViewById(R.id.publish_form_input_address); + inputDeposit = root.findViewById(R.id.publish_form_input_deposit); + inputOtherLicenseDescription = root.findViewById(R.id.publish_form_input_license_other); + layoutOtherLicenseDescription = root.findViewById(R.id.publish_form_license_other_layout); + priceCurrencySpinner = root.findViewById(R.id.publish_form_currency_spinner); + languageSpinner = root.findViewById(R.id.publish_form_language_spinner); + licenseSpinner = root.findViewById(R.id.publish_form_license_spinner); + + linkPublishCancel = root.findViewById(R.id.publish_form_cancel); + buttonPublish = root.findViewById(R.id.publish_form_publish_button); + + Context context = getContext(); + FlexboxLayoutManager flm1 = new FlexboxLayoutManager(context); + FlexboxLayoutManager flm2 = new FlexboxLayoutManager(context); + FlexboxLayoutManager flm3 = new FlexboxLayoutManager(context); + addedTagsList = root.findViewById(R.id.form_added_tags); + addedTagsList.setLayoutManager(flm1); + suggestedTagsList = root.findViewById(R.id.form_suggested_tags); + suggestedTagsList.setLayoutManager(flm2); + + root.findViewById(R.id.form_mature_tags_container).setVisibility(View.VISIBLE); + matureTagsList = root.findViewById(R.id.form_mature_tags); + matureTagsList.setLayoutManager(flm3); + + addedTagsAdapter = new TagListAdapter(new ArrayList<>(), context); + addedTagsAdapter.setCustomizeMode(TagListAdapter.CUSTOMIZE_MODE_REMOVE); + addedTagsAdapter.setClickListener(this); + addedTagsList.setAdapter(addedTagsAdapter); + + suggestedTagsAdapter = new TagListAdapter(new ArrayList<>(), getContext()); + suggestedTagsAdapter.setCustomizeMode(TagListAdapter.CUSTOMIZE_MODE_ADD); + suggestedTagsAdapter.setClickListener(this); + suggestedTagsList.setAdapter(suggestedTagsAdapter); + + matureTagsAdapter = new TagListAdapter(Helper.getTagObjectsForTags(Predefined.MATURE_TAGS), context); + matureTagsAdapter.setCustomizeMode(TagListAdapter.CUSTOMIZE_MODE_ADD); + matureTagsAdapter.setClickListener(this); + matureTagsList.setAdapter(matureTagsAdapter); + + inlineChannelCreator = root.findViewById(R.id.container_inline_channel_form_create); + inlineChannelCreatorInputName = root.findViewById(R.id.inline_channel_form_input_name); + inlineChannelCreatorInputDeposit = root.findViewById(R.id.inline_channel_form_input_deposit); + inlineChannelCreatorInlineBalance = root.findViewById(R.id.inline_channel_form_inline_balance_container); + inlineChannelCreatorInlineBalanceValue = root.findViewById(R.id.inline_channel_form_inline_balance_value); + inlineChannelCreatorProgress = root.findViewById(R.id.inline_channel_form_create_progress); + inlineChannelCreatorCancelLink = root.findViewById(R.id.inline_channel_form_cancel_link); + inlineChannelCreatorCreateButton = root.findViewById(R.id.inline_channel_form_create_button); + + initUi(); + + return root; + } + + private void initUi() { + Context context = getContext(); + languageSpinner.setAdapter(new LanguageSpinnerAdapter(context, R.layout.spinner_item_generic)); + licenseSpinner.setAdapter(new LicenseSpinnerAdapter(context, R.layout.spinner_item_generic)); + + licenseSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() { + @Override + public void onItemSelected(AdapterView adapterView, View view, int position, long l) { + License license = (License) adapterView.getAdapter().getItem(position); + boolean otherLicense = Arrays.asList( + Predefined.LICENSE_COPYRIGHTED.toLowerCase(), + Predefined.LICENSE_OTHER.toLowerCase()).contains(license.getName().toLowerCase()); + Helper.setViewVisibility(layoutOtherLicenseDescription, otherLicense ? View.VISIBLE : View.GONE); + if (!otherLicense) { + inputOtherLicenseDescription.setText(null); + } + } + + @Override + public void onNothingSelected(AdapterView adapterView) { + + } + }); + + linkGenerateAddress.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (!editMode) { + inputAddress.setText(Helper.generateUrl()); + } + } + }); + + switchPrice.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { + @Override + public void onCheckedChanged(CompoundButton compoundButton, boolean checked) { + Helper.setViewVisibility(textNoPrice, checked ? View.GONE : View.VISIBLE); + Helper.setViewVisibility(layoutPrice, checked ? View.VISIBLE : View.GONE); + } + }); + + inputAddress.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) { + String value = Helper.getValue(charSequence); + boolean invalid = !Helper.isNullOrEmpty(value) && !LbryUri.isNameValid(value); + Helper.setViewVisibility(textInlineAddressInvalid, invalid ? View.VISIBLE : View.INVISIBLE); + } + + @Override + public void afterTextChanged(Editable editable) { + + } + }); + + inputDeposit.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + Helper.setViewVisibility(inlineDepositBalanceContainer, hasFocus ? View.VISIBLE : View.GONE); + } + }); + + linkShowExtraFields.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (layoutExtraFields.getVisibility() != View.VISIBLE) { + layoutExtraFields.setVisibility(View.VISIBLE); + linkShowExtraFields.setText(R.string.hide_extra_fields); + scrollView.post(new Runnable() { + @Override + public void run() { + scrollView.fullScroll(NestedScrollView.FOCUS_DOWN); + } + }); + } else { + layoutExtraFields.setVisibility(View.GONE); + linkShowExtraFields.setText(R.string.show_extra_fields); + } + } + }); + + mediaContainer.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + checkStoragePermissionAndLaunchFilePicker(); + } + }); + + channelSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() { + @Override + public void onItemSelected(AdapterView adapterView, View view, int position, long l) { + Object item = adapterView.getItemAtPosition(position); + if (item instanceof Claim) { + Claim claim = (Claim) item; + if (claim.isPlaceholder() && !claim.isPlaceholderAnonymous()) { + if (!fetchingChannels) { + showInlineChannelCreator(); + } + } else { + hideInlineChannelCreator(); + } + } + } + + @Override + public void onNothingSelected(AdapterView adapterView) { + + } + }); + + inputTagFilter.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) { + String value = Helper.getValue(charSequence); + setFilter(value); + } + + @Override + public void afterTextChanged(Editable editable) { + + } + }); + + linkPublishCancel.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (transcodeInProgress) { + // show alert confirming the user is sure, and then cancel + FFmpeg.cancel(); + transcodeInProgress = false; + } + + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).onBackPressed(); + } + } + }); + + buttonPublish.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (uploading) { + Snackbar.make(view, R.string.publish_no_thumbnail, Snackbar.LENGTH_LONG).show(); + return; + } else if (Helper.isNullOrEmpty(uploadedThumbnailUrl)) { + showError(getString(R.string.publish_thumbnail_in_progress)); + return; + } + + if (transcodeInProgress) { + Snackbar.make(view, R.string.optimization_in_progress, Snackbar.LENGTH_LONG).show(); + return; + } + + // check minimum deposit + String depositString = Helper.getValue(inputDeposit.getText()); + double depositAmount = 0; + try { + depositAmount = Double.valueOf(depositString); + } catch (NumberFormatException ex) { + // pass + showError(getString(R.string.please_enter_valid_deposit)); + return; + } + if (depositAmount < Helper.MIN_DEPOSIT) { + String error = getResources().getQuantityString(R.plurals.min_deposit_required, depositAmount == 1 ? 1 : 2, String.valueOf(Helper.MIN_DEPOSIT)); + showError(error); + return; + } + if (Lbry.walletBalance == null || Lbry.walletBalance.getAvailable().doubleValue() < depositAmount) { + showError(getString(R.string.deposit_more_than_balance)); + return; + } + + String priceString = Helper.getValue(inputPrice.getText()); + double priceAmount = Helper.parseDouble(priceString, 0); + if (switchPrice.isChecked() && priceAmount == 0) { + showError(getString(R.string.price_amount_not_set)); + return; + } + + Claim claim = buildPublishClaim(); + if (validatePublishClaim(claim)) { + publishClaim(claim); + } + } + }); + + setupInlineChannelCreator( + inlineChannelCreator, + inlineChannelCreatorInputName, + inlineChannelCreatorInputDeposit, + inlineChannelCreatorInlineBalance, + inlineChannelCreatorInlineBalanceValue, + inlineChannelCreatorCancelLink, + inlineChannelCreatorCreateButton, + inlineChannelCreatorProgress + ); + } + + @Override + public void onStart() { + super.onStart(); + MainActivity activity = (MainActivity) getContext(); + if (activity != null) { + activity.hideSearchBar(); + activity.showNavigationBackIcon(); + activity.lockDrawer(); + activity.hideFloatingWalletBalance(); + + activity.addFilePickerListener(this); + activity.addWalletBalanceListener(this); + + ActionBar actionBar = activity.getSupportActionBar(); + if (actionBar != null) { + actionBar.setTitle(editMode ? R.string.edit_content : R.string.new_publish); + } + } + } + + @Override + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) getContext(); + activity.restoreToggle(); + activity.showFloatingWalletBalance(); + if (!MainActivity.startingFilePickerActivity) { + activity.removeWalletBalanceListener(this); + activity.removeFilePickerListener(this); + activity.removeNavFragment(PublishFormFragment.class, NavMenuItem.ID_ITEM_NEW_PUBLISH); + if (transcodeInProgress) { + FFmpeg.cancel(); + } + } + } + + super.onStop(); + } + + private void checkParams() { + Map params = getParams(); + if (params != null) { + if (params.containsKey("claim")) { + Claim claim = (Claim) params.get("claim"); + if (claim != null && !claim.equals(this.currentClaim)) { + this.currentClaim = claim; + editFieldsLoaded = false; + } + } else if (params.containsKey("galleryItem")) { + currentGalleryItem = (GalleryItem) params.get("galleryItem"); + } else if (params.containsKey("directFilePath")) { + currentFilePath = (String) params.get("directFilePath"); + } + + if (this.currentClaim == null && params.containsKey("suggestedUrl")) { + String suggestedUrl = (String) params.get("suggestedUrl"); + if (!Helper.isNullOrEmpty(suggestedUrl) && Helper.isNullOrEmpty(Helper.getValue(inputAddress.getText()))) { + Helper.setViewText(inputAddress, (String) params.get("suggestedUrl")); + } + } + } else { + // shouldn't actually happen + cancelOnFatalCondition(getString(R.string.no_file_found)); + } + } + + private void checkStoragePermissionAndLaunchFilePicker() { + Context context = getContext(); + if (MainActivity.hasPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, context)) { + launchPickerPending = false; + launchFilePicker(); + } else { + launchPickerPending = true; + MainActivity.requestPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, + MainActivity.REQUEST_STORAGE_PERMISSION, + getString(R.string.storage_permission_rationale_images), + context, + true); + } + } + + private void launchFilePicker() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity.startingFilePickerActivity = true; + Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT); + intent.setType("image/*"); + ((MainActivity) context).startActivityForResult( + Intent.createChooser(intent, getString(R.string.select_thumbnail)), + MainActivity.REQUEST_FILE_PICKER); + } + } + + private void updateFieldsFromCurrentClaim() { + if (currentClaim != null && !editFieldsLoaded) { + Context context = getContext(); + Claim.StreamMetadata metadata = (Claim.StreamMetadata) currentClaim.getValue(); + uploadedThumbnailUrl = currentClaim.getThumbnailUrl(); + if (context != null && !Helper.isNullOrEmpty(uploadedThumbnailUrl)) { + Glide.with(context.getApplicationContext()).load(uploadedThumbnailUrl).centerCrop().into(imageThumbnail); + } + + inputTitle.setText(currentClaim.getTitle()); + inputDescription.setText(currentClaim.getDescription()); + if (addedTagsAdapter != null && currentClaim.getTagObjects() != null) { + addedTagsAdapter.addTags(currentClaim.getTagObjects()); + updateSuggestedTags(currentFilter, SUGGESTED_LIMIT, true); + } + + if (metadata.getFee() != null) { + Fee fee = metadata.getFee(); + switchPrice.setChecked(true); + inputPrice.setText(fee.getAmount()); + priceCurrencySpinner.setSelection("lbc".equalsIgnoreCase(fee.getCurrency()) ? 0 : 1); + } + + inputAddress.setText(currentClaim.getName()); + inputDeposit.setText(currentClaim.getAmount()); + + if (metadata.getLanguages() != null && metadata.getLanguages().size() > 0) { + // get the first language + String langCode = metadata.getLanguages().get(0); + int langCodePosition = ((LanguageSpinnerAdapter) languageSpinner.getAdapter()).getItemPosition(langCode); + if (langCodePosition > -1) { + languageSpinner.setSelection(langCodePosition); + } + } + + if (!Helper.isNullOrEmpty(metadata.getLicense())) { + LicenseSpinnerAdapter adapter = (LicenseSpinnerAdapter) licenseSpinner.getAdapter(); + int licPosition = adapter.getItemPosition(metadata.getLicense()); + if (licPosition == -1) { + licPosition = adapter.getItemPosition(Predefined.LICENSE_OTHER); + } + if (licPosition > -1) { + licenseSpinner.setSelection(licPosition); + } + + License selectedLicense = (License) licenseSpinner.getSelectedItem(); + boolean otherLicense = Arrays.asList( + Predefined.LICENSE_COPYRIGHTED.toLowerCase(), + Predefined.LICENSE_OTHER.toLowerCase()).contains(selectedLicense.getName().toLowerCase()); + inputOtherLicenseDescription.setText(otherLicense ? metadata.getLicense() : null); + } + + inputAddress.setEnabled(false); + editMode = true; + editFieldsLoaded = true; + } + } + + private void checkPublishFile() { + if (publishFileChecked) { + return; + } + + String filePath = ""; + String thumbnailPath = null; + if (currentGalleryItem != null) { + // check gallery item type + filePath = currentGalleryItem.getFilePath(); + thumbnailPath = currentGalleryItem.getThumbnailPath(); + } else if (currentFilePath != null) { + filePath = currentFilePath; + } + + File file = new File(filePath); + if (!file.exists()) { + // file doesn't exist. although this shouldn't happen + cancelOnFatalCondition(getString(R.string.no_file_found)); + return; + } + + // check content type + String type = null; + String extension = MimeTypeMap.getFileExtensionFromUrl(Uri.fromFile(file).toString()); + if (extension != null) { + type = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension); + } + + boolean isVideo = false; + boolean isImage = !Helper.isNullOrEmpty(type) && type.startsWith("image"); + if (!Helper.isNullOrEmpty(type) && type.startsWith("video")) { + // ffmpeg video handling + isVideo = true; + if (!transcodeInProgress) { + probeVideo(filePath); + } + } + + if (isVideo || isImage) { + checkAndUploadThumbnail(filePath, thumbnailPath, isVideo ? "video" : "image"); + } + + Helper.setViewVisibility(cardVideoOptimization, isVideo ? View.VISIBLE : View.GONE); + + publishFileChecked = true; + } + + private void checkAndUploadThumbnail(String filePath, String thumbnailPath, String type) { + boolean thumbnailValid = false; + if (!Helper.isNullOrEmpty(thumbnailPath)) { + File file = new File(thumbnailPath); + // make sure the file exists and it's not an empty file + thumbnailValid = file.exists() && file.length() > 0; + } + + if (!thumbnailValid) { + createAndUploadThumbnail(filePath, type); + } else { + uploadThumbnail(thumbnailPath); + } + } + + private void createAndUploadThumbnail(String filePath, String type) { + Context context = getContext(); + CreateThumbnailTask task = new CreateThumbnailTask(filePath, type, context, new CreateThumbnailTask.CreateThumbnailHandler() { + @Override + public void onSuccess(String thumbnailPath) { + uploadThumbnail(thumbnailPath); + } + + @Override + public void onError(Exception error) { + if (context != null) { + showError(getString(R.string.thumbnail_creation_failed)); + } + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void uploadThumbnail(String thumbnailPath) { + if (uploading) { + Snackbar.make(getView(), R.string.wait_for_upload, Snackbar.LENGTH_LONG).show(); + return; + } + + Context context = getContext(); + if (context != null) { + Glide.with(context.getApplicationContext()).load(thumbnailPath).centerCrop().into(imageThumbnail); + } + + uploading = true; + uploadedThumbnailUrl = null; + UploadImageTask task = new UploadImageTask(thumbnailPath, uploadProgress, new UploadImageTask.UploadThumbnailHandler() { + @Override + public void onSuccess(String url) { + lastSelectedThumbnailFile = thumbnailPath; + uploadedThumbnailUrl = url; + uploading = false; + } + + @Override + public void onError(Exception error) { + View view = getView(); + if (context != null && view != null) { + showError(getString(R.string.image_upload_failed)); + } + lastSelectedThumbnailFile = null; + imageThumbnail.setImageDrawable(null); + uploading = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void probeVideo(String filePath) { + VideoProbeTask task = new VideoProbeTask(filePath, new VideoProbeTask.VideoProbeHandler() { + @Override + public void onVideoProbed(VideoInformation result) { + checkAndTranscodeVideo(filePath, result); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + + } + + private void checkAndTranscodeVideo(String filePath, VideoInformation videoInformation) { + boolean transcodeRequired = (videoInformation == null || + !H264_CODEC.equalsIgnoreCase(videoInformation.getCodecName()) || + MAX_VIDEO_DIMENSION < videoInformation.getWidth() || MAX_VIDEO_DIMENSION < videoInformation.getHeight() || + MAX_BITRATE < videoInformation.getBitrate()); + + String scalePart = ""; + if (videoInformation != null) { + // check the max dimension that we need to scale + int videoWidth = videoInformation.getWidth(); + int videoHeight = videoInformation.getHeight(); + // get the highest dimension + int maxDimension = Math.max(videoWidth, videoHeight); + if (maxDimension > MAX_VIDEO_DIMENSION) { + scalePart = maxDimension == videoWidth ? String.format("-vf scale=%d:-2", MAX_VIDEO_DIMENSION) : String.format("-vf scale=-2:%d", MAX_VIDEO_DIMENSION); + } + } + + Context context = getContext(); + String outputPath = String.format("%s/videos", Utils.getAppInternalStorageDir(context)); + File dir = new File(outputPath); + if (!dir.isDirectory()) { + dir.mkdirs(); + } + + boolean hasFullDuration = videoInformation != null && videoInformation.getDurationSeconds() > 0; + Helper.setViewVisibility(optimizationRealProgress, hasFullDuration ? View.VISIBLE : View.GONE); + Helper.setViewVisibility(optimizationProgress, hasFullDuration ? View.GONE : View.VISIBLE); + + File sourceFile = new File(filePath); + String filename = sourceFile.getName(); + if (!filename.endsWith(".mp4")) { + int lastDotIndex = filename.lastIndexOf('.'); + filename = String.format("%s.mp4", lastDotIndex > -1 ? filename.substring(0, lastDotIndex) : filename); + } + + String videoFilePath = String.format("%s/%s", outputPath, filename); + File targetFile = new File(videoFilePath); + if (targetFile.exists()) { + targetFile.delete(); + } + + transcodeInProgress = true; + videoTranscodeTask = new VideoTranscodeTask(filePath, videoFilePath, scalePart, transcodeRequired, new VideoTranscodeTask.VideoTranscodeHandler() { + @Override + public void onProgress(int time) { + if (context != null) { + int currentDuration = Double.valueOf(time / 1000.0).intValue(); + int fullDuration = videoInformation != null ? videoInformation.getDurationSeconds() : 0; + long elapsed = System.currentTimeMillis() - transcodeStartTime; + String completedDurationText = Helper.formatDuration(currentDuration); + if (fullDuration > 0) { + completedDurationText = String.format("%s / %s", completedDurationText, Helper.formatDuration(fullDuration)); + int percentComplete = Double.valueOf(Math.ceil((double) currentDuration / (double) fullDuration * 100.0)).intValue(); + optimizationRealProgress.setProgress(percentComplete); + } + + + String text = context.getString(R.string.completed_video_duration, completedDurationText); + Helper.setViewText(textOptimizationProgress, text); + Helper.setViewText(textOptimizationElapsed, Helper.formatDuration(Double.valueOf(elapsed / 1000.0).longValue())); + } + } + + @Override + public void onSuccess(String outputFilePath) { + transcodedFilePath = outputFilePath; + transcodeInProgress = false; + Helper.setViewText(textOptimizationStatus, R.string.video_optimized); + Helper.setViewVisibility(optimizationRealProgress, View.GONE); + Helper.setViewVisibility(optimizationProgress, View.GONE); + Helper.setViewVisibility(textOptimizationProgress, View.GONE); + } + + @Override + public void onErrorOrCancelled() { + transcodeInProgress = false; + Helper.setViewText(textOptimizationStatus, R.string.video_optimize_failed); + Helper.setViewVisibility(optimizationRealProgress, View.GONE); + Helper.setViewVisibility(optimizationProgress, View.GONE); + Helper.setViewVisibility(textOptimizationProgress, View.GONE); + } + }); + + transcodeStartTime = System.currentTimeMillis(); + videoTranscodeTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void cancelOnFatalCondition(String message) { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.showError(message); + new Handler().postDelayed(new Runnable() { + @Override + public void run() { + activity.onBackPressed(); + } + }, 100); + } + } + + public void onResume() { + super.onResume(); + if (!Lbry.SDK_READY) { + cancelOnFatalCondition(getString(R.string.sdk_initializing_functionality)); + return; + } + + checkParams(); + updateFieldsFromCurrentClaim(); + + if (currentClaim == null && (currentGalleryItem != null || !Helper.isNullOrEmpty(currentFilePath))) { + // load file information + checkPublishFile(); + } + + Context context = getContext(); + Helper.setWunderbarValue(null, context); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Channel Form", "ChannelForm"); + activity.addStoragePermissionListener(this); + if (editMode) { + ActionBar actionBar = activity.getSupportActionBar(); + if (actionBar != null) { + actionBar.setTitle(R.string.edit_content); + } + } + } + + if (!Lbry.SDK_READY) { + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.addSdkStatusListener(this); + } + } else { + onSdkReady(); + } + + String filterText = Helper.getValue(inputTagFilter.getText()); + updateSuggestedTags(filterText, SUGGESTED_LIMIT, true); + } + + public void onSdkReady() { + fetchChannels(); + onWalletBalanceUpdated(Lbry.walletBalance); + } + + private void fetchChannels() { + if (Lbry.ownChannels != null && Lbry.ownChannels.size() > 0) { + updateChannelList(Lbry.ownChannels); + return; + } + + fetchingChannels = true; + disableChannelSpinner(); + ClaimListTask task = new ClaimListTask(Claim.TYPE_CHANNEL, progressLoadingChannels, new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + Lbry.ownChannels = new ArrayList<>(claims); + updateChannelList(Lbry.ownChannels); + enableChannelSpinner(); + fetchingChannels = false; + } + + @Override + public void onError(Exception error) { + enableChannelSpinner(); + fetchingChannels = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + private void disableChannelSpinner() { + Helper.setViewEnabled(channelSpinner, false); + hideInlineChannelCreator(); + } + private void enableChannelSpinner() { + Helper.setViewEnabled(channelSpinner, true); + if (channelSpinner != null) { + Claim selectedClaim = (Claim) channelSpinner.getSelectedItem(); + if (selectedClaim != null) { + if (selectedClaim.isPlaceholder()) { + showInlineChannelCreator(); + } else { + hideInlineChannelCreator(); + } + } + } + } + private void showInlineChannelCreator() { + Helper.setViewVisibility(inlineChannelCreator, View.VISIBLE); + } + private void hideInlineChannelCreator() { + Helper.setViewVisibility(inlineChannelCreator, View.GONE); + } + + private void updateChannelList(List channels) { + if (channelSpinnerAdapter == null) { + Context context = getContext(); + if (context != null) { + channelSpinnerAdapter = new InlineChannelSpinnerAdapter(context, R.layout.spinner_item_channel, new ArrayList<>(channels)); + channelSpinnerAdapter.addPlaceholder(true); + channelSpinnerAdapter.notifyDataSetChanged(); + } + } else { + channelSpinnerAdapter.clear(); + channelSpinnerAdapter.addAll(channels); + channelSpinnerAdapter.addPlaceholder(true); + channelSpinnerAdapter.notifyDataSetChanged(); + } + + if (channelSpinner != null) { + channelSpinner.setAdapter(channelSpinnerAdapter); + } + + if (channelSpinnerAdapter != null && channelSpinner != null) { + if (editMode && currentClaim.getSigningChannel() != null && !editChannelSpinnerLoaded) { + int position = channelSpinnerAdapter.getItemPosition(currentClaim.getSigningChannel()); + if (position > -1) { + channelSpinner.setSelection(position); + } + editChannelSpinnerLoaded = true; + } else { + if (channelSpinnerAdapter.getCount() > 2) { + // if anonymous displayed, select first channel if available + channelSpinner.setSelection(2); + } else if (channelSpinnerAdapter.getCount() > 1) { + // select anonymous + channelSpinner.setSelection(1); + } + } + } + } + + private Claim buildPublishClaim() { + Claim claim = new Claim(); + + claim.setName(Helper.getValue(inputAddress.getText())); + claim.setAmount(Helper.getValue(inputDeposit.getText())); + + Claim.StreamMetadata metadata = new Claim.StreamMetadata(); + metadata.setTitle(Helper.getValue(inputTitle.getText())); + metadata.setDescription(Helper.getValue(inputDescription.getText())); + metadata.setTags(Helper.getTagsForTagObjects(addedTagsAdapter.getTags())); + + Claim selectedChannel = (Claim) channelSpinner.getSelectedItem(); + if (selectedChannel != null && !selectedChannel.isPlaceholder() && !selectedChannel.isPlaceholderAnonymous()) { + claim.setSigningChannel(selectedChannel); + } + if (switchPrice.isChecked()) { + Fee fee = new Fee(); + fee.setCurrency((String) priceCurrencySpinner.getSelectedItem()); + fee.setAmount(Helper.getValue(inputPrice.getText())); + metadata.setFee(fee); + } + if (!Helper.isNullOrEmpty(uploadedThumbnailUrl)) { + Claim.Resource thumbnail = new Claim.Resource(); + thumbnail.setUrl(uploadedThumbnailUrl); + metadata.setThumbnail(thumbnail); + } + + Language selectedLanguage = (Language) languageSpinner.getSelectedItem(); + if (selectedLanguage != null) { + metadata.setLanguages(Arrays.asList(selectedLanguage.getCode())); + } + + License selectedLicense = (License) licenseSpinner.getSelectedItem(); + if (selectedLicense != null) { + boolean otherLicense = Arrays.asList( + Predefined.LICENSE_COPYRIGHTED.toLowerCase(), + Predefined.LICENSE_OTHER.toLowerCase()).contains(selectedLicense.getName().toLowerCase()); + metadata.setLicense(otherLicense ? Helper.getValue(inputOtherLicenseDescription.getText()) : selectedLicense.getName()); + metadata.setLicenseUrl(selectedLicense.getUrl()); + } + + claim.setValueType(Claim.TYPE_STREAM); + claim.setValue(metadata); + + return claim; + } + + private boolean validatePublishClaim(Claim claim) { + if (Helper.isNullOrEmpty(claim.getTitle())) { + showError(getString(R.string.please_provide_title)); + return false; + } + if (Helper.isNullOrEmpty(claim.getName())) { + showError(getString(R.string.please_specify_address)); + return false; + } + if (!LbryUri.isNameValid(claim.getName())) { + showError(getString(R.string.address_invalid_characters)); + return false; + } + if (!editMode && Helper.claimNameExists(claim.getName())) { + showError(getString(R.string.address_already_used)); + return false; + } + + String publishFilePath = currentGalleryItem != null ? currentGalleryItem.getFilePath() : currentFilePath; + if (!editMode && Helper.isNullOrEmpty(publishFilePath) && Helper.isNullOrEmpty(transcodedFilePath)) { + showError(getString(R.string.no_file_selected)); + return false; + } + + return true; + } + + private void publishClaim(Claim claim) { + String finalFilePath = transcodedFilePath; + if (Helper.isNullOrEmpty(finalFilePath)) { + finalFilePath = currentGalleryItem != null ? currentGalleryItem.getFilePath() : currentFilePath; + } + saveInProgress = true; + PublishClaimTask task = new PublishClaimTask(claim, finalFilePath, progressPublish, new ClaimResultHandler() { + @Override + public void beforeStart() { + preSave(); + } + + @Override + public void onSuccess(Claim claimResult) { + postSave(); + + // Run the logPublish task + if (!BuildConfig.DEBUG) { + LogPublishTask logPublish = new LogPublishTask(claimResult); + logPublish.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + // publish done + Bundle bundle = new Bundle(); + bundle.putString("claim_id", claimResult.getClaimId()); + bundle.putString("claim_name", claimResult.getName()); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_PUBLISH, bundle); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.showMessage(R.string.publish_successful); + activity.openPublishesOnSuccessfulPublish(); + } + } + + @Override + public void onError(Exception error) { + showError(error.getMessage()); + postSave(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void preSave() { + saveInProgress = true; + + // disable input views + Helper.setViewEnabled(channelSpinner, false); + Helper.setViewEnabled(inputTitle, false); + Helper.setViewEnabled(inputDescription, false); + Helper.setViewEnabled(inputTagFilter, false); + Helper.setViewEnabled(inputAddress, false); + Helper.setViewEnabled(inputDeposit, false); + Helper.setViewEnabled(inputPrice, false); + Helper.setViewEnabled(inputOtherLicenseDescription, false); + Helper.setViewEnabled(switchPrice, false); + Helper.setViewEnabled(languageSpinner, false); + Helper.setViewEnabled(licenseSpinner, false); + Helper.setViewEnabled(priceCurrencySpinner, false); + Helper.setViewEnabled(linkGenerateAddress, false); + + Helper.setViewEnabled(linkShowExtraFields, false); + Helper.setViewEnabled(linkPublishCancel, false); + Helper.setViewEnabled(buttonPublish, false); + } + + private void postSave() { + Helper.setViewEnabled(channelSpinner, true); + Helper.setViewEnabled(inputTitle, true); + Helper.setViewEnabled(inputDescription, true); + Helper.setViewEnabled(inputTagFilter, false); + Helper.setViewEnabled(inputAddress, editMode ? false : true); + Helper.setViewEnabled(inputDeposit, true); + Helper.setViewEnabled(inputPrice, true); + Helper.setViewEnabled(inputOtherLicenseDescription, true); + Helper.setViewEnabled(switchPrice, true); + Helper.setViewEnabled(languageSpinner, true); + Helper.setViewEnabled(licenseSpinner, true); + Helper.setViewEnabled(priceCurrencySpinner, true); + Helper.setViewEnabled(linkGenerateAddress, true); + + Helper.setViewEnabled(linkShowExtraFields, true); + Helper.setViewEnabled(linkPublishCancel, true); + Helper.setViewEnabled(buttonPublish, true); + + saveInProgress = false; + } + + + @Override + public boolean shouldHideGlobalPlayer() { + return true; + } + + @Override + public boolean shouldSuspendGlobalPlayer() { + return true; + } + + @Override + public void onTagClicked(Tag tag, int customizeMode) { + if (customizeMode == TagListAdapter.CUSTOMIZE_MODE_ADD) { + addTag(tag); + } else if (customizeMode == TagListAdapter.CUSTOMIZE_MODE_REMOVE) { + removeTag(tag); + } + } + + public void setFilter(String filter) { + currentFilter = filter; + updateSuggestedTags(currentFilter, SUGGESTED_LIMIT, true); + } + private void checkNoAddedTags() { + Helper.setViewVisibility(noTagsView, addedTagsAdapter == null || addedTagsAdapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + private void checkNoTagResults() { + Helper.setViewVisibility(noTagResultsView, suggestedTagsAdapter == null || suggestedTagsAdapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + public void addTag(Tag tag) { + if (saveInProgress) { + return; + } + + if (addedTagsAdapter.getTags().contains(tag)) { + Snackbar.make(getView(), getString(R.string.tag_already_added, tag.getName()), Snackbar.LENGTH_LONG).show(); + return; + } + if (addedTagsAdapter.getItemCount() == 5) { + Snackbar.make(getView(), R.string.tag_limit_reached, Snackbar.LENGTH_LONG).show(); + return; + } + + addedTagsAdapter.addTag(tag); + if (suggestedTagsAdapter != null) { + suggestedTagsAdapter.removeTag(tag); + } + updateSuggestedTags(currentFilter, SUGGESTED_LIMIT, false); + + checkNoAddedTags(); + checkNoTagResults(); + } + public void removeTag(Tag tag) { + if (saveInProgress) { + return; + } + addedTagsAdapter.removeTag(tag); + updateSuggestedTags(currentFilter, SUGGESTED_LIMIT, false); + checkNoAddedTags(); + checkNoTagResults(); + } + private void updateSuggestedTags(String filter, int limit, boolean clearPrevious) { + UpdateSuggestedTagsTask task = new UpdateSuggestedTagsTask( + filter, + limit, + addedTagsAdapter, + suggestedTagsAdapter, + clearPrevious, + true, new UpdateSuggestedTagsTask.KnownTagsHandler() { + @Override + public void onSuccess(List tags) { + if (suggestedTagsAdapter == null) { + suggestedTagsAdapter = new TagListAdapter(tags, getContext()); + suggestedTagsAdapter.setCustomizeMode(TagListAdapter.CUSTOMIZE_MODE_ADD); + suggestedTagsAdapter.setClickListener(PublishFormFragment.this); + if (suggestedTagsList != null) { + suggestedTagsList.setAdapter(suggestedTagsAdapter); + } + } else { + suggestedTagsAdapter.setTags(tags); + } + + checkNoAddedTags(); + checkNoTagResults(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + @Override + public void onWalletBalanceUpdated(WalletBalance walletBalance) { + if (walletBalance != null && inlineDepositBalanceValue != null) { + inlineDepositBalanceValue.setText(Helper.shortCurrencyFormat(walletBalance.getAvailable().doubleValue())); + } + } + + private void setupInlineChannelCreator( + View container, + TextInputEditText inputChannelName, + TextInputEditText inputDeposit, + View inlineBalanceView, + TextView inlineBalanceValue, + View linkCancel, + MaterialButton buttonCreate, + View progressView) { + inputDeposit.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + Helper.setViewVisibility(inlineBalanceView, hasFocus ? View.VISIBLE : View.INVISIBLE); + } + }); + + linkCancel.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + Helper.setViewText(inputChannelName, null); + Helper.setViewText(inputDeposit, null); + Helper.setViewVisibility(container, View.GONE); + } + }); + + buttonCreate.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + // validate deposit and channel name + String channelNameString = Helper.normalizeChannelName(Helper.getValue(inputChannelName.getText())); + Claim claimToSave = new Claim(); + claimToSave.setName(channelNameString); + String channelName = claimToSave.getName().startsWith("@") ? claimToSave.getName().substring(1) : claimToSave.getName(); + String depositString = Helper.getValue(inputDeposit.getText()); + if ("@".equals(channelName) || Helper.isNullOrEmpty(channelName)) { + showError(getString(R.string.please_enter_channel_name)); + return; + } + if (!LbryUri.isNameValid(channelName)) { + showError(getString(R.string.channel_name_invalid_characters)); + return; + } + if (Helper.channelExists(channelName)) { + showError(getString(R.string.channel_name_already_created)); + return; + } + + double depositAmount = 0; + try { + depositAmount = Double.valueOf(depositString); + } catch (NumberFormatException ex) { + // pass + showError(getString(R.string.please_enter_valid_deposit)); + return; + } + if (depositAmount == 0) { + String error = getResources().getQuantityString(R.plurals.min_deposit_required, depositAmount == 1 ? 1 : 2, String.valueOf(Helper.MIN_DEPOSIT)); + showError(error); + return; + } + if (Lbry.walletBalance == null || Lbry.walletBalance.getAvailable().doubleValue() < depositAmount) { + showError(getString(R.string.deposit_more_than_balance)); + return; + } + + ChannelCreateUpdateTask task = new ChannelCreateUpdateTask( + claimToSave, new BigDecimal(depositString), false, progressView, new ClaimResultHandler() { + @Override + public void beforeStart() { + Helper.setViewEnabled(inputChannelName, false); + Helper.setViewEnabled(inputDeposit, false); + Helper.setViewEnabled(buttonCreate, false); + Helper.setViewEnabled(linkCancel, false); + } + + @Override + public void onSuccess(Claim claimResult) { + if (!BuildConfig.DEBUG) { + LogPublishTask logPublishTask = new LogPublishTask(claimResult); + logPublishTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + // channel created + Bundle bundle = new Bundle(); + bundle.putString("claim_id", claimResult.getClaimId()); + bundle.putString("claim_name", claimResult.getName()); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_CHANNEL_CREATE, bundle); + + // add the claim to the channel list and set it as the selected item + channelSpinnerAdapter.add(claimResult); + channelSpinner.setSelection(channelSpinnerAdapter.getCount() - 1); + + Helper.setViewEnabled(inputChannelName, true); + Helper.setViewEnabled(inputDeposit, true); + Helper.setViewEnabled(buttonCreate, true); + Helper.setViewEnabled(linkCancel, true); + } + + @Override + public void onError(Exception error) { + Helper.setViewEnabled(inputChannelName, true); + Helper.setViewEnabled(inputDeposit, true); + Helper.setViewEnabled(buttonCreate, true); + Helper.setViewEnabled(linkCancel, true); + showError(error.getMessage()); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + }); + + Helper.setViewText(inlineBalanceValue, Helper.shortCurrencyFormat(Lbry.walletBalance.getAvailable().doubleValue())); + } + + private void showError(String message) { + Context context = getContext(); + if (context != null) { + Snackbar.make(getView(), message, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + } + + private void checkUploadButton() { + + } + + @Override + public void onStoragePermissionGranted() { + if (launchPickerPending) { + launchPickerPending = false; + launchFilePicker(); + } + } + + @Override + public void onStoragePermissionRefused() { + showError(getString(R.string.storage_permission_rationale_images)); + launchPickerPending = false; + } + + @Override + public void onFilePicked(String filePath) { + if (Helper.isNullOrEmpty(filePath)) { + Snackbar.make(getView(), R.string.undetermined_image_filepath, Snackbar.LENGTH_LONG).setBackgroundTint( + ContextCompat.getColor(getContext(), R.color.red)).show(); + return; + } + + Context context = getContext(); + if (context != null) { + if (filePath.equalsIgnoreCase(lastSelectedThumbnailFile)) { + // previous selected cover was uploaded successfully + return; + } + + Uri fileUri = Uri.fromFile(new File(filePath)); + Glide.with(context.getApplicationContext()).load(fileUri).centerCrop().into(imageThumbnail); + uploadThumbnail(filePath); + } + } + + @Override + public void onFilePickerCancelled() { + // nothing to do here + // At some point in the future, allow file picking for publish file? + } + + private static class VideoProbeTask extends AsyncTask { + private String filePath; + private VideoProbeHandler handler; + public VideoProbeTask(String filePath, VideoProbeHandler handler) { + this.filePath = filePath; + this.handler = handler; + } + protected VideoInformation doInBackground(Void... params) { + try { + int code = FFprobe.execute(String.format("-v quiet -show_streams -select_streams v -print_format json -i \"%s\"", filePath)); + if (code == Config.RETURN_CODE_SUCCESS) { + String json = Config.getLastCommandOutput(); + JSONObject result = new JSONObject(json); + if (result.has("streams")) { + JSONArray streams = result.getJSONArray("streams"); + if (streams.length() > 0) { + JSONObject stream = streams.getJSONObject(0); + VideoInformation videoInformation = VideoInformation.fromJSONObject(stream); + return videoInformation; + } + } + } + } catch (JSONException ex) { + // pass + } + return null; + } + protected void onPostExecute(VideoInformation result) { + if (handler != null) { + handler.onVideoProbed(result); + } + } + + public interface VideoProbeHandler { + void onVideoProbed(VideoInformation result); + } + } + + private static class VideoTranscodeTask extends AsyncTask { + + private String filePath; + private String scaleFlag; + private String outputFilePath; + private boolean transcodeRequired; + private VideoTranscodeHandler handler; + + public VideoTranscodeTask(String filePath, String outputFilePath, String scaleFlag, boolean transcodeRequired, VideoTranscodeHandler handler) { + this.handler = handler; + this.filePath = filePath; + this.outputFilePath = outputFilePath; + this.scaleFlag = scaleFlag; + this.transcodeRequired = transcodeRequired; + } + + protected Boolean doInBackground(Void... params) { + String movFlagsCommand = String.format("-i \"%s\" -movflags +faststart \"%s\"", filePath, outputFilePath); + String command = transcodeRequired ? String.format( + "-i \"%s\" " + + "-c:v libx264 " + + "-c:a aac -b:a 128k " + + "%s " + + "-crf 27 -preset ultrafast " + + "-pix_fmt yuv420p " + + "-maxrate 5000K -bufsize 5000K " + + "-movflags +faststart \"%s\"", filePath, scaleFlag, outputFilePath) : movFlagsCommand; + + Config.enableStatisticsCallback(new StatisticsCallback() { + @Override + public void apply(Statistics statistics) { + publishProgress(statistics.getTime()); + } + }); + int code = FFmpeg.execute(command); + return code == Config.RETURN_CODE_SUCCESS; + } + + protected void onProgressUpdate(Integer... times) { + if (handler != null) { + for (Integer time : times) { + handler.onProgress(time); + } + } + } + + protected void onPostExecute(Boolean result) { + if (handler != null) { + if (result) { + handler.onSuccess(outputFilePath); + } else { + handler.onErrorOrCancelled(); + } + } + } + + public interface VideoTranscodeHandler { + void onProgress(int time); + void onSuccess(String outputFilePath); + void onErrorOrCancelled(); + } + } + + @Data + private static class VideoInformation { + private String codecName; + private int width; + private int height; + private int durationSeconds; + private long bitrate; + + private static int tryParseDuration(JSONObject streamObject) { + String durationString = Helper.getJSONString("duration", "0", streamObject); + double parsedDuration = Helper.parseDouble(durationString, 0); + if (parsedDuration > 0) { + return Double.valueOf(parsedDuration).intValue(); + } + + try { + if (streamObject.has("tags") && !streamObject.isNull("tags")) { + JSONObject tags = streamObject.getJSONObject("tags"); + String tagDurationString = Helper.getJSONString("DURATION", null, tags); + if (Helper.isNull(tagDurationString)) { + tagDurationString = Helper.getJSONString("duration", null, tags); + } + if (!Helper.isNullOrEmpty(tagDurationString) && tagDurationString.indexOf(':') > -1) { + String[] parts = tagDurationString.split(":"); + if (parts.length == 3) { + int hours = Helper.parseInt(parts[0], 0); + int minutes = Helper.parseInt(parts[1], 0); + int seconds = Helper.parseDouble(parts[2], 0).intValue(); + return (hours * 60 * 60) + (minutes * 60) + seconds; + } + } + + } + } catch (JSONException ex) { + return 0; + } + + return 0; + } + + public static VideoInformation fromJSONObject(JSONObject streamObject) { + VideoInformation info = new VideoInformation(); + info.setCodecName(Helper.getJSONString("codec_name", null, streamObject)); + info.setWidth(Helper.getJSONInt("width", 0, streamObject)); + info.setHeight(Helper.getJSONInt("height", 0, streamObject)); + info.setBitrate(Helper.getJSONLong("bit_rate", 0, streamObject)); + info.setDurationSeconds(tryParseDuration(streamObject)); + + return info; + } + } + + private static class CreateThumbnailTask extends AsyncTask { + private Context context; + private String filePath; + private String type; + private CreateThumbnailHandler handler; + private Exception error; + public CreateThumbnailTask(String filePath, String type, Context context, CreateThumbnailHandler handler) { + this.context = context; + this.type = type; + this.filePath = filePath; + this.handler = handler; + } + protected String doInBackground(Void... params) { + String thumbnailPath = null; + FileOutputStream os = null; + Bitmap thumbnail = null; + try { + File cacheDir = context.getExternalCacheDir(); + File thumbnailsDir = new File(String.format("%s/thumbnails", cacheDir.getAbsolutePath())); + if (!thumbnailsDir.isDirectory()) { + thumbnailsDir.mkdirs(); + } + + // save the thumbnail to the path + thumbnailPath = String.format("%s/%s.png", thumbnailsDir.getAbsolutePath(), Helper.makeid(8)); + if ("video".equals(type)) { + thumbnail = ThumbnailUtils.createVideoThumbnail(filePath, MediaStore.Video.Thumbnails.MINI_KIND); + } else { + Bitmap source = BitmapFactory.decodeFile(filePath); + // MINI_KIND dimensions + thumbnail = Bitmap.createScaledBitmap(source, 512, 384, false); + } + + os = new FileOutputStream(thumbnailPath); + thumbnail.compress(Bitmap.CompressFormat.PNG, 80, os); + } catch (Exception ex) { + error = ex; + return null; + } finally { + Helper.closeCloseable(os); + } + + return thumbnailPath; + } + protected void onPostExecute(String thumbnailPath) { + if (handler != null) { + if (!Helper.isNullOrEmpty(thumbnailPath)) { + handler.onSuccess(thumbnailPath); + } else { + handler.onError(error); + } + } + } + + public interface CreateThumbnailHandler { + void onSuccess(String thumbnailPath); + void onError(Exception error); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/publish/PublishFragment.java b/app/src/main/java/io/lbry/browser/ui/publish/PublishFragment.java new file mode 100644 index 00000000..af3a7eff --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/publish/PublishFragment.java @@ -0,0 +1,436 @@ +package io.lbry.browser.ui.publish; + +import android.Manifest; +import android.annotation.SuppressLint; +import android.content.Context; +import android.content.Intent; +import android.content.pm.PackageManager; +import android.graphics.Color; +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 android.widget.Toast; + +import androidx.annotation.NonNull; +import androidx.camera.core.Camera; +import androidx.camera.core.CameraSelector; +import androidx.camera.core.CameraX; +import androidx.camera.core.Preview; +import androidx.camera.lifecycle.ProcessCameraProvider; +import androidx.camera.view.PreviewView; +import androidx.core.content.ContextCompat; +import androidx.lifecycle.LifecycleOwner; +import androidx.recyclerview.widget.GridLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import com.google.android.material.snackbar.Snackbar; +import com.google.common.util.concurrent.ListenableFuture; + +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.concurrent.ExecutionException; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.GalleryGridAdapter; +import io.lbry.browser.listener.CameraPermissionListener; +import io.lbry.browser.listener.FilePickerListener; +import io.lbry.browser.listener.StoragePermissionListener; +import io.lbry.browser.model.GalleryItem; +import io.lbry.browser.model.NavMenuItem; +import io.lbry.browser.tasks.localdata.LoadGalleryItemsTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; + +public class PublishFragment extends BaseFragment implements + CameraPermissionListener, FilePickerListener, StoragePermissionListener { + + private PreviewView cameraPreview; + private RecyclerView galleryGrid; + private GalleryGridAdapter adapter; + private TextView noVideosLoaded; + private View loading; + + private View buttonRecord; + private View buttonTakePhoto; + private View buttonUpload; + + private boolean loadGalleryItemsPending; + private boolean launchFilePickerPending; + private boolean recordPending; + private boolean takePhotoPending; + private ListenableFuture cameraProviderFuture; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_publish, container, false); + + noVideosLoaded = root.findViewById(R.id.publish_grid_no_videos); + loading = root.findViewById(R.id.publish_grid_loading); + cameraPreview = root.findViewById(R.id.publish_camera_preview); + + Context context = getContext(); + galleryGrid = root.findViewById(R.id.publish_video_grid); + GridLayoutManager glm = new GridLayoutManager(context, 3); + galleryGrid.setLayoutManager(glm); + galleryGrid.addItemDecoration(new GalleryGridAdapter.GalleryGridItemDecoration( + 3, Helper.getScaledValue(3, context.getResources().getDisplayMetrics().density))); + + buttonRecord = root.findViewById(R.id.publish_record_button); + buttonTakePhoto = root.findViewById(R.id.publish_photo_button); + buttonUpload = root.findViewById(R.id.publish_upload_button); + + buttonRecord.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + checkCameraPermissionAndRecord(); + } + }); + buttonTakePhoto.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + checkCameraPermissionAndTakePhoto(); + } + }); + buttonUpload.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + checkStoragePermissionAndLaunchFilePicker(); + } + }); + + return root; + } + + private boolean cameraAvailable() { + Context context = getContext(); + return context != null && context.getPackageManager().hasSystemFeature(PackageManager.FEATURE_CAMERA_ANY); + } + + private void showCameraPreview() { + buttonRecord.setBackgroundColor(Color.TRANSPARENT); + buttonTakePhoto.setBackgroundColor(Color.TRANSPARENT); + displayPreviewWithCameraX(); + } + + private void displayPreviewWithCameraX() { + Context context = getContext(); + if (context != null) { + cameraProviderFuture = ProcessCameraProvider.getInstance(context); + cameraProviderFuture.addListener(new Runnable() { + @Override + public void run() { + try { + ProcessCameraProvider cameraProvider = cameraProviderFuture.get(); + if (cameraProvider != null) { + Preview preview = new Preview.Builder().build(); + CameraSelector cameraSelector = new CameraSelector.Builder() + .requireLensFacing(CameraSelector.LENS_FACING_BACK) + .build(); + + Camera camera = cameraProvider.bindToLifecycle((LifecycleOwner) context, cameraSelector, preview); + preview.setSurfaceProvider(cameraPreview.createSurfaceProvider(camera.getCameraInfo())); + } + } catch (ExecutionException | InterruptedException ex) { + // pass + } + } + }, ContextCompat.getMainExecutor(context)); + } + } + + private void checkCameraPermissionAndRecord() { + if (!Lbry.SDK_READY) { + Snackbar.make(getView(), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + Context context = getContext(); + if (!MainActivity.hasPermission(Manifest.permission.CAMERA, context)) { + recordPending = true; + MainActivity.requestPermission( + Manifest.permission.CAMERA, + MainActivity.REQUEST_CAMERA_PERMISSION, + getString(R.string.camera_permission_rationale_record), + context, + true); + } else { + record(); + } + } + + private void checkCameraPermissionAndTakePhoto() { + if (!Lbry.SDK_READY) { + Snackbar.make(getView(), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + Context context = getContext(); + if (!MainActivity.hasPermission(Manifest.permission.CAMERA, context)) { + takePhotoPending = true; + MainActivity.requestPermission( + Manifest.permission.CAMERA, + MainActivity.REQUEST_CAMERA_PERMISSION, + getString(R.string.camera_permission_rationale_photo), + context, + true); + } else { + takePhoto(); + } + } + + private void takePhoto() { + Context context = getContext(); + if (context instanceof MainActivity) { + takePhotoPending = false; + ((MainActivity) context).requestTakePhoto(); + } + } + + private void record() { + Context context = getContext(); + if (context instanceof MainActivity) { + recordPending = false; + ((MainActivity) context).requestVideoCapture(); + } + } + + private void checkStoragePermissionAndLaunchFilePicker() { + if (!Lbry.SDK_READY) { + Snackbar.make(getView(), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + Context context = getContext(); + if (MainActivity.hasPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, context)) { + launchFilePickerPending = false; + launchFilePicker(); + } else { + launchFilePickerPending = true; + MainActivity.requestPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, + MainActivity.REQUEST_STORAGE_PERMISSION, + getString(R.string.storage_permission_rationale_images), + context, + true); + } + } + + private void launchFilePicker() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity.startingFilePickerActivity = true; + Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT); + intent.setType("*/*"); + ((MainActivity) context).startActivityForResult( + Intent.createChooser(intent, getString(R.string.upload_file)), + MainActivity.REQUEST_FILE_PICKER); + } + } + + public void onResume() { + super.onResume(); + Context context = getContext(); + Helper.setWunderbarValue(null, context); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Publish", "Publish"); + activity.addCameraPermissionListener(this); + activity.addFilePickerListener(this); + activity.addStoragePermissionListener(this); + activity.hideFloatingWalletBalance(); + + + if (cameraAvailable() && MainActivity.hasPermission(Manifest.permission.CAMERA, context)) { + showCameraPreview(); + } + } + + checkStoragePermissionAndLoadVideos(); + } + + @SuppressLint("RestrictedApi") + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.removeCameraPermissionListener(this); + activity.removeStoragePermissionListener(this); + activity.showFloatingWalletBalance(); + if (!MainActivity.startingFilePickerActivity) { + activity.removeFilePickerListener(this); + } + } + CameraX.unbindAll(); + super.onStop(); + } + + private void checkStoragePermissionAndLoadVideos() { + Context context = getContext(); + if (MainActivity.hasPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE, context)) { + loadGalleryItems(); + } else { + loadGalleryItemsPending = true; + MainActivity.requestPermission( + Manifest.permission.WRITE_EXTERNAL_STORAGE, + MainActivity.REQUEST_STORAGE_PERMISSION, + getString(R.string.storage_permission_rationale_download), + context, + true); + } + } + + private void loadGalleryItems() { + Context context = getContext(); + Helper.setViewVisibility(noVideosLoaded, View.GONE); + LoadGalleryItemsTask task = new LoadGalleryItemsTask(loading, context, new LoadGalleryItemsTask.LoadGalleryHandler() { + @Override + public void onItemLoaded(GalleryItem item) { + if (context != null) { + if (adapter == null) { + adapter = new GalleryGridAdapter(Arrays.asList(item), context); + adapter.setListener(new GalleryGridAdapter.GalleryItemClickListener() { + @Override + public void onGalleryItemClicked(GalleryItem item) { + if (!Lbry.SDK_READY) { + Snackbar.make(getView(), R.string.sdk_initializing_functionality, Snackbar.LENGTH_LONG).show(); + return; + } + + Context context = getContext(); + if (context instanceof MainActivity) { + Map params = new HashMap<>(); + params.put("galleryItem", item); + params.put("suggestedUrl", getSuggestedPublishUrl()); + ((MainActivity) context).openFragment(PublishFormFragment.class, true, NavMenuItem.ID_ITEM_NEW_PUBLISH, params); + } + } + }); + } else { + adapter.addItem(item); + } + + if (galleryGrid.getAdapter() == null) { + galleryGrid.setAdapter(adapter); + } + Helper.setViewVisibility(loading, adapter == null || adapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + } + + @Override + public void onAllItemsLoaded(List items) { + if (context != null) { + if (adapter == null) { + adapter = new GalleryGridAdapter(items, context); + } else { + adapter.addItems(items); + } + + if (galleryGrid.getAdapter() == null) { + galleryGrid.setAdapter(adapter); + } + } + checkNoVideosLoaded(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void checkNoVideosLoaded() { + Helper.setViewVisibility(noVideosLoaded, adapter == null || adapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + + @Override + public void onCameraPermissionGranted() { + if (recordPending) { + // record video + record(); + } else if (takePhotoPending) { + // take a photo + takePhoto(); + } + } + + @Override + public void onCameraPermissionRefused() { + if (takePhotoPending) { + takePhotoPending = false; + Snackbar.make(getView(), R.string.camera_permission_rationale_photo, Toast.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + return; + } + + recordPending = false; + Snackbar.make(getView(), R.string.camera_permission_rationale_record, Toast.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + + @Override + public void onRecordAudioPermissionGranted() { + + } + + @Override + public void onRecordAudioPermissionRefused() { + + } + + @Override + public void onStoragePermissionGranted() { + if (loadGalleryItemsPending) { + loadGalleryItemsPending = false; + loadGalleryItems(); + } + if (launchFilePickerPending) { + launchFilePickerPending = false; + launchFilePicker(); + } + } + + @Override + public void onStoragePermissionRefused() { + Snackbar.make(getView(), R.string.storage_permission_rationale_videos, Snackbar.LENGTH_LONG).setBackgroundTint( + ContextCompat.getColor(getContext(), R.color.red) + ).show(); + } + + public String getSuggestedPublishUrl() { + Map params = getParams(); + if (params != null && params.containsKey("suggestedUrl")) { + return (String) params.get("suggestedUrl"); + } + return null; + } + + @Override + public boolean shouldHideGlobalPlayer() { + return true; + } + + @Override + public boolean shouldSuspendGlobalPlayer() { + return true; + } + + @Override + public void onFilePicked(String filePath) { + Context context = getContext(); + if (context instanceof MainActivity) { + Map params = new HashMap<>(); + params.put("directFilePath", filePath); + params.put("suggestedUrl", getSuggestedPublishUrl()); + ((MainActivity) context).openFragment(PublishFormFragment.class, true, NavMenuItem.ID_ITEM_NEW_PUBLISH, params); + } + } + + @Override + public void onFilePickerCancelled() { + + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/publish/PublishesFragment.java b/app/src/main/java/io/lbry/browser/ui/publish/PublishesFragment.java new file mode 100644 index 00000000..0ba048ff --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/publish/PublishesFragment.java @@ -0,0 +1,320 @@ +package io.lbry.browser.ui.publish; + +import android.content.Context; +import android.content.DialogInterface; +import android.graphics.Color; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.Menu; +import android.view.MenuItem; +import android.view.View; +import android.view.ViewGroup; +import android.widget.Button; +import android.widget.ProgressBar; + +import androidx.annotation.NonNull; +import androidx.appcompat.app.AlertDialog; +import androidx.appcompat.view.ActionMode; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import com.google.android.material.floatingactionbutton.FloatingActionButton; +import com.google.android.material.snackbar.Snackbar; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.ClaimListAdapter; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.listener.SelectionModeListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.NavMenuItem; +import io.lbry.browser.tasks.claim.AbandonHandler; +import io.lbry.browser.tasks.claim.AbandonStreamTask; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ClaimListTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; + +public class PublishesFragment extends BaseFragment implements ActionMode.Callback, SelectionModeListener, SdkStatusListener { + + private Button buttonNewPublish; + private FloatingActionButton fabNewPublish; + private ActionMode actionMode; + private View emptyView; + private View layoutSdkInitializing; + private ProgressBar loading; + private ProgressBar bigLoading; + private RecyclerView contentList; + private ClaimListAdapter adapter; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_publishes, container, false); + + buttonNewPublish = root.findViewById(R.id.publishes_create_button); + fabNewPublish = root.findViewById(R.id.publishes_fab_new_publish); + buttonNewPublish.setOnClickListener(newPublishClickListener); + fabNewPublish.setOnClickListener(newPublishClickListener); + + emptyView = root.findViewById(R.id.publishes_empty_container); + layoutSdkInitializing = root.findViewById(R.id.container_sdk_initializing); + contentList = root.findViewById(R.id.publishes_list); + LinearLayoutManager llm = new LinearLayoutManager(getContext()); + contentList.setLayoutManager(llm); + loading = root.findViewById(R.id.publishes_list_loading); + bigLoading = root.findViewById(R.id.publishes_list_big_loading); + + layoutSdkInitializing.setVisibility(Lbry.SDK_READY ? View.GONE : View.VISIBLE); + + return root; + } + + private View.OnClickListener newPublishClickListener = new View.OnClickListener() { + @Override + public void onClick(View view) { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openFragment(PublishFragment.class, true, NavMenuItem.ID_ITEM_NEW_PUBLISH); + } + } + }; + + @Override + public void onStart() { + super.onStart(); + Context context = getContext(); + if (context != null) { + MainActivity activity = (MainActivity) context; + activity.hideFloatingWalletBalance(); + } + } + + @Override + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.showFloatingWalletBalance(); + } + super.onStop(); + } + + public void onResume() { + super.onResume(); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.setWunderbarValue(null); + LbryAnalytics.setCurrentScreen(activity, "Publishes", "Publishes"); + } + + if (!Lbry.SDK_READY) { + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.addSdkStatusListener(this); + } + } else { + onSdkReady(); + } + } + + public void onSdkReady() { + Helper.setViewVisibility(layoutSdkInitializing, View.GONE); + Helper.setViewVisibility(fabNewPublish, View.VISIBLE); + if (adapter != null && contentList != null) { + contentList.setAdapter(adapter); + } + fetchPublishes(); + } + + public View getLoading() { + return (adapter == null || adapter.getItemCount() == 0) ? bigLoading : loading; + } + + private void checkNoPublishes() { + Helper.setViewVisibility(emptyView, adapter == null || adapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + + private void fetchPublishes() { + Helper.setViewVisibility(emptyView, View.GONE); + ClaimListTask task = new ClaimListTask(Arrays.asList(Claim.TYPE_STREAM, Claim.TYPE_REPOST), getLoading(), new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + Lbry.ownClaims = Helper.filterDeletedClaims(new ArrayList<>(claims)); + Context context = getContext(); + if (adapter == null) { + adapter = new ClaimListAdapter(claims, context); + adapter.setCanEnterSelectionMode(true); + adapter.setSelectionModeListener(PublishesFragment.this); + adapter.setListener(new ClaimListAdapter.ClaimListItemListener() { + @Override + public void onClaimClicked(Claim claim) { + if (context instanceof MainActivity) { + ((MainActivity) context).openFileClaim(claim); + } + } + }); + if (contentList != null) { + contentList.setAdapter(adapter); + } + } else { + adapter.setItems(claims); + } + + checkNoPublishes(); + } + + @Override + public void onError(Exception error) { + checkNoPublishes(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + public void onEnterSelectionMode() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.startSupportActionMode(this); + } + } + public void onItemSelectionToggled() { + if (actionMode != null) { + actionMode.setTitle(String.valueOf(adapter.getSelectedCount())); + actionMode.invalidate(); + } + } + public void onExitSelectionMode() { + if (actionMode != null) { + actionMode.finish(); + } + } + + @Override + public boolean onCreateActionMode(ActionMode actionMode, Menu menu) { + this.actionMode = actionMode; + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (!activity.isDarkMode()) { + activity.getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_VISIBLE); + } + } + + actionMode.getMenuInflater().inflate(R.menu.menu_claim_list, menu); + return true; + } + @Override + public void onDestroyActionMode(ActionMode actionMode) { + if (adapter != null) { + adapter.clearSelectedItems(); + adapter.setInSelectionMode(false); + adapter.notifyDataSetChanged(); + } + Context context = getContext(); + if (context != null) { + MainActivity activity = (MainActivity) context; + if (!activity.isDarkMode()) { + activity.getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR); + } + } + this.actionMode = null; + } + + @Override + public boolean onPrepareActionMode(androidx.appcompat.view.ActionMode actionMode, Menu menu) { + int selectionCount = adapter != null ? adapter.getSelectedCount() : 0; + menu.findItem(R.id.action_edit).setVisible(selectionCount == 1); + return true; + } + + @Override + public boolean onActionItemClicked(androidx.appcompat.view.ActionMode actionMode, MenuItem menuItem) { + if (R.id.action_edit == menuItem.getItemId()) { + if (adapter != null && adapter.getSelectedCount() > 0) { + Claim claim = adapter.getSelectedItems().get(0); + // start channel editor with the claim + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openPublishForm(claim); + } + + actionMode.finish(); + return true; + } + } + if (R.id.action_delete == menuItem.getItemId()) { + if (adapter != null && adapter.getSelectedCount() > 0) { + final List selectedClaims = new ArrayList<>(adapter.getSelectedItems()); + String message = getResources().getQuantityString(R.plurals.confirm_delete_publishes, selectedClaims.size()); + AlertDialog.Builder builder = new AlertDialog.Builder(getContext()). + setTitle(R.string.delete_selection). + setMessage(message) + .setPositiveButton(R.string.yes, new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + handleDeleteSelectedClaims(selectedClaims); + } + }).setNegativeButton(R.string.no, null); + builder.show(); + return true; + } + } + + return false; + } + + private void handleDeleteSelectedClaims(List selectedClaims) { + List claimIds = new ArrayList<>(); + + for (Claim claim : selectedClaims) { + claimIds.add(claim.getClaimId()); + } + + if (actionMode != null) { + actionMode.finish(); + } + + Helper.setViewVisibility(contentList, View.INVISIBLE); + Helper.setViewVisibility(fabNewPublish, View.INVISIBLE); + AbandonStreamTask task = new AbandonStreamTask(claimIds, bigLoading, new AbandonHandler() { + @Override + public void onComplete(List successfulClaimIds, List failedClaimIds, List errors) { + View root = getView(); + if (root != null) { + if (failedClaimIds.size() > 0) { + Snackbar.make(root, R.string.one_or_more_publishes_failed_abandon, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } else if (successfulClaimIds.size() == claimIds.size()) { + try { + String message = getResources().getQuantityString(R.plurals.publishes_deleted, successfulClaimIds.size()); + Snackbar.make(root, message, Snackbar.LENGTH_LONG).show(); + } catch (IllegalStateException ex) { + // pass + } + } + } + + Lbry.abandonedClaimIds.addAll(successfulClaimIds); + if (adapter != null) { + adapter.setItems(Helper.filterDeletedClaims(adapter.getItems())); + } + + Helper.setViewVisibility(contentList, View.VISIBLE); + Helper.setViewVisibility(fabNewPublish, View.VISIBLE); + checkNoPublishes(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } +} 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..3721ca27 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/verification/EmailVerificationFragment.java @@ -0,0 +1,201 @@ +package io.lbry.browser.ui.verification; + +import android.graphics.Color; +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(Color.RED).setTextColor(Color.WHITE).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(Color.RED).setTextColor(Color.WHITE).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(Color.RED).setTextColor(Color.WHITE).show(); + Helper.setViewEnabled(buttonResend, true); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/verification/ManualVerificationFragment.java b/app/src/main/java/io/lbry/browser/ui/verification/ManualVerificationFragment.java new file mode 100644 index 00000000..e901cb13 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/verification/ManualVerificationFragment.java @@ -0,0 +1,36 @@ +package io.lbry.browser.ui.verification; + +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 io.lbry.browser.R; +import io.lbry.browser.listener.SignInListener; +import io.lbry.browser.utils.Helper; +import lombok.Setter; + +public class ManualVerificationFragment extends Fragment { + @Setter + private SignInListener listener; + + public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_verification_manual, container, false); + + Helper.applyHtmlForTextView((TextView) root.findViewById(R.id.verification_manual_discord_verify)); + root.findViewById(R.id.verification_manual_continue_button).setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (listener != null) { + listener.onManualVerifyContinue(); + } + } + }); + + return root; + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/verification/PhoneVerificationFragment.java b/app/src/main/java/io/lbry/browser/ui/verification/PhoneVerificationFragment.java new file mode 100644 index 00000000..72789af0 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/verification/PhoneVerificationFragment.java @@ -0,0 +1,173 @@ +package io.lbry.browser.ui.verification; + +import android.content.Context; +import android.graphics.Color; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.EditText; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.core.content.res.ResourcesCompat; +import androidx.fragment.app.Fragment; + +import com.google.android.material.button.MaterialButton; +import com.google.android.material.snackbar.Snackbar; +import com.hbb20.CountryCodePicker; + +import io.lbry.browser.R; +import io.lbry.browser.listener.SignInListener; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.tasks.verification.PhoneNewVerifyTask; +import io.lbry.browser.utils.Helper; +import lombok.Setter; + +public class PhoneVerificationFragment extends Fragment { + @Setter + private SignInListener listener; + + private View layoutCollect; + private View layoutVerify; + private MaterialButton continueButton; + private MaterialButton verifyButton; + private View editButton; + private TextView textVerifyParagraph; + private CountryCodePicker countryCodePicker; + private EditText inputPhoneNumber; + private EditText inputVerificationCode; + private ProgressBar newLoading; + private ProgressBar verifyLoading; + + private String currentCountryCode; + private String currentPhoneNumber; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_verification_phone, container, false); + + layoutCollect = root.findViewById(R.id.verification_phone_collect_container); + layoutVerify = root.findViewById(R.id.verification_phone_verify_container); + continueButton = root.findViewById(R.id.verification_phone_continue_button); + verifyButton = root.findViewById(R.id.verification_phone_verify_button); + editButton = root.findViewById(R.id.verification_phone_edit_button); + textVerifyParagraph = root.findViewById(R.id.verification_phone_verify_paragraph); + + countryCodePicker = root.findViewById(R.id.verification_phone_country_code); + inputPhoneNumber = root.findViewById(R.id.verification_phone_input); + inputVerificationCode = root.findViewById(R.id.verification_phone_code_input); + + newLoading = root.findViewById(R.id.verification_phone_new_progress); + verifyLoading = root.findViewById(R.id.verification_phone_verify_progress); + + Context context = getContext(); + countryCodePicker.setTypeFace(ResourcesCompat.getFont(context, R.font.inter_light)); + countryCodePicker.registerCarrierNumberEditText(inputPhoneNumber); + + continueButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + currentCountryCode = countryCodePicker.getSelectedCountryCode(); + currentPhoneNumber = Helper.getValue(inputPhoneNumber.getText()); + + if (Helper.isNullOrEmpty(currentPhoneNumber) || !countryCodePicker.isValidFullNumber()) { + Snackbar.make(getView(), R.string.please_enter_valid_phone, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + return; + } + + addPhoneNumber(); + } + }); + + verifyButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + String code = Helper.getValue(inputVerificationCode.getText()); + if (Helper.isNullOrEmpty(code)) { + Snackbar.make(getView(), R.string.please_enter_verification_code, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + return; + } + verifyPhoneNumber(code); + } + }); + + editButton.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + layoutVerify.setVisibility(View.GONE); + layoutCollect.setVisibility(View.VISIBLE); + } + }); + + return root; + } + + private void addPhoneNumber() { + PhoneNewVerifyTask task = new PhoneNewVerifyTask(currentCountryCode, currentPhoneNumber, null, newLoading, new GenericTaskHandler() { + @Override + public void beforeStart() { + Helper.setViewEnabled(continueButton, false); + Helper.setViewVisibility(continueButton, View.GONE); + } + + @Override + public void onSuccess() { + if (listener != null) { + listener.onPhoneAdded(currentCountryCode, currentPhoneNumber); + } + + Helper.setViewText(textVerifyParagraph, getString(R.string.enter_phone_verify_code, countryCodePicker.getFullNumberWithPlus())); + Helper.setViewVisibility(layoutCollect, View.GONE); + Helper.setViewVisibility(layoutVerify, View.VISIBLE); + Helper.setViewEnabled(continueButton, true); + Helper.setViewVisibility(continueButton, View.VISIBLE); + } + + @Override + public void onError(Exception error) { + if (error != null && getView() != null) { + Snackbar.make(getView(), error.getMessage(), Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + Helper.setViewEnabled(continueButton, true); + Helper.setViewVisibility(continueButton, View.VISIBLE); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void verifyPhoneNumber(String verificationCode) { + PhoneNewVerifyTask task = new PhoneNewVerifyTask(currentCountryCode, currentPhoneNumber, verificationCode, verifyLoading, new GenericTaskHandler() { + @Override + public void beforeStart() { + Helper.setViewEnabled(verifyButton, false); + Helper.setViewEnabled(editButton, false); + } + + @Override + public void onSuccess() { + if (listener != null) { + listener.onPhoneVerified(); + } + Helper.setViewEnabled(verifyButton, true); + Helper.setViewEnabled(editButton, true); + } + + @Override + public void onError(Exception error) { + if (getView() != null && error != null) { + Snackbar.make(getView(), error.getMessage(), Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + Helper.setViewEnabled(verifyButton, true); + Helper.setViewEnabled(editButton, 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..bece6300 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/verification/WalletVerificationFragment.java @@ -0,0 +1,235 @@ +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; + Lbryio.lastRemoteHash = walletSync.getHash(); + 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) && !Helper.isNullOrEmpty(Lbryio.lastRemoteHash)) { + new SyncSetTask(Lbryio.lastRemoteHash, hash, data, null).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + 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 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) && !Helper.isNullOrEmpty(Lbryio.lastRemoteHash)) { + new SyncSetTask(Lbryio.lastRemoteHash, hash, data, null).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, null, 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/InvitesFragment.java b/app/src/main/java/io/lbry/browser/ui/wallet/InvitesFragment.java new file mode 100644 index 00000000..a718f106 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/wallet/InvitesFragment.java @@ -0,0 +1,572 @@ +package io.lbry.browser.ui.wallet; + +import android.content.ClipData; +import android.content.ClipboardManager; +import android.content.Context; +import android.graphics.Color; +import android.os.AsyncTask; +import android.os.Bundle; +import android.text.Editable; +import android.text.TextWatcher; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.AdapterView; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.appcompat.widget.AppCompatSpinner; +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.textfield.TextInputEditText; +import com.google.android.material.textfield.TextInputLayout; + +import java.math.BigDecimal; +import java.util.ArrayList; +import java.util.List; + +import io.lbry.browser.BuildConfig; +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.InlineChannelSpinnerAdapter; +import io.lbry.browser.adapter.InviteeListAdapter; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.listener.WalletBalanceListener; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.WalletBalance; +import io.lbry.browser.model.lbryinc.Invitee; +import io.lbry.browser.tasks.claim.ClaimListResultHandler; +import io.lbry.browser.tasks.claim.ClaimListTask; +import io.lbry.browser.tasks.GenericTaskHandler; +import io.lbry.browser.tasks.claim.ChannelCreateUpdateTask; +import io.lbry.browser.tasks.claim.ClaimResultHandler; +import io.lbry.browser.tasks.lbryinc.LogPublishTask; +import io.lbry.browser.tasks.lbryinc.FetchInviteStatusTask; +import io.lbry.browser.tasks.lbryinc.FetchReferralCodeTask; +import io.lbry.browser.tasks.lbryinc.InviteByEmailTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; + +public class InvitesFragment extends BaseFragment implements SdkStatusListener, WalletBalanceListener { + + private static final String INVITE_LINK_FORMAT = "https://lbry.tv/$/invite/%s:%s"; + + private boolean fetchingChannels; + private View layoutAccountDriver; + private View layoutSdkInitializing; + private TextView textLearnMoreLink; + private MaterialButton buttonGetStarted; + + private View buttonCopyInviteLink; + private TextView textInviteLink; + private TextInputLayout layoutInputEmail; + private TextInputEditText inputEmail; + private MaterialButton buttonInviteByEmail; + + private RecyclerView inviteHistoryList; + private InviteeListAdapter inviteHistoryAdapter; + private InlineChannelSpinnerAdapter channelSpinnerAdapter; + private AppCompatSpinner channelSpinner; + private View progressLoadingChannels; + private View progressLoadingInviteByEmail; + private View progressLoadingStatus; + + private View inlineChannelCreator; + private TextInputEditText inlineChannelCreatorInputName; + private TextInputEditText inlineChannelCreatorInputDeposit; + private View inlineChannelCreatorInlineBalance; + private TextView inlineChannelCreatorInlineBalanceValue; + private View inlineChannelCreatorCancelLink; + private View inlineChannelCreatorProgress; + private MaterialButton inlineChannelCreatorCreateButton; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_invites, container, false); + + layoutAccountDriver = root.findViewById(R.id.invites_account_driver_container); + layoutSdkInitializing = root.findViewById(R.id.container_sdk_initializing); + textLearnMoreLink = root.findViewById(R.id.invites_account_driver_learn_more); + buttonGetStarted = root.findViewById(R.id.invites_get_started_button); + + textInviteLink = root.findViewById(R.id.invites_invite_link); + buttonCopyInviteLink = root.findViewById(R.id.invites_copy_invite_link); + layoutInputEmail = root.findViewById(R.id.invites_email_input_layout); + inputEmail = root.findViewById(R.id.invites_email_input); + buttonInviteByEmail = root.findViewById(R.id.invites_email_button); + + progressLoadingChannels = root.findViewById(R.id.invites_loading_channels_progress); + progressLoadingInviteByEmail = root.findViewById(R.id.invites_loading_invite_by_email_progress); + progressLoadingStatus = root.findViewById(R.id.invites_loading_status_progress); + + inviteHistoryList = root.findViewById(R.id.invite_history_list); + LinearLayoutManager llm = new LinearLayoutManager(getContext()); + inviteHistoryList.setLayoutManager(llm); + + channelSpinner = root.findViewById(R.id.invites_channel_spinner); + + inlineChannelCreator = root.findViewById(R.id.container_inline_channel_form_create); + inlineChannelCreatorInputName = root.findViewById(R.id.inline_channel_form_input_name); + inlineChannelCreatorInputDeposit = root.findViewById(R.id.inline_channel_form_input_deposit); + inlineChannelCreatorInlineBalance = root.findViewById(R.id.inline_channel_form_inline_balance_container); + inlineChannelCreatorInlineBalanceValue = root.findViewById(R.id.inline_channel_form_inline_balance_value); + inlineChannelCreatorProgress = root.findViewById(R.id.inline_channel_form_create_progress); + inlineChannelCreatorCancelLink = root.findViewById(R.id.inline_channel_form_cancel_link); + inlineChannelCreatorCreateButton = root.findViewById(R.id.inline_channel_form_create_button); + + initUi(); + + return root; + } + + private void initUi() { + layoutAccountDriver.setVisibility(Lbryio.isSignedIn() ? View.GONE : View.VISIBLE); + layoutSdkInitializing.setVisibility(Lbry.SDK_READY ? View.GONE : View.VISIBLE); + Helper.applyHtmlForTextView(textLearnMoreLink); + + inputEmail.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + layoutInputEmail.setHint(hasFocus ? getString(R.string.email) : + Helper.getValue(inputEmail.getText()).length() > 0 ? + getString(R.string.email) : getString(R.string.invite_email_placeholder)); + } + }); + inputEmail.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) { + Helper.setViewEnabled(buttonInviteByEmail, charSequence.length() > 0); + } + + @Override + public void afterTextChanged(Editable editable) { + + } + }); + buttonInviteByEmail.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + String email = Helper.getValue(inputEmail.getText()); + if (email.indexOf("@") == -1) { + showError(getString(R.string.provide_valid_email)); + return; + } + + InviteByEmailTask task = new InviteByEmailTask(email, progressLoadingInviteByEmail, new GenericTaskHandler() { + @Override + public void beforeStart() { + Helper.setViewEnabled(buttonInviteByEmail, false); + } + + @Override + public void onSuccess() { + Snackbar.make(getView(), getString(R.string.invite_sent_to, email), Snackbar.LENGTH_LONG).show(); + Helper.setViewText(inputEmail, null); + Helper.setViewEnabled(buttonInviteByEmail, true); + fetchInviteStatus(); + } + + @Override + public void onError(Exception error) { + showError(error.getMessage()); + Helper.setViewEnabled(buttonInviteByEmail, true); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + }); + + buttonGetStarted.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).simpleSignIn(); + } + } + }); + + channelSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() { + @Override + public void onItemSelected(AdapterView adapterView, View view, int position, long l) { + Object item = adapterView.getItemAtPosition(position); + if (item instanceof Claim) { + Claim claim = (Claim) item; + if (claim.isPlaceholder()) { + if (!fetchingChannels) { + showInlineChannelCreator(); + } + } else { + hideInlineChannelCreator(); + // build invite link + updateInviteLink(claim); + } + } + } + + @Override + public void onNothingSelected(AdapterView adapterView) { + + } + }); + + textInviteLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + copyInviteLink(); + } + }); + buttonCopyInviteLink.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + copyInviteLink(); + } + }); + + setupInlineChannelCreator( + inlineChannelCreator, + inlineChannelCreatorInputName, + inlineChannelCreatorInputDeposit, + inlineChannelCreatorInlineBalance, + inlineChannelCreatorInlineBalanceValue, + inlineChannelCreatorCancelLink, + inlineChannelCreatorCreateButton, + inlineChannelCreatorProgress + ); + } + + private void updateInviteLink(Claim claim) { + LbryUri canonical = LbryUri.tryParse(claim.getCanonicalUrl()); + String link = String.format(INVITE_LINK_FORMAT, + canonical != null ? String.format("@%s", canonical.getChannelName()) : claim.getName(), + canonical != null ? canonical.getChannelClaimId() : claim.getClaimId()); + textInviteLink.setText(link); + } + private void copyInviteLink() { + Context context = getContext(); + if (context != null && textInviteLink != null) { + ClipboardManager clipboard = (ClipboardManager) context.getSystemService(Context.CLIPBOARD_SERVICE); + ClipData data = ClipData.newPlainText("inviteLink", textInviteLink.getText()); + clipboard.setPrimaryClip(data); + } + Snackbar.make(getView(), R.string.invite_link_copied, Snackbar.LENGTH_SHORT).show(); + } + + private void updateChannelList(List channels) { + if (channelSpinnerAdapter == null) { + Context context = getContext(); + channelSpinnerAdapter = new InlineChannelSpinnerAdapter(context, R.layout.spinner_item_channel, new ArrayList<>(channels)); + channelSpinnerAdapter.addPlaceholder(false); + channelSpinnerAdapter.notifyDataSetChanged(); + } else { + channelSpinnerAdapter.clear(); + channelSpinnerAdapter.addAll(channels); + channelSpinnerAdapter.addPlaceholder(false); + channelSpinnerAdapter.notifyDataSetChanged(); + } + + if (channelSpinner != null) { + channelSpinner.setAdapter(channelSpinnerAdapter); + } + + if (channelSpinnerAdapter.getCount() > 1) { + channelSpinner.setSelection(1); + } + } + + public void onResume() { + super.onResume(); + layoutAccountDriver.setVisibility(Lbryio.isSignedIn() ? View.GONE : View.VISIBLE); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Invites", "Invites"); + } + + fetchInviteStatus(); + if (!Lbry.SDK_READY) { + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.addSdkStatusListener(this); + activity.addWalletBalanceListener(this); + } + } else { + onSdkReady(); + } + } + + public void onSdkReady() { + Helper.setViewVisibility(layoutSdkInitializing, View.GONE); + fetchChannels(); + } + + public void onStart() { + super.onStart(); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.setWunderbarValue(null); + activity.hideFloatingWalletBalance(); + } + } + + public void clearInputFocus() { + inputEmail.clearFocus(); + inlineChannelCreatorInputName.clearFocus(); + inlineChannelCreatorInputDeposit.clearFocus(); + } + + public void onStop() { + clearInputFocus(); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.removeSdkStatusListener(this); + activity.removeWalletBalanceListener(this); + activity.showFloatingWalletBalance(); + } + super.onStop(); + } + + private void showInlineChannelCreator() { + Helper.setViewVisibility(inlineChannelCreator, View.VISIBLE); + } + private void hideInlineChannelCreator() { + Helper.setViewVisibility(inlineChannelCreator, View.GONE); + } + + private void fetchDefaultInviteLink() { + FetchReferralCodeTask task = new FetchReferralCodeTask(null, new FetchReferralCodeTask.FetchReferralCodeHandler() { + @Override + public void onSuccess(String referralCode) { + String previousLink = Helper.getValue(textInviteLink.getText()); + if (Helper.isNullOrEmpty(previousLink)) { + Helper.setViewText(textInviteLink, String.format("https://lbry.tv/$/invite/%s", referralCode)); + } + } + + @Override + public void onError(Exception error) { + // pass + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void disableChannelSpinner() { + Helper.setViewEnabled(channelSpinner, false); + hideInlineChannelCreator(); + } + private void enableChannelSpinner() { + Helper.setViewEnabled(channelSpinner, true); + Claim selectedClaim = (Claim) channelSpinner.getSelectedItem(); + if (selectedClaim != null) { + if (selectedClaim.isPlaceholder()) { + showInlineChannelCreator(); + } else { + hideInlineChannelCreator(); + } + } + } + + private void fetchChannels() { + if (Lbry.ownChannels != null && Lbry.ownChannels.size() > 0) { + updateChannelList(Lbry.ownChannels); + return; + } + + fetchingChannels = true; + disableChannelSpinner(); + ClaimListTask task = new ClaimListTask(Claim.TYPE_CHANNEL, progressLoadingChannels, new ClaimListResultHandler() { + @Override + public void onSuccess(List claims) { + Lbry.ownChannels = new ArrayList<>(claims); + updateChannelList(Lbry.ownChannels); + if (Lbry.ownChannels == null || Lbry.ownChannels.size() == 0) { + fetchDefaultInviteLink(); + } + enableChannelSpinner(); + fetchingChannels = false; + } + + @Override + public void onError(Exception error) { + fetchDefaultInviteLink(); + enableChannelSpinner(); + fetchingChannels = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void fetchInviteStatus() { + FetchInviteStatusTask task = new FetchInviteStatusTask(progressLoadingStatus, new FetchInviteStatusTask.FetchInviteStatusHandler() { + @Override + public void onSuccess(List invitees) { + if (inviteHistoryAdapter == null) { + inviteHistoryAdapter = new InviteeListAdapter(invitees, getContext()); + inviteHistoryAdapter.addHeader(); + } else { + inviteHistoryAdapter.addInvitees(invitees); + } + if (inviteHistoryList != null) { + inviteHistoryList.setAdapter(inviteHistoryAdapter); + } + Helper.setViewVisibility(inviteHistoryList, + inviteHistoryAdapter == null || inviteHistoryAdapter.getItemCount() < 2 ? View.GONE : View.VISIBLE + ); + } + + @Override + public void onError(Exception error) { + Helper.setViewVisibility(inviteHistoryList, + inviteHistoryAdapter == null || inviteHistoryAdapter.getItemCount() < 2 ? View.GONE : View.VISIBLE + ); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void setupInlineChannelCreator( + View container, + TextInputEditText inputChannelName, + TextInputEditText inputDeposit, + View inlineBalanceView, + TextView inlineBalanceValue, + View linkCancel, + MaterialButton buttonCreate, + View progressView) { + inputDeposit.setOnFocusChangeListener(new View.OnFocusChangeListener() { + @Override + public void onFocusChange(View view, boolean hasFocus) { + Helper.setViewVisibility(inlineBalanceView, hasFocus ? View.VISIBLE : View.INVISIBLE); + } + }); + + linkCancel.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + Helper.setViewText(inputChannelName, null); + Helper.setViewText(inputDeposit, null); + Helper.setViewVisibility(container, View.GONE); + } + }); + + buttonCreate.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + // validate deposit and channel name + String channelNameString = Helper.normalizeChannelName(Helper.getValue(inputChannelName.getText())); + Claim claimToSave = new Claim(); + claimToSave.setName(channelNameString); + String channelName = claimToSave.getName().startsWith("@") ? claimToSave.getName().substring(1) : claimToSave.getName(); + String depositString = Helper.getValue(inputDeposit.getText()); + if ("@".equals(channelName) || Helper.isNullOrEmpty(channelName)) { + showError(getString(R.string.please_enter_channel_name)); + return; + } + if (!LbryUri.isNameValid(channelName)) { + showError(getString(R.string.channel_name_invalid_characters)); + return; + } + if (Helper.channelExists(channelName)) { + showError(getString(R.string.channel_name_already_created)); + return; + } + + double depositAmount = 0; + try { + depositAmount = Double.valueOf(depositString); + } catch (NumberFormatException ex) { + // pass + showError(getString(R.string.please_enter_valid_deposit)); + return; + } + if (depositAmount == 0) { + String error = getResources().getQuantityString(R.plurals.min_deposit_required, depositAmount == 1 ? 1 : 2, String.valueOf(Helper.MIN_DEPOSIT)); + showError(error); + return; + } + if (Lbry.walletBalance == null || Lbry.walletBalance.getAvailable().doubleValue() < depositAmount) { + showError(getString(R.string.deposit_more_than_balance)); + return; + } + + ChannelCreateUpdateTask task = new ChannelCreateUpdateTask( + claimToSave, new BigDecimal(depositString), false, progressView, new ClaimResultHandler() { + @Override + public void beforeStart() { + Helper.setViewEnabled(inputChannelName, false); + Helper.setViewEnabled(inputDeposit, false); + Helper.setViewEnabled(buttonCreate, false); + Helper.setViewEnabled(linkCancel, false); + } + + @Override + public void onSuccess(Claim claimResult) { + if (!BuildConfig.DEBUG) { + LogPublishTask logPublishTask = new LogPublishTask(claimResult); + logPublishTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + // channel created + Bundle bundle = new Bundle(); + bundle.putString("claim_id", claimResult.getClaimId()); + bundle.putString("claim_name", claimResult.getName()); + LbryAnalytics.logEvent(LbryAnalytics.EVENT_CHANNEL_CREATE, bundle); + + // add the claim to the channel list and set it as the selected item + channelSpinnerAdapter.add(claimResult); + channelSpinner.setSelection(channelSpinnerAdapter.getCount() - 1); + + Helper.setViewEnabled(inputChannelName, true); + Helper.setViewEnabled(inputDeposit, true); + Helper.setViewEnabled(buttonCreate, true); + Helper.setViewEnabled(linkCancel, true); + } + + @Override + public void onError(Exception error) { + Helper.setViewEnabled(inputChannelName, true); + Helper.setViewEnabled(inputDeposit, true); + Helper.setViewEnabled(buttonCreate, true); + Helper.setViewEnabled(linkCancel, true); + showError(error.getMessage()); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + }); + + Helper.setViewText(inlineBalanceValue, Helper.shortCurrencyFormat(Lbry.walletBalance.getAvailable().doubleValue())); + } + + @Override + public void onWalletBalanceUpdated(WalletBalance walletBalance) { + if (walletBalance != null && inlineChannelCreatorInlineBalanceValue != null) { + inlineChannelCreatorInlineBalanceValue.setText(Helper.shortCurrencyFormat(walletBalance.getAvailable().doubleValue())); + } + } + + private void showError(String message) { + Context context = getContext(); + if (context != null) { + Snackbar.make(getView(), message, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + } + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/wallet/RewardsFragment.java b/app/src/main/java/io/lbry/browser/ui/wallet/RewardsFragment.java new file mode 100644 index 00000000..eee6ebb0 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/wallet/RewardsFragment.java @@ -0,0 +1,298 @@ +package io.lbry.browser.ui.wallet; + +import android.content.Context; +import android.content.SharedPreferences; +import android.graphics.Color; +import android.graphics.Typeface; +import android.os.AsyncTask; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.EditText; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.preference.PreferenceManager; +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.text.DecimalFormat; +import java.util.List; + +import io.lbry.browser.MainActivity; +import io.lbry.browser.R; +import io.lbry.browser.adapter.RewardListAdapter; +import io.lbry.browser.listener.SdkStatusListener; +import io.lbry.browser.model.lbryinc.Reward; +import io.lbry.browser.tasks.lbryinc.ClaimRewardTask; +import io.lbry.browser.tasks.lbryinc.FetchRewardsTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.Lbryio; + +public class RewardsFragment extends BaseFragment implements RewardListAdapter.RewardClickListener, SdkStatusListener { + + private boolean rewardClaimInProgress; + private View layoutAccountDriver; + private View layoutSdkInitializing; + private View linkNotInterested; + private TextView textAccountDriverTitle; + private TextView textFreeCreditsWorth; + private TextView textLearnMoreLink; + private MaterialButton buttonGetStarted; + + private ProgressBar rewardsLoading; + private RewardListAdapter adapter; + private RecyclerView rewardList; + private TextView linkFilterUnclaimed; + private TextView linkFilterAll; + + public View onCreateView(@NonNull LayoutInflater inflater, + ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_rewards, container, false); + + layoutAccountDriver = root.findViewById(R.id.rewards_account_driver_container); + layoutSdkInitializing = root.findViewById(R.id.container_sdk_initializing); + linkNotInterested = root.findViewById(R.id.rewards_not_interested_link); + textAccountDriverTitle = root.findViewById(R.id.rewards_account_driver_title); + textFreeCreditsWorth = root.findViewById(R.id.rewards_account_driver_credits_worth); + textLearnMoreLink = root.findViewById(R.id.rewards_account_driver_learn_more); + buttonGetStarted = root.findViewById(R.id.rewards_get_started_button); + + linkFilterUnclaimed = root.findViewById(R.id.rewards_filter_link_unclaimed); + linkFilterAll = root.findViewById(R.id.rewards_filter_link_all); + rewardList = root.findViewById(R.id.rewards_list); + rewardsLoading = root.findViewById(R.id.rewards_list_loading); + + Context context = getContext(); + LinearLayoutManager llm = new LinearLayoutManager(context); + rewardList.setLayoutManager(llm); + adapter = new RewardListAdapter(Lbryio.allRewards, context); + adapter.setClickListener(this); + adapter.setDisplayMode(RewardListAdapter.DISPLAY_MODE_UNCLAIMED); + rewardList.setAdapter(adapter); + + initUi(); + + return root; + } + + + public void onResume() { + super.onResume(); + checkRewardsStatus(); + fetchRewards(); + + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Rewards", "Rewards"); + } + + if (!Lbry.SDK_READY) { + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.addSdkStatusListener(this); + } + } else { + onSdkReady(); + } + } + + public void onSdkReady() { + Helper.setViewVisibility(layoutSdkInitializing, View.GONE); + } + + public void onStart() { + super.onStart(); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.setWunderbarValue(null); + activity.hideFloatingWalletBalance(); + } + } + + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.removeSdkStatusListener(this); + activity.showFloatingWalletBalance(); + } + super.onStop(); + } + + private void fetchRewards() { + Helper.setViewVisibility(rewardList, View.INVISIBLE); + FetchRewardsTask task = new FetchRewardsTask(rewardsLoading, new FetchRewardsTask.FetchRewardsHandler() { + @Override + public void onSuccess(List rewards) { + Lbryio.updateRewardsLists(rewards); + updateUnclaimedRewardsValue(); + + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).showFloatingUnclaimedRewards(); + } + + if (adapter == null) { + adapter = new RewardListAdapter(rewards, getContext()); + adapter.setClickListener(RewardsFragment.this); + adapter.setDisplayMode(RewardListAdapter.DISPLAY_MODE_UNCLAIMED); + rewardList.setAdapter(adapter); + } else { + adapter.setRewards(rewards); + } + Helper.setViewVisibility(rewardList, View.VISIBLE); + } + + @Override + public void onError(Exception error) { + // pass + Helper.setViewVisibility(rewardList, View.VISIBLE); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void initUi() { + layoutSdkInitializing.setVisibility(Lbry.SDK_READY ? View.GONE : View.VISIBLE); + + linkNotInterested.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + + + Context context = getContext(); + + if (context instanceof MainActivity) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(context); + sp.edit().putBoolean(MainActivity.PREFERENCE_KEY_INTERNAL_REWARDS_NOT_INTERESTED, true).apply(); + + MainActivity activity = (MainActivity) context; + activity.hideFloatingRewardsValue(); + activity.onBackPressed(); + } + } + }); + buttonGetStarted.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).rewardsSignIn(); + } + } + }); + + linkFilterAll.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + linkFilterUnclaimed.setTypeface(null, Typeface.NORMAL); + linkFilterAll.setTypeface(null, Typeface.BOLD); + adapter.setDisplayMode(RewardListAdapter.DISPLAY_MODE_ALL); + if (adapter.getItemCount() == 1) { + fetchRewards(); + } + } + }); + linkFilterUnclaimed.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + linkFilterUnclaimed.setTypeface(null, Typeface.BOLD); + linkFilterAll.setTypeface(null, Typeface.NORMAL); + adapter.setDisplayMode(RewardListAdapter.DISPLAY_MODE_UNCLAIMED); + if (adapter.getItemCount() == 1) { + fetchRewards(); + } + } + }); + + updateUnclaimedRewardsValue(); + layoutAccountDriver.setVisibility(Lbryio.currentUser != null && Lbryio.currentUser.isRewardApproved() ? View.GONE : View.VISIBLE); + Helper.applyHtmlForTextView(textLearnMoreLink); + } + + private void checkRewardsStatus() { + Helper.setViewVisibility(layoutAccountDriver, Lbryio.currentUser != null && Lbryio.currentUser.isRewardApproved() ? View.GONE : View.VISIBLE); + } + + public void updateUnclaimedRewardsValue() { + try { + String accountDriverTitle = getResources().getQuantityString( + R.plurals.available_credits, + Lbryio.totalUnclaimedRewardAmount == 1 ? 1 : 2, + Helper.shortCurrencyFormat(Lbryio.totalUnclaimedRewardAmount)); + double unclaimedRewardAmountUsd = Lbryio.totalUnclaimedRewardAmount * Lbryio.LBCUSDRate; + Helper.setViewText(textAccountDriverTitle, accountDriverTitle); + Helper.setViewText(textFreeCreditsWorth, getString(R.string.free_credits_worth, Helper.SIMPLE_CURRENCY_FORMAT.format(unclaimedRewardAmountUsd))); + } catch (IllegalStateException ex) { + // pass + } + + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).updateRewardsUsdVale(); + } + } + + @Override + public void onRewardClicked(Reward reward, View loadingView) { + if (rewardClaimInProgress || reward.isCustom()) { + return; + } + claimReward(reward.getRewardType(), null, null, null, loadingView); + } + + @Override + public void onCustomClaimButtonClicked(String code, EditText inputCustomCode, MaterialButton buttonClaim, View loadingView) { + if (rewardClaimInProgress) { + return; + } + claimReward(Reward.TYPE_REWARD_CODE, code, inputCustomCode, buttonClaim, loadingView); + } + + private void claimReward(String type, String code, EditText inputClaimCode, MaterialButton buttonClaim, View loadingView) { + rewardClaimInProgress = true; + Helper.setViewEnabled(buttonClaim, false); + Helper.setViewEnabled(inputClaimCode, false); + ClaimRewardTask task = new ClaimRewardTask(type, code, loadingView, getContext(), new ClaimRewardTask.ClaimRewardHandler() { + @Override + public void onSuccess(double amountClaimed, String message) { + if (Helper.isNullOrEmpty(message)) { + message = getResources().getQuantityString( + R.plurals.claim_reward_message, + amountClaimed == 1 ? 1 : 2, + new DecimalFormat(Helper.LBC_CURRENCY_FORMAT_PATTERN).format(amountClaimed)); + } + View view = getView(); + if (view != null) { + Snackbar.make(view, message, Snackbar.LENGTH_LONG).show(); + } + Helper.setViewEnabled(buttonClaim, true); + Helper.setViewEnabled(inputClaimCode, true); + rewardClaimInProgress = false; + + fetchRewards(); + } + + @Override + public void onError(Exception error) { + Snackbar.make(getView(), error.getMessage(), Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + Helper.setViewEnabled(buttonClaim, true); + Helper.setViewEnabled(inputClaimCode, true); + rewardClaimInProgress = false; + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } +} diff --git a/app/src/main/java/io/lbry/browser/ui/wallet/TransactionHistoryFragment.java b/app/src/main/java/io/lbry/browser/ui/wallet/TransactionHistoryFragment.java new file mode 100644 index 00000000..d9b70c7c --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/wallet/TransactionHistoryFragment.java @@ -0,0 +1,168 @@ +package io.lbry.browser.ui.wallet; + +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.ProgressBar; + +import androidx.annotation.NonNull; +import androidx.appcompat.app.ActionBar; +import androidx.recyclerview.widget.DividerItemDecoration; +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.TransactionListAdapter; +import io.lbry.browser.model.Transaction; +import io.lbry.browser.tasks.wallet.TransactionListTask; +import io.lbry.browser.ui.BaseFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; + +public class TransactionHistoryFragment extends BaseFragment implements TransactionListAdapter.TransactionClickListener { + + private static final int TRANSACTION_PAGE_LIMIT = 50; + private boolean transactionsHaveReachedEnd; + private boolean transactionsLoading; + private ProgressBar loading; + private RecyclerView transactionList; + private TransactionListAdapter adapter; + private View noTransactionsView; + private int currentTransactionPage; + + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { + View root = inflater.inflate(R.layout.fragment_transaction_history, container, false); + + loading = root.findViewById(R.id.transaction_history_loading); + transactionList = root.findViewById(R.id.transaction_history_list); + noTransactionsView = root.findViewById(R.id.transaction_history_no_transactions); + + Context context = getContext(); + LinearLayoutManager llm = new LinearLayoutManager(context); + transactionList.setLayoutManager(llm); + transactionList.addItemDecoration(new DividerItemDecoration(context, DividerItemDecoration.VERTICAL)); + + transactionList.addOnScrollListener(new RecyclerView.OnScrollListener() { + @Override + public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { + if (transactionsLoading) { + 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 (!transactionsHaveReachedEnd) { + // load more + currentTransactionPage++; + loadTransactions(); + } + } + } + } + }); + + return root; + } + + @Override + public void onResume() { + super.onResume(); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Transaction History", "TransactionHistory"); + } + + if (adapter != null && adapter.getItemCount() > 0 && transactionList != null) { + transactionList.setAdapter(adapter); + } + loadTransactions(); + } + + private void checkNoTransactions() { + Helper.setViewVisibility(noTransactionsView, adapter == null || adapter.getItemCount() == 0 ? View.VISIBLE : View.GONE); + } + + private void loadTransactions() { + currentTransactionPage = currentTransactionPage == 0 ? 1 : currentTransactionPage; + transactionsLoading = true; + TransactionListTask task = new TransactionListTask(currentTransactionPage, TRANSACTION_PAGE_LIMIT, loading, new TransactionListTask.TransactionListHandler() { + @Override + public void onSuccess(List transactions, boolean hasReachedEnd) { + transactionsLoading = false; + transactionsHaveReachedEnd = hasReachedEnd; + if (adapter == null) { + adapter = new TransactionListAdapter(transactions, getContext()); + adapter.setListener(TransactionHistoryFragment.this); + if (transactionList != null) { + transactionList.setAdapter(adapter); + } + } else { + adapter.addTransactions(transactions); + } + checkNoTransactions(); + } + + @Override + public void onError(Exception error) { + transactionsLoading = false; + checkNoTransactions(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + @Override + public void onStart() { + super.onStart(); + MainActivity activity = (MainActivity) getContext(); + if (activity != null) { + activity.hideSearchBar(); + activity.hideFloatingWalletBalance(); + activity.showNavigationBackIcon(); + activity.lockDrawer(); + + ActionBar actionBar = activity.getSupportActionBar(); + if (actionBar != null) { + actionBar.setTitle(R.string.transaction_history); + } + } + } + + @Override + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.restoreToggle(); + activity.showFloatingWalletBalance(); + } + super.onStop(); + } + + public void onTransactionClicked(Transaction transaction) { + // Don't do anything? Or open the transaction in a browser? + } + public void onClaimUrlClicked(LbryUri uri) { + Context context = getContext(); + if (uri != null && context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (uri.isChannel()) { + activity.openChannelUrl(uri.toString()); + } else { + activity.openFileUrl(uri.toString()); + } + } + } +} 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..7ae5ec9e --- /dev/null +++ b/app/src/main/java/io/lbry/browser/ui/wallet/WalletFragment.java @@ -0,0 +1,537 @@ +package io.lbry.browser.ui.wallet; + +import android.content.ClipData; +import android.content.ClipboardManager; +import android.content.Context; +import android.content.SharedPreferences; +import android.graphics.Color; +import android.os.AsyncTask; +import android.os.Bundle; +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.content.ContextCompat; +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.math.BigDecimal; +import java.text.DecimalFormat; +import java.text.DecimalFormatSymbols; +import java.util.List; +import java.util.Locale; + +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.NavMenuItem; +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.ui.publish.PublishFragment; +import io.lbry.browser.utils.Helper; +import io.lbry.browser.utils.Lbry; +import io.lbry.browser.utils.LbryAnalytics; +import io.lbry.browser.utils.LbryUri; +import io.lbry.browser.utils.Lbryio; + +public class WalletFragment extends BaseFragment implements SdkStatusListener, WalletBalanceListener { + + 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.container_sdk_initializing); + 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()); + recentTransactionsAdapter.setListener(new TransactionListAdapter.TransactionClickListener() { + @Override + public void onTransactionClicked(Transaction transaction) { + + } + + @Override + public void onClaimUrlClicked(LbryUri uri) { + Context context = getContext(); + if (uri != null && context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + if (uri.isChannel()) { + activity.openChannelUrl(uri.toString()); + } else { + activity.openFileUrl(uri.toString()); + } + } + } + }); + 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(Color.RED).setTextColor(Color.WHITE).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(Color.RED).setTextColor(Color.WHITE).show(); + return false; + } + } catch (NumberFormatException ex) { + // pass + Snackbar.make(getView(), R.string.invalid_amount, Snackbar.LENGTH_LONG). + setBackgroundTint(Color.RED).setTextColor(Color.WHITE).show(); + return false; + } + } + + return true; + } + + @SuppressWarnings("ClickableViewAccessibility") + private void initUi() { + onWalletBalanceUpdated(Lbry.walletBalance); + + Helper.applyHtmlForTextView(textConvertCredits); + Helper.applyHtmlForTextView(textConvertCreditsBittrex); + Helper.applyHtmlForTextView(textWhatSyncMeans); + Helper.applyHtmlForTextView(linkManualBackup); + Helper.applyHtmlForTextView(linkSyncFAQ); + + Context context = getContext(); + LinearLayoutManager llm = new LinearLayoutManager(context); + recentTransactionsList.setLayoutManager(llm); + DividerItemDecoration itemDecoration = new DividerItemDecoration(context, DividerItemDecoration.VERTICAL); + itemDecoration.setDrawable(ContextCompat.getDrawable(context, R.drawable.thin_divider)); + recentTransactionsList.addItemDecoration(itemDecoration); + + textEarnMoreTips.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + if (context instanceof MainActivity) { + ((MainActivity) context).openFragment(PublishFragment.class, true, NavMenuItem.ID_ITEM_NEW_PUBLISH); + } + } + }); + + buttonSignUp.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).walletSyncSignIn(); + } + } + }); + 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); + } + }); + + linkViewAll.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View view) { + Context context = getContext(); + if (context instanceof MainActivity) { + ((MainActivity) context).openFragment(TransactionHistoryFragment.class, true, NavMenuItem.ID_ITEM_WALLET); + } + } + }); + + 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() { + 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 = new DecimalFormat(Helper.SDK_AMOUNT_FORMAT, new DecimalFormatSymbols(Locale.US)). + format(new BigDecimal(amountString).doubleValue()); + + disableSendControls(); + WalletSendTask task = new WalletSendTask(recipientAddress, amount, walletSendProgress, new WalletSendTask.WalletSendHandler() { + @Override + public void onSuccess() { + double sentAmount = Double.valueOf(amount); + String message = getResources().getQuantityString( + R.plurals.you_sent_credits, sentAmount == 1.0 ? 1 : 2, + new DecimalFormat("#,###.##").format(sentAmount)); + 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(Color.RED).setTextColor(Color.WHITE).show(); + enableSendControls(); + } + }); + task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + + private void checkReceiveAddress() { + Context context = getContext(); + String receiveAddress = null; + if (context != null) { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(context); + 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 boolean hasSkippedAccount() { + SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getContext()); + return sp.getBoolean(MainActivity.PREFERENCE_KEY_INTERNAL_SKIP_WALLET_ACCOUNT, false); + } + + public void onResume() { + super.onResume(); + Context context = getContext(); + Helper.setWunderbarValue(null, context); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + LbryAnalytics.setCurrentScreen(activity, "Wallet", "Wallet"); + } + + Helper.setViewVisibility(layoutAccountRecommended, hasSkippedAccount() || Lbryio.isSignedIn() ? View.GONE : View.VISIBLE); + if (!Lbry.SDK_READY) { + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.addSdkStatusListener(this); + } + + checkReceiveAddress(); + } else { + onSdkReady(); + } + } + + public void onPause() { + hasFetchedRecentTransactions = false; + super.onPause(); + } + public void onStart() { + super.onStart(); + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.setWunderbarValue(null); + activity.hideFloatingWalletBalance(); + } + } + + public void onStop() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.removeWalletBalanceListener(this); + activity.showFloatingWalletBalance(); + } + super.onStop(); + } + + public void onSdkReady() { + Context context = getContext(); + if (context instanceof MainActivity) { + MainActivity activity = (MainActivity) context; + activity.syncWalletAndLoadPreferences(); + activity.addWalletBalanceListener(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); + } + + 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; + + String formattedBalance = Helper.SIMPLE_CURRENCY_FORMAT.format(balance); + Helper.setViewText(textWalletBalance, balance > 0 && formattedBalance.equals("0") ? Helper.FULL_LBC_CURRENCY_FORMAT.format(balance) : formattedBalance); + 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", Helper.SIMPLE_CURRENCY_FORMAT.format(usdBalance))); + Helper.setViewText(textTipsBalanceUSD, String.format("≈$%s", Helper.SIMPLE_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/ExoplayerAudioRenderer.java b/app/src/main/java/io/lbry/browser/utils/ExoplayerAudioRenderer.java new file mode 100644 index 00000000..832db7f2 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/ExoplayerAudioRenderer.java @@ -0,0 +1,53 @@ +package io.lbry.browser.utils; + +import android.content.Context; +import android.os.Handler; + +import androidx.annotation.Nullable; + +import com.google.android.exoplayer2.DefaultRenderersFactory; +import com.google.android.exoplayer2.Renderer; +import com.google.android.exoplayer2.audio.AudioProcessor; +import com.google.android.exoplayer2.audio.AudioRendererEventListener; +import com.google.android.exoplayer2.audio.TeeAudioProcessor; +import com.google.android.exoplayer2.drm.DrmSessionManager; +import com.google.android.exoplayer2.drm.FrameworkMediaCrypto; +import com.google.android.exoplayer2.mediacodec.MediaCodecSelector; + +import java.util.ArrayList; + +public class ExoplayerAudioRenderer extends DefaultRenderersFactory { + + private TeeAudioProcessor.AudioBufferSink audioBufferSink; + + public ExoplayerAudioRenderer(Context context, TeeAudioProcessor.AudioBufferSink audioBufferSink) { + super(context); + this.audioBufferSink = audioBufferSink; + } + + @Override + protected void buildAudioRenderers( + Context context, + int extensionRendererMode, + MediaCodecSelector mediaCodecSelector, + @Nullable DrmSessionManager drmSessionManager, + boolean playClearSamplesWithoutKeys, + boolean enableDecoderFallback, + AudioProcessor[] audioProcessors, + Handler eventHandler, + AudioRendererEventListener eventListener, + ArrayList out) { + AudioProcessor[] audioProcessorList = { new TeeAudioProcessor(audioBufferSink) }; + super.buildAudioRenderers( + context, + extensionRendererMode, + mediaCodecSelector, + drmSessionManager, + playClearSamplesWithoutKeys, + enableDecoderFallback, + audioProcessorList, + eventHandler, + eventListener, + out); + } +} 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..108d287c --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/Helper.java @@ -0,0 +1,756 @@ +package io.lbry.browser.utils; + +import android.annotation.SuppressLint; +import android.content.BroadcastReceiver; +import android.content.ContentUris; +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.net.Uri; +import android.os.AsyncTask; +import android.os.Build; +import android.os.Environment; +import android.provider.DocumentsContract; +import android.provider.MediaStore; +import android.text.method.LinkMovementMethod; +import android.view.ContextMenu; +import android.view.View; +import android.widget.LinearLayout; +import android.widget.ProgressBar; +import android.widget.TextView; + +import androidx.core.content.ContextCompat; +import androidx.core.text.HtmlCompat; +import androidx.recyclerview.widget.GridLayoutManager; +import androidx.recyclerview.widget.LinearLayoutManager; +import androidx.recyclerview.widget.RecyclerView; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.io.Closeable; +import java.io.File; +import java.io.IOException; +import java.text.DecimalFormat; +import java.text.SimpleDateFormat; +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.MainActivity; +import io.lbry.browser.data.DatabaseHelper; +import io.lbry.browser.dialog.ContentFromDialogFragment; +import io.lbry.browser.dialog.ContentSortDialogFragment; +import io.lbry.browser.model.Claim; +import io.lbry.browser.model.LbryFile; +import io.lbry.browser.model.Tag; +import io.lbry.browser.model.UrlSuggestion; +import io.lbry.browser.model.ViewHistory; +import io.lbry.browser.tasks.localdata.SaveUrlHistoryTask; +import io.lbry.browser.tasks.localdata.SaveViewHistoryTask; +import okhttp3.MediaType; + +public final class Helper { + public static final String UNKNOWN = "Unknown"; + public static final String METHOD_GET = "GET"; + public static final String METHOD_POST = "POST"; + public static final String ISO_DATE_FORMAT_PATTERN = "yyyy-MM-dd'T'HH:mm:ss.SSS"; + public static final String SDK_AMOUNT_FORMAT = "0.0#######"; + 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 int CONTENT_PAGE_SIZE = 25; + public static final double MIN_DEPOSIT = 0.001; + public static final String LBC_CURRENCY_FORMAT_PATTERN = "#,###.##"; + public static final String FILE_SIZE_FORMAT_PATTERN = "#,###.#"; + public static final DecimalFormat LBC_CURRENCY_FORMAT = new DecimalFormat(LBC_CURRENCY_FORMAT_PATTERN); + public static final DecimalFormat FULL_LBC_CURRENCY_FORMAT = new DecimalFormat("#,###.########"); + public static final DecimalFormat SIMPLE_CURRENCY_FORMAT = new DecimalFormat("#,##0.00"); + public static final SimpleDateFormat FILESTAMP_FORMAT = new SimpleDateFormat("yyyyMMdd_HHmmss"); + public static final String EXPLORER_TX_PREFIX = "https://explorer.lbry.com/tx"; + + public static final List PLAYBACK_SPEEDS = Arrays.asList(0.25, 0.5, 0.75, 1.0, 1.25, 1.5, 1.75, 2.0); + + public static boolean isNull(String value) { + return value == null; + } + public static boolean isNullOrEmpty(String value) { + return value == null || value.trim().length() == 0; + } + + public static void buildPlaybackSpeedMenu(ContextMenu menu) { + int order = 0; + DecimalFormat formatter = new DecimalFormat("0.##"); + for (Double speed : PLAYBACK_SPEEDS) { + menu.add(0, Double.valueOf(speed * 100).intValue(), ++order, String.format("%sx", formatter.format(speed))); + } + } + + 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 setViewProgress(ProgressBar progressBar, int progress) { + if (progressBar != null) { + progressBar.setProgress(progress); + } + } + + 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 Double parseDouble(Object value, double defaultValue) { + try { + return Double.parseDouble(String.valueOf(value)); + } 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 String[] formatBytesParts(long bytes, boolean showTB) { + DecimalFormat formatter = new DecimalFormat(FILE_SIZE_FORMAT_PATTERN); + if (bytes < 1048576) { + // less than 1MB + return new String[] { formatter.format(bytes / 1024.0), "KB" }; + } + if (bytes < 1073741824) { + // less than 1GB + return new String[] { formatter.format(bytes / (1024.0 * 1024.0)), "MB" }; + } + if (showTB) { + if (bytes < (1073741824L * 1024L)) { + return new String[] { formatter.format(bytes / (1024.0 * 1024.0 * 1024.0)), "GB" }; + } + return new String[] { formatter.format(bytes / (1024.0 * 1024.0 * 1024.0 * 1024.0)), "TB" }; + } + return new String[] { formatter.format(bytes / (1024.0 * 1024.0 * 1024.0)), "GB" }; + } + + public static String formatBytes(long bytes, boolean showTB) { + DecimalFormat formatter = new DecimalFormat(FILE_SIZE_FORMAT_PATTERN); + if (bytes < 1048576) { + // less than 1MB + return String.format("%sKB", formatter.format(bytes / 1024.0)); + } + if (bytes < 1073741824) { + // less than 1GB + return String.format("%sMB", formatter.format(bytes / (1024.0 * 1024.0))); + } + if (showTB) { + if (bytes < (1073741824L * 1024L)) { + return String.format("%sGB", formatter.format(bytes / (1024.0 * 1024.0 * 1024.0))); + } + return String.format("%sTB", formatter.format(bytes / (1024.0 * 1024.0 * 1024.0 * 1024.0))); + } + return String.format("%sGB", formatter.format(bytes / (1024.0 * 1024.0 * 1024.0))); + } + + 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) { + DecimalFormat format = new DecimalFormat("#,###.#"); + if (value > 1000000000.00) { + return String.format("%sB", format.format(value / 1000000000.0)); + } + if (value > 1000000.0) { + return String.format("%sM",format.format( value / 1000000.0)); + } + if (value > 1000.0) { + return String.format("%sK", format.format(value / 1000.0)); + } + + if (value == 0) { + return "0"; + } + + return format.format(value).equals("0") ? FULL_LBC_CURRENCY_FORMAT.format(value) : format.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); + } + } + + public static List getTagObjectsForTags(List tags) { + List tagObjects = new ArrayList<>(tags.size()); + for (String tag : tags) { + tagObjects.add(new Tag(tag)); + } + return tagObjects; + } + public static List getTagsForTagObjects(List tagObjects) { + List tags = new ArrayList<>(tagObjects.size()); + for (Tag tagObject : tagObjects) { + tags.add(tagObject.getLowercaseName()); + } + return tags; + } + public static List mergeKnownTags(List fetchedTags) { + List allKnownTags = getTagObjectsForTags(Predefined.DEFAULT_KNOWN_TAGS); + List followIndexes = new ArrayList<>(); + for (Tag tag : fetchedTags) { + if (!allKnownTags.contains(tag)) { + allKnownTags.add(tag); + } else if (tag.isFollowed()) { + followIndexes.add(allKnownTags.indexOf(tag)); + } + } + for (int index : followIndexes) { + allKnownTags.get(index).setFollowed(true); + } + return allKnownTags; + } + public static List filterFollowedTags(List tags) { + List followedTags = new ArrayList<>(); + for (Tag tag : tags) { + if (tag.isFollowed()) { + followedTags.add(tag); + } + } + return followedTags; + } + public static List filterDeletedClaims(List claims) { + List filtered = new ArrayList<>(); + for (Claim claim : claims) { + if (!Lbry.abandonedClaimIds.contains(claim.getClaimId())) { + filtered.add(claim); + } + } + return filtered; + } + + public static void setWunderbarValue(String value, Context context) { + if (context instanceof MainActivity) { + ((MainActivity) context).setWunderbarValue(value); + } + } + + public static String getDeviceName() { + if (Helper.isNullOrEmpty(Build.MANUFACTURER) || UNKNOWN.equalsIgnoreCase(Build.MANUFACTURER)) { + return Build.MODEL; + } + return String.format("%s %s", Build.MANUFACTURER, Build.MODEL); + } + + public static boolean channelExists(String channelName) { + for (Claim claim : Lbry.ownChannels) { + if (channelName.equalsIgnoreCase(claim.getName())) { + return true; + } + } + return false; + } + public static boolean claimNameExists(String claimName) { + for (Claim claim : Lbry.ownClaims) { + if (claimName.equalsIgnoreCase(claim.getName())) { + return true; + } + } + return false; + } + + public static String getRealPathFromURI_API19(final Context context, final Uri uri) { + return getRealPathFromURI_API19(context, uri, false); + } + /** + * https://gist.github.com/HBiSoft/15899990b8cd0723c3a894c1636550a8 + */ + @SuppressLint("NewApi") + public static String getRealPathFromURI_API19(final Context context, final Uri uri, boolean folderPath) { + + 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) { + String extStorageDirectory = Environment.getExternalStorageDirectory().toString(); + return folderPath ? + String.format("%s/Download", extStorageDirectory) : + String.format("%s/Download/%s", extStorageDirectory, fileName); + } + + String id = DocumentsContract.getDocumentId(uri); + if (id.startsWith("raw:")) { + id = id.replaceFirst("raw:", ""); + File file = new File(id); + if (file.exists()) + return id; + } + + String[] contentUriPrefixesToTry = new String[]{ + "content://downloads/public_downloads", + "content://downloads/my_downloads", + "content://downloads/all_downloads" + }; + + for (String contentUriPrefix : contentUriPrefixesToTry) { + Uri contentUri = Helper.parseInt(id, -1) > 0 ? + ContentUris.withAppendedId(Uri.parse(contentUriPrefix), Long.valueOf(id)) : + Uri.parse(contentUriPrefix); + try { + String path = getDataColumn(context, contentUri, null, null); + if (path != null) { + return path; + } + } catch (Exception ex) { + // pass + } + } + } + // 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 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; + } + + 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; + } + + /** + * @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()); + } + + public static void applyHtmlForTextView(TextView textView) { + textView.setMovementMethod(LinkMovementMethod.getInstance()); + textView.setText(HtmlCompat.fromHtml(textView.getText().toString(), HtmlCompat.FROM_HTML_MODE_LEGACY)); + } + + public static List filterInvalidReposts(List claims) { + List filtered = new ArrayList<>(); + for (Claim claim : claims) { + if (Claim.TYPE_REPOST.equalsIgnoreCase(claim.getValueType()) && claim.getRepostedClaim() == null) { + continue; + } + filtered.add(claim); + } + return filtered; + } + + public static List filterDownloads(List files) { + List filtered = new ArrayList<>(); + for (int i = 0; i < files.size(); i++) { + LbryFile file = files.get(i); + // remove own claims as well + if (file.getClaim() != null && Lbry.ownClaims.contains(file.getClaim())) { + continue; + } + if (!Helper.isNullOrEmpty(file.getDownloadPath())) { + filtered.add(file); + } + } + return filtered; + } + + public static List claimsFromFiles(List files) { + List claims = new ArrayList<>(); + for (LbryFile file : files) { + claims.add(file.getClaim()); + } + return claims; + } + + public static List claimsFromViewHistory(List history) { + List claims = new ArrayList<>(); + for (ViewHistory item : history) { + claims.add(Claim.fromViewHistory(item)); + } + return claims; + } + + public static void saveUrlHistory(String url, String title, int type) { + DatabaseHelper dbHelper = DatabaseHelper.getInstance(); + if (dbHelper != null) { + UrlSuggestion suggestion = new UrlSuggestion(); + suggestion.setUri(LbryUri.tryParse(url)); + suggestion.setType(type); + suggestion.setText(Helper.isNull(title) ? "" : title); + new SaveUrlHistoryTask(suggestion, dbHelper, null).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + + public static void saveViewHistory(String url, Claim claim) { + DatabaseHelper dbHelper = DatabaseHelper.getInstance(); + if (dbHelper != null) { + ViewHistory viewHistory = ViewHistory.fromClaimWithUrlAndDeviceName(claim, url, getDeviceName()); + new SaveViewHistoryTask(viewHistory, dbHelper, null).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); + } + } + public static String normalizeChannelName(String channelName) { + if (!channelName.startsWith("@")) { + return String.format("@%s", channelName); + } + return channelName; + } + + public static int getScaledValue(int value, float scale) { + return (int) (value * scale + 0.5f); + } + + public static String generateUrl() { + Random random = new Random(); + StringBuilder sb = new StringBuilder(); + sb.append(Predefined.ADJECTIVES.get(random.nextInt(Predefined.ADJECTIVES.size()))).append("-"). + append(Predefined.ADJECTIVES.get(random.nextInt(Predefined.ADJECTIVES.size()))).append("-"). + append(Predefined.ANIMALS.get(random.nextInt(Predefined.ANIMALS.size()))); + return sb.toString().toLowerCase(); + } + + public static void refreshRecyclerView(RecyclerView rv) { + if (rv == null) { + return; + } + + RecyclerView.Adapter adapter = rv.getAdapter(); + int prevScrollPosition = 0; + + RecyclerView.LayoutManager lm = rv.getLayoutManager(); + if (lm instanceof LinearLayoutManager) { + prevScrollPosition = ((LinearLayoutManager) lm).findLastCompletelyVisibleItemPosition(); + } else if (lm instanceof GridLayoutManager) { + prevScrollPosition = ((GridLayoutManager) lm).findLastCompletelyVisibleItemPosition(); + } + + rv.setAdapter(null); + rv.setAdapter(adapter); + rv.scrollToPosition(prevScrollPosition > 0 ? prevScrollPosition : 0); + } + + public static String makeid(int length) { + Random random = new Random(); + String chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"; + StringBuilder id = new StringBuilder(); + for (int i = 0; i < length; i++) { + id.append(chars.charAt(random.nextInt(chars.length()))); + } + return id.toString(); + } +} 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..ec2ed59d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/Lbry.java @@ -0,0 +1,493 @@ +package io.lbry.browser.utils; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.io.IOException; +import java.security.KeyStore; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Date; +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.ClaimSearchCacheValue; +import io.lbry.browser.model.LbryFile; +import io.lbry.browser.model.Tag; +import io.lbry.browser.model.Transaction; +import io.lbry.browser.model.WalletBalance; +import io.lbry.lbrysdk.Utils; +import kotlin.Pair; +import okhttp3.CacheControl; +import okhttp3.Headers; +import okhttp3.OkHttpClient; +import okhttp3.Request; +import okhttp3.RequestBody; +import okhttp3.Response; + +public final class Lbry { + private static final Object lock = new Object(); + public static LinkedHashMap claimCache = new LinkedHashMap<>(); + public static LinkedHashMap, ClaimSearchCacheValue> claimSearchCache = new LinkedHashMap<>(); + public static WalletBalance walletBalance = new WalletBalance(); + public static List knownTags = new ArrayList<>(); + public static List followedTags = new ArrayList<>(); + public static List ownClaims = new ArrayList<>(); + public static List ownChannels = new ArrayList<>(); // Make this a subset of ownClaims? + public static List abandonedClaimIds = new ArrayList<>(); + + public static final int TTL_CLAIM_SEARCH_VALUE = 120000; // 2-minute TTL for cache + 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"; + public static final String TAG = "Lbry"; + + // 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_FILE_DELETE = "file_delete"; + public static final String METHOD_GET = "get"; + public static final String METHOD_PUBLISH = "publish"; + + 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_VERSION = "version"; + + 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_CREATE = "support_create"; + 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 final String METHOD_CHANNEL_ABANDON = "channel_abandon"; + public static final String METHOD_CHANNEL_CREATE = "channel_create"; + public static final String METHOD_CHANNEL_UPDATE = "channel_update"; + + + public static final String METHOD_CLAIM_LIST = "claim_list"; + public static final String METHOD_PURCHASE_LIST = "purchase_list"; + public static final String METHOD_STREAM_ABANDON = "stream_abandon"; + public static final String METHOD_STREAM_REPOST = "stream_repost"; + + public static KeyStore KEYSTORE; + public static boolean SDK_READY = false; + + public static void startupInit() { + abandonedClaimIds = new ArrayList<>(); + ownChannels = new ArrayList<>(); + ownClaims = new ArrayList<>(); + knownTags = new ArrayList<>(); + followedTags = new ArrayList<>(); + } + + 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); + } + if (value instanceof Double) { + jsonParams.put(param.getKey(), (double) value); + } else { + jsonParams.put(param.getKey(), value == null ? JSONObject.NULL : 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")) { + if (json.isNull("result")) { + return null; + } + return json.get("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(!Helper.isNullOrEmpty(errorMessage) ? errorMessage : 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); + + addClaimToCache(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 LbryFile get(boolean saveFile) throws ApiCallException { + LbryFile file = null; + Map params = new HashMap<>(); + params.put("save_file", saveFile); + try { + JSONObject result = (JSONObject) parseResponse(apiCall(METHOD_GET, params)); + file = LbryFile.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, boolean downloads, int page, int pageSize) throws ApiCallException { + List files = new ArrayList<>(); + Map params = new HashMap<>(); + if (!Helper.isNullOrEmpty(claimId)) { + params.put("claim_id", claimId); + } + if (downloads) { + params.put("download_path", null); + params.put("comparison", "ne"); + } + if (page > 0) { + params.put("page", page); + } + if (pageSize > 0) { + params.put("page_size", pageSize); + } + 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); + LbryFile file = LbryFile.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)) { + ClaimSearchCacheValue value = claimSearchCache.get(options); + if (!value.isExpired(TTL_CLAIM_SEARCH_VALUE)) { + return claimSearchCache.get(options).getClaims(); + } + } + + 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); + + addClaimToCache(claim); + } + } + + claimSearchCache.put(options, new ClaimSearchCacheValue(claims, System.currentTimeMillis())); + } 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: %s", method, ex.getMessage()), ex); + } + return response; + } + public static Object genericApiCall(String method) throws ApiCallException { + return genericApiCall(method, null); + } + public static void addFollowedTag(Tag tag) { + synchronized (lock) { + if (!followedTags.contains(tag)) { + followedTags.add(tag); + } + } + } + public static void removeFollowedTag(Tag tag) { + synchronized (lock) { + followedTags.remove(tag); + } + } + public static void addKnownTag(Tag tag) { + synchronized (lock) { + if (!knownTags.contains(tag)) { + knownTags.add(tag); + } + } + } + + public static void addClaimToCache(Claim claim) { + ClaimCacheKey fullKey = ClaimCacheKey.fromClaim(claim); + ClaimCacheKey shortUrlKey = ClaimCacheKey.fromClaimShortUrl(claim); + ClaimCacheKey permanentUrlKey = ClaimCacheKey.fromClaimPermanentUrl(claim); + claimCache.put(fullKey, claim); + claimCache.put(permanentUrlKey, claim); + if (!Helper.isNullOrEmpty(shortUrlKey.getUrl())) { + claimCache.put(shortUrlKey, claim); + } + } + + public static void unsetFilesForCachedClaims(List claimIds) { + for (String claimId : claimIds) { + ClaimCacheKey key = new ClaimCacheKey(); + key.setClaimId(claimId); + if (claimCache.containsKey(key)) { + claimCache.get(key).setFile(null); + } + } + } +} diff --git a/app/src/main/java/io/lbry/browser/utils/LbryAnalytics.java b/app/src/main/java/io/lbry/browser/utils/LbryAnalytics.java new file mode 100644 index 00000000..ce20980c --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/LbryAnalytics.java @@ -0,0 +1,53 @@ +package io.lbry.browser.utils; + +import android.app.Activity; +import android.content.Context; +import android.os.Bundle; + +import com.google.firebase.analytics.FirebaseAnalytics; + +public class LbryAnalytics { + + public static final String EVENT_APP_EXCEPTION = "app_exception"; + public static final String EVENT_APP_LAUNCH = "app_launch"; + public static final String EVENT_EMAIL_ADDED = "email_added"; + public static final String EVENT_EMAIL_VERIFIED = "email_verified"; + public static final String EVENT_FIRST_RUN_COMPLETED = "first_run_completed"; + public static final String EVENT_FIRST_USER_AUTH = "first_user_auth"; + public static final String EVENT_LBRY_NOTIFICATION_OPEN = "lbry_notification_open"; + public static final String EVENT_LBRY_NOTIFICATION_RECEIVE = "lbry_notification_receive"; + public static final String EVENT_OPEN_FILE_PAGE = "open_file_page"; + public static final String EVENT_PLAY = "play"; + public static final String EVENT_PURCHASE_URI = "purchase_uri"; + public static final String EVENT_REWARD_ELIGIBILITY_COMPLETED = "reward_eligibility_completed"; + public static final String EVENT_TAG_FOLLOW = "tag_follow"; + public static final String EVENT_TAG_UNFOLLOW = "tag_unfollow"; + public static final String EVENT_PUBLISH = "publish"; + public static final String EVENT_CHANNEL_CREATE = "channel_create"; + public static final String EVENT_SEARCH = "search"; + + private static FirebaseAnalytics analytics; + + public static void init(Context context) { + analytics = FirebaseAnalytics.getInstance(context); + } + + public static void setCurrentScreen(Activity activity, String name, String className) { + analytics.setCurrentScreen(activity, name, className); + } + + public static void logEvent(String name) { + logEvent(name, null); + } + + public static void logEvent(String name, Bundle bundle) { + analytics.logEvent(name, bundle); + } + + public static void logException(String message, String exceptionName) { + Bundle bundle = new Bundle(); + bundle.putString("message", message); + bundle.putString("name", exceptionName); + logEvent("app_exception", bundle); + } +} 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..5d9f815f --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/LbryUri.java @@ -0,0 +1,336 @@ +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 LBRY_TV_BASE_URL = "https://lbry.tv"; + 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; + + private boolean isChannelUrl() { + return (!Helper.isNullOrEmpty(channelName) && Helper.isNullOrEmpty(streamName)) || (!Helper.isNullOrEmpty(claimName) && claimName.startsWith("@")); + } + + public static boolean isNameValid(String name) { + return !Pattern.compile(REGEX_INVALID_URI).matcher(name).find(); + } + + public static LbryUri tryParse(String url) { + try { + return parse(url, false); + } catch (LbryUriException ex) { + return null; + } + } + 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)); + } + } + + if (components.size() == 0) { + throw new LbryUriException("Regular expression error occurred while trying to parse the value"); + } + + // 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())) { + 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()), "")); + 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, boolean vanity) { + 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; + } + + StringBuilder sb = new StringBuilder(); + if (includeProto) { + sb.append(protoDefault); + } + sb.append(primaryClaimName); + if (vanity) { + return sb.toString(); + } + + 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; + + 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 String toTvString() { + 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; + } + + StringBuilder sb = new StringBuilder(); + sb.append(LBRY_TV_BASE_URL).append('/'); + sb.append(primaryClaimName); + + 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; + + if (!Helper.isNullOrEmpty(primaryClaimId)) { + sb.append(':').append(primaryClaimId); + } + if (!Helper.isNullOrEmpty(secondaryClaimName)) { + sb.append('/').append(secondaryClaimName); + } + if (!Helper.isNullOrEmpty(secondaryClaimId)) { + sb.append(':').append(secondaryClaimId); + } + return sb.toString(); + } + + public static String normalize(String url) throws LbryUriException { + return parse(url).toString(); + } + + public String toVanityString() { + return build(true, PROTO_DEFAULT, true); + } + public String toString() { + return build(true, PROTO_DEFAULT, false); + } + 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..02e9b07d --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/Lbryio.java @@ -0,0 +1,345 @@ +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.Reward; +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 { + + // TODO: Get this from the bundled aar + public static String SDK_VERSION = "0.74.0"; + + 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 subscriptions = 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 List allRewards = new ArrayList<>(); + public static List unclaimedRewards = new ArrayList<>(); + public static double totalUnclaimedRewardAmount = 0; + + 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(Helper.isNull(option.getValue()) ? "" : 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 | IllegalStateException ex) { + LbryAnalytics.logException(String.format("/user/me failed: %s", ex.getMessage()), ex.getClass().getName()); + android.util.Log.e(TAG, "Could not 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<>(); + if (context instanceof MainActivity) { + String firebaseToken = ((MainActivity) context).getFirebaseMessagingToken(); + if (!Helper.isNullOrEmpty(firebaseToken)) { + options.put("firebase_token", firebaseToken); + } + } + options.put("app_version", appVersion); + options.put("app_id", Lbry.INSTALLATION_ID); + options.put("node_id", ""); + options.put("daemon_version", SDK_VERSION); + 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) { + // pass + } + } + + 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; + } + } + + public static void addSubscription(Subscription subscription) { + synchronized (lock) { + if (!subscriptions.contains(subscription)) { + subscriptions.add(subscription); + } + } + } + public static void removeSubscription(Subscription subscription) { + synchronized (lock) { + subscriptions.remove(subscription); + } + } + public static void addCachedResolvedSubscription(Claim claim) { + synchronized (lock) { + if (!cacheResolvedSubscriptions.contains(claim)) { + cacheResolvedSubscriptions.add(claim); + } + } + } + public static void removeCachedResolvedSubscription(Claim claim) { + synchronized (lock) { + cacheResolvedSubscriptions.remove(claim); + } + } + + public static boolean isFollowing(Subscription subscription) { + return subscriptions.contains(subscription); + } + public static boolean isFollowing(Claim claim) { + return subscriptions.contains(Subscription.fromClaim(claim)); + } + + public static void updateRewardsLists(List rewards) { + synchronized (lock) { + allRewards.clear(); + unclaimedRewards.clear(); + totalUnclaimedRewardAmount = 0; + for (int i = 0; i < rewards.size(); i++) { + Reward reward = rewards.get(i); + allRewards.add(reward); + if (!reward.isClaimed()) { + unclaimedRewards.add(reward); + totalUnclaimedRewardAmount += reward.getRewardAmount(); + } + } + } + } +} 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..cd8d4da2 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/Lighthouse.java @@ -0,0 +1,119 @@ +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 Map, List> searchCache = new HashMap<>(); + + private static Map buildSearchOptionsKey(String rawQuery, int size, int from, boolean nsfw, String relatedTo) { + Map options = new HashMap<>(); + options.put("s", rawQuery); + options.put("size", size); + options.put("from", from); + options.put("nsfw", nsfw); + if (!Helper.isNullOrEmpty(relatedTo)) { + options.put("related_to", relatedTo); + } + return options; + } + + 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); + } + + Map cacheKey = buildSearchOptionsKey(rawQuery, size, from, nsfw, relatedTo); + if (searchCache.containsKey(cacheKey)) { + return searchCache.get(cacheKey); + } + + 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); + } + searchCache.put(cacheKey, results); + } 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/java/io/lbry/browser/utils/Predefined.java b/app/src/main/java/io/lbry/browser/utils/Predefined.java new file mode 100644 index 00000000..55c75711 --- /dev/null +++ b/app/src/main/java/io/lbry/browser/utils/Predefined.java @@ -0,0 +1,1898 @@ +package io.lbry.browser.utils; + +import java.util.Arrays; +import java.util.List; +import java.util.Map; + +import io.lbry.browser.R; +import io.lbry.browser.model.Language; +import io.lbry.browser.model.License; + +public final class Predefined { + public static final List DEFAULT_KNOWN_TAGS = Arrays.asList( + "free speech", + "censorship", + "gaming", + "pop culture", + "entertainment", + "technology", + "music", + "funny", + "education", + "learning", + "news", + "gameplay", + "nature", + "beliefs", + "comedy", + "games", + "film & animation", + "whothinks", + "game", + "weapons", + "blockchain", + "video game", + "sports", + "walkthrough", + "art", + "pc", + "minecraft", + "playthrough", + "economics", + "automotive", + "play", + "tutorial", + "twitch", + "how to", + "ps4", + "bitcoin", + "fortnite", + "commentary", + "lets play", + "fun", + "politics", + "travel", + "food", + "science", + "xbox", + "liberal", + "democrat", + "progressive", + "survival", + "non-profits", + "activism", + "cryptocurrency", + "playstation", + "nintendo", + "government", + "steam", + "podcast", + "gamer", + "horror", + "conservative", + "reaction", + "trailer", + "love", + "cnn", + "republican", + "political", + "hangoutsonair", + "hoa", + "msnbc", + "cbs", + "anime", + "donald trump", + "fiction", + "fox news", + "crypto", + "ethereum", + "call of duty", + "android", + "multiplayer", + "epic", + "rpg", + "adventure", + "secular talk", + "btc", + "atheist", + "atheism", + "video games", + "ps3", + "cod", + "online", + "agnostic", + "movie", + "fps", + "lets", + "mod", + "world", + "reviews", + "sharefactory", + "space", + "pokemon", + "stream", + "hilarious", + "lol", + "sony", + "god", + "dance", + "pvp", + "tech", + "strategy", + "zombies", + "fail", + "film", + "xbox360", + "animation", + "unboxing", + "money", + "wwe", + "mods", + "indie", + "pubg", + "ios", + "history", + "rap", + "mobile", + "trump", + "hack", + "flat earth", + "trap", + "humor", + "vlogging", + "fox", + "news radio", + "facebook", + "edm", + "fitness", + "vaping", + "hip hop", + "secular", + "jesus", + "song", + "vape", + "guitar", + "remix", + "mining", + "daily", + "diy", + "pets", + "videogame", + "death", + "funny moments", + "religion", + "media", + "viral", + "war", + "nbc", + "freedom", + "gold", + "family", + "meme", + "zombie", + "photography", + "chill", + "sniper", + "computer", + "iphone", + "dragon", + "bible", + "pro", + "overwatch", + "litecoin", + "gta", + "house", + "fire", + "bass", + "truth", + "crash", + "mario", + "league of legends", + "wii", + "mmorpg", + "health", + "marvel", + "racing", + "apple", + "instrumental", + "earth", + "destiny", + "satire", + "race", + "training", + "electronic", + "boss", + "roblox", + "family friendly", + "california", + "react", + "christian", + "mmo", + "twitter", + "help", + "star", + "cars", + "random", + "top 10", + "ninja", + "guns", + "linux", + "lessons", + "vegan", + "future", + "dota 2", + "studio", + "star wars", + "shooting", + "nasa", + "rock", + "league", + "subscribe", + "water", + "gta v", + "car", + "samsung", + "music video", + "skyrim", + "dog", + "comics", + "shooter game", + "bo3", + "halloween", + "liberty", + "eth", + "conspiracy", + "knife", + "fashion", + "stories", + "vapor", + "nvidia", + "cute", + "beat", + "nintendo switch", + "fantasy", + "christmas", + "world of warcraft", + "industry", + "cartoon", + "garden", + "animals", + "windows", + "happy", + "magic", + "memes", + "design", + "tactical", + "fallout 4", + "puzzle", + "parody", + "rv", + "beats", + "building", + "disney", + "drone", + "ps2", + "beach", + "metal", + "christianity", + "business", + "mix", + "bo2", + "cover", + "senate", + "4k", + "united states", + "final", + "hero", + "playing", + "dlc", + "ubisoft", + "halo", + "pc gaming", + "raw", + "investing", + "online learning", + "software", + "ark", + "mojang", + "console", + "battle royale", + "canon", + "microsoft", + "camping", + "ufo", + "progressive talk", + "switch", + "fpv", + "arcade", + "school", + "driving", + "bodybuilding", + "drama", + "retro", + "science fiction", + "eggs", + "australia", + "modded", + "rainbow", + "gamers", + "resident evil", + "drawing", + "brasil", + "england", + "hillary clinton", + "singing", + "final fantasy", + "hiphop", + "video blog", + "mature", + "quad", + "noob", + "simulation", + "illuminati", + "poetry", + "dayz", + "manga", + "howto", + "insane", + "press", + "special", + "church", + "ico", + "weird", + "libertarian", + "crafting", + "level", + "comic", + "sandbox", + "daily vlog", + "outdoor", + "black ops", + "sound", + "christ", + "duty", + "juvenile fiction", + "pc game", + "how-to", + "ww2", + "creepy", + "artist", + "galaxy", + "destiny 2", + "new music", + "quest", + "lee", + "pacman", + "super smash bros", + "day", + "survival horror", + "patreon", + "bitcoin price", + "trending", + "open world", + "wii u", + "dope", + "reaper", + "sniping", + "dubstep", + "truck", + "planet", + "dc", + "amazon", + "spirituality", + "universe", + "video game culture", + "community", + "cat", + "aliens", + "tourism", + "altcoins", + "style", + "travel trailer", + "rda", + "gun", + "secret", + "far cry 5", + "auto", + "culture", + "dj", + "mw2", + "lord", + "full time rving", + "role-playing game", + "prank", + "grand theft auto", + "master", + "wrestling", + "sci-fi", + "workout", + "ghost", + "fake news", + "silly", + "season", + "bo4", + "trading", + "extreme", + "economy", + "combat", + "plays", + "muslim", + "pubg mobile", + "clips", + "bo1", + "paypal", + "sims", + "exploration", + "light", + "ripple", + "paranormal", + "football", + "capcom", + "rta", + "discord", + "batman", + "player", + "server", + "anarchy", + "military", + "playlist", + "cosplay", + "rv park", + "rant", + "edit", + "germany", + "reading", + "chris", + "flash", + "loot", + "bitcoin gratis", + "game reviews", + "movies", + "stupid", + "latest news", + "squad gameplay", + "guru", + "timelapse", + "black ops 3", + "holiday", + "soul", + "motivation", + "mw3", + "vacation", + "sega", + "19th century", + "pop", + "sims 4", + "post", + "smok", + "island", + "scotland", + "paladins", + "warrior", + "creepypasta", + "role-playing", + "solar", + "vr", + "animal", + "peace", + "consciousness", + "dota", + "audio", + "mass effect", + "humour", + "first look", + "videogames", + "future bass", + "freestyle", + "hardcore", + "portugal", + "dantdm", + "teaser", + "lbry", + "coronavirus", + "covidcuts", + "covid-19" + ); + public static final List MATURE_TAGS = Arrays.asList("mature", "nsfw", "porn", "xxx"); + public static final List ADJECTIVES = Arrays.asList( + "aback", + "abaft", + "abandoned", + "abashed", + "aberrant", + "abhorrent", + "abiding", + "abject", + "ablaze", + "able", + "abnormal", + "aboard", + "aboriginal", + "abortive", + "abounding", + "abrasive", + "abrupt", + "absent", + "absorbed", + "absorbing", + "abstracted", + "absurd", + "abundant", + "abusive", + "acceptable", + "accessible", + "accidental", + "accurate", + "acid", + "acidic", + "acoustic", + "acrid", + "actually", + "ad", + "hoc", + "adamant", + "adaptable", + "addicted", + "adhesive", + "adjoining", + "adorable", + "adventurous", + "afraid", + "aggressive", + "agonizing", + "agreeable", + "ahead", + "ajar", + "alcoholic", + "alert", + "alike", + "alive", + "alleged", + "alluring", + "aloof", + "amazing", + "ambiguous", + "ambitious", + "amuck", + "amused", + "amusing", + "ancient", + "angry", + "animated", + "annoyed", + "annoying", + "anxious", + "apathetic", + "aquatic", + "aromatic", + "arrogant", + "ashamed", + "aspiring", + "assorted", + "astonishing", + "attractive", + "auspicious", + "automatic", + "available", + "average", + "awake", + "aware", + "awesome", + "awful", + "axiomatic", + "bad", + "barbarous", + "bashful", + "bawdy", + "beautiful", + "befitting", + "belligerent", + "beneficial", + "bent", + "berserk", + "best", + "better", + "bewildered", + "big", + "billowy", + "bite-sized", + "bitter", + "bizarre", + "black", + "black-and-white", + "bloody", + "blue", + "blue-eyed", + "blushing", + "boiling", + "boorish", + "bored", + "boring", + "bouncy", + "boundless", + "brainy", + "brash", + "brave", + "brawny", + "breakable", + "breezy", + "brief", + "bright", + "bright", + "broad", + "broken", + "brown", + "bumpy", + "burly", + "bustling", + "busy", + "cagey", + "calculating", + "callous", + "calm", + "capable", + "capricious", + "careful", + "careless", + "caring", + "cautious", + "ceaseless", + "certain", + "changeable", + "charming", + "cheap", + "cheerful", + "chemical", + "chief", + "childlike", + "chilly", + "chivalrous", + "chubby", + "chunky", + "clammy", + "classy", + "clean", + "clear", + "clever", + "cloistered", + "cloudy", + "closed", + "clumsy", + "cluttered", + "coherent", + "cold", + "colorful", + "colossal", + "combative", + "comfortable", + "common", + "complete", + "complex", + "concerned", + "condemned", + "confused", + "conscious", + "cooing", + "cool", + "cooperative", + "coordinated", + "courageous", + "cowardly", + "crabby", + "craven", + "crazy", + "creepy", + "crooked", + "crowded", + "cruel", + "cuddly", + "cultured", + "cumbersome", + "curious", + "curly", + "curved", + "curvy", + "cut", + "cute", + "cute", + "cynical", + "daffy", + "daily", + "damaged", + "damaging", + "damp", + "dangerous", + "dapper", + "dark", + "dashing", + "dazzling", + "dead", + "deadpan", + "deafening", + "dear", + "debonair", + "decisive", + "decorous", + "deep", + "deeply", + "defeated", + "defective", + "defiant", + "delicate", + "delicious", + "delightful", + "demonic", + "delirious", + "dependent", + "depressed", + "deranged", + "descriptive", + "deserted", + "detailed", + "determined", + "devilish", + "didactic", + "different", + "difficult", + "diligent", + "direful", + "dirty", + "disagreeable", + "disastrous", + "discreet", + "disgusted", + "disgusting", + "disillusioned", + "dispensable", + "distinct", + "disturbed", + "divergent", + "dizzy", + "domineering", + "doubtful", + "drab", + "draconian", + "dramatic", + "dreary", + "drunk", + "dry", + "dull", + "dusty", + "dynamic", + "dysfunctional", + "eager", + "early", + "earsplitting", + "earthy", + "easy", + "eatable", + "economic", + "educated", + "efficacious", + "efficient", + "eight", + "elastic", + "elated", + "elderly", + "electric", + "elegant", + "elfin", + "elite", + "embarrassed", + "eminent", + "empty", + "enchanted", + "enchanting", + "encouraging", + "endurable", + "energetic", + "enormous", + "entertaining", + "enthusiastic", + "envious", + "equable", + "equal", + "erect", + "erratic", + "ethereal", + "evanescent", + "evasive", + "even", + "excellent", + "excited", + "exciting", + "exclusive", + "exotic", + "expensive", + "extra-large", + "extra-small", + "exuberant", + "exultant", + "fabulous", + "faded", + "faint", + "fair", + "faithful", + "fallacious", + "false", + "familiar", + "famous", + "fanatical", + "fancy", + "fantastic", + "far", + "far-flung", + "fascinated", + "fast", + "fat", + "faulty", + "fearful", + "fearless", + "feeble", + "feigned", + "female", + "fertile", + "festive", + "few", + "fierce", + "filthy", + "fine", + "finicky", + "first", + "five", + "fixed", + "flagrant", + "flaky", + "flashy", + "flat", + "flawless", + "flimsy", + "flippant", + "flowery", + "fluffy", + "fluttering", + "foamy", + "foolish", + "foregoing", + "forgetful", + "fortunate", + "four", + "frail", + "fragile", + "frantic", + "free", + "freezing", + "frequent", + "fresh", + "fretful", + "friendly", + "frightened", + "frightening", + "full", + "fumbling", + "functional", + "funny", + "furry", + "furtive", + "future", + "futuristic", + "fuzzy", + "gabby", + "gainful", + "gamy", + "gaping", + "garrulous", + "gaudy", + "general", + "gentle", + "giant", + "giddy", + "gifted", + "gigantic", + "glamorous", + "gleaming", + "glib", + "glistening", + "glorious", + "glossy", + "godly", + "good", + "goofy", + "gorgeous", + "graceful", + "grandiose", + "grateful", + "gratis", + "gray", + "greasy", + "great", + "greedy", + "green", + "grey", + "grieving", + "groovy", + "grotesque", + "grouchy", + "grubby", + "gruesome", + "grumpy", + "guarded", + "guiltless", + "gullible", + "gusty", + "guttural", + "habitual", + "half", + "hallowed", + "halting", + "handsome", + "handsomely", + "handy", + "hanging", + "hapless", + "happy", + "hard", + "hard-to-find", + "harmonious", + "harsh", + "hateful", + "heady", + "healthy", + "heartbreaking", + "heavenly", + "heavy", + "hellish", + "helpful", + "helpless", + "hesitant", + "hideous", + "high", + "highfalutin", + "high-pitched", + "hilarious", + "hissing", + "historical", + "holistic", + "hollow", + "homeless", + "homely", + "honorable", + "horrible", + "hospitable", + "hot", + "huge", + "hulking", + "humdrum", + "humorous", + "hungry", + "hurried", + "hurt", + "hushed", + "husky", + "hypnotic", + "hysterical", + "icky", + "icy", + "idiotic", + "ignorant", + "ill", + "illegal", + "ill-fated", + "ill-informed", + "illustrious", + "imaginary", + "immense", + "imminent", + "impartial", + "imperfect", + "impolite", + "important", + "imported", + "impossible", + "incandescent", + "incompetent", + "inconclusive", + "industrious", + "incredible", + "inexpensive", + "infamous", + "innate", + "innocent", + "inquisitive", + "insidious", + "instinctive", + "intelligent", + "interesting", + "internal", + "invincible", + "irate", + "irritating", + "itchy", + "jaded", + "jagged", + "jazzy", + "jealous", + "jittery", + "jobless", + "jolly", + "joyous", + "judicious", + "juicy", + "jumbled", + "jumpy", + "juvenile", + "kaput", + "keen", + "kind", + "kindhearted", + "kindly", + "knotty", + "knowing", + "knowledgeable", + "known", + "labored", + "lackadaisical", + "lacking", + "lame", + "lamentable", + "languid", + "large", + "last", + "late", + "laughable", + "lavish", + "lazy", + "lean", + "learned", + "left", + "legal", + "lethal", + "level", + "lewd", + "light", + "like", + "likeable", + "limping", + "literate", + "little", + "lively", + "lively", + "living", + "lonely", + "long", + "longing", + "long-term", + "loose", + "lopsided", + "loud", + "loutish", + "lovely", + "loving", + "low", + "lowly", + "lucky", + "ludicrous", + "lumpy", + "lush", + "luxuriant", + "lying", + "lyrical", + "macabre", + "macho", + "maddening", + "madly", + "magenta", + "magical", + "magnificent", + "majestic", + "makeshift", + "male", + "malicious", + "mammoth", + "maniacal", + "many", + "marked", + "massive", + "married", + "marvelous", + "material", + "materialistic", + "mature", + "mean", + "measly", + "meaty", + "medical", + "meek", + "mellow", + "melodic", + "melted", + "merciful", + "mere", + "messy", + "mighty", + "military", + "milky", + "mindless", + "miniature", + "minor", + "miscreant", + "misty", + "mixed", + "moaning", + "modern", + "moldy", + "momentous", + "motionless", + "mountainous", + "muddled", + "mundane", + "murky", + "mushy", + "mute", + "mysterious", + "naive", + "nappy", + "narrow", + "nasty", + "natural", + "naughty", + "nauseating", + "near", + "neat", + "nebulous", + "necessary", + "needless", + "needy", + "neighborly", + "nervous", + "new", + "next", + "nice", + "nifty", + "nimble", + "nine", + "nippy", + "noiseless", + "noisy", + "nonchalant", + "nondescript", + "nonstop", + "normal", + "nostalgic", + "nosy", + "noxious", + "null", + "numberless", + "numerous", + "nutritious", + "nutty", + "oafish", + "obedient", + "obeisant", + "obese", + "obnoxious", + "obscene", + "obsequious", + "observant", + "obsolete", + "obtainable", + "oceanic", + "odd", + "offbeat", + "old", + "old-fashioned", + "omniscient", + "one", + "onerous", + "open", + "opposite", + "optimal", + "orange", + "ordinary", + "organic", + "ossified", + "outgoing", + "outrageous", + "outstanding", + "oval", + "overconfident", + "overjoyed", + "overrated", + "overt", + "overwrought", + "painful", + "painstaking", + "pale", + "paltry", + "panicky", + "panoramic", + "parallel", + "parched", + "parsimonious", + "past", + "pastoral", + "pathetic", + "peaceful", + "penitent", + "perfect", + "periodic", + "permissible", + "perpetual", + "petite", + "petite", + "phobic", + "physical", + "picayune", + "pink", + "piquant", + "placid", + "plain", + "plant", + "plastic", + "plausible", + "pleasant", + "plucky", + "pointless", + "poised", + "polite", + "political", + "poor", + "possessive", + "possible", + "powerful", + "precious", + "premium", + "present", + "pretty", + "previous", + "pricey", + "prickly", + "private", + "probable", + "productive", + "profuse", + "protective", + "proud", + "psychedelic", + "psychotic", + "public", + "puffy", + "pumped", + "puny", + "purple", + "purring", + "pushy", + "puzzled", + "puzzling", + "quack", + "quaint", + "quarrelsome", + "questionable", + "quick", + "quickest", + "quiet", + "quirky", + "quixotic", + "quizzical", + "rabid", + "racial", + "ragged", + "rainy", + "rambunctious", + "rampant", + "rapid", + "rare", + "raspy", + "ratty", + "ready", + "real", + "rebel", + "receptive", + "recondite", + "red", + "redundant", + "reflective", + "regular", + "relieved", + "remarkable", + "reminiscent", + "repulsive", + "resolute", + "resonant", + "responsible", + "rhetorical", + "rich", + "right", + "righteous", + "rightful", + "rigid", + "ripe", + "ritzy", + "roasted", + "robust", + "romantic", + "roomy", + "rotten", + "rough", + "round", + "royal", + "ruddy", + "rude", + "rural", + "rustic", + "ruthless", + "sable", + "sad", + "safe", + "salty", + "same", + "sassy", + "satisfying", + "savory", + "scandalous", + "scarce", + "scared", + "scary", + "scattered", + "scientific", + "scintillating", + "scrawny", + "screeching", + "second", + "second-hand", + "secret", + "secretive", + "sedate", + "seemly", + "selective", + "selfish", + "separate", + "serious", + "shaggy", + "shaky", + "shallow", + "sharp", + "shiny", + "shivering", + "shocking", + "short", + "shrill", + "shut", + "shy", + "sick", + "silent", + "silent", + "silky", + "silly", + "simple", + "simplistic", + "sincere", + "six", + "skillful", + "skinny", + "sleepy", + "slim", + "slimy", + "slippery", + "sloppy", + "slow", + "small", + "smart", + "smelly", + "smiling", + "smoggy", + "smooth", + "sneaky", + "snobbish", + "snotty", + "soft", + "soggy", + "solid", + "somber", + "sophisticated", + "sordid", + "sore", + "sore", + "sour", + "sparkling", + "special", + "spectacular", + "spicy", + "spiffy", + "spiky", + "spiritual", + "spiteful", + "splendid", + "spooky", + "spotless", + "spotted", + "spotty", + "spurious", + "squalid", + "square", + "squealing", + "squeamish", + "staking", + "stale", + "standing", + "statuesque", + "steadfast", + "steady", + "steep", + "stereotyped", + "sticky", + "stiff", + "stimulating", + "stingy", + "stormy", + "straight", + "strange", + "striped", + "strong", + "stupendous", + "stupid", + "sturdy", + "subdued", + "subsequent", + "substantial", + "successful", + "succinct", + "sudden", + "sulky", + "super", + "superb", + "superficial", + "supreme", + "swanky", + "sweet", + "sweltering", + "swift", + "symptomatic", + "synonymous", + "taboo", + "tacit", + "tacky", + "talented", + "tall", + "tame", + "tan", + "tangible", + "tangy", + "tart", + "tasteful", + "tasteless", + "tasty", + "tawdry", + "tearful", + "tedious", + "teeny", + "teeny-tiny", + "telling", + "temporary", + "ten", + "tender", + "tense", + "tense", + "tenuous", + "terrible", + "terrific", + "tested", + "testy", + "thankful", + "therapeutic", + "thick", + "thin", + "thinkable", + "third", + "thirsty", + "thoughtful", + "thoughtless", + "threatening", + "three", + "thundering", + "tidy", + "tight", + "tightfisted", + "tiny", + "tired", + "tiresome", + "toothsome", + "torpid", + "tough", + "towering", + "tranquil", + "trashy", + "tremendous", + "tricky", + "trite", + "troubled", + "truculent", + "true", + "truthful", + "two", + "typical", + "ubiquitous", + "ugliest", + "ugly", + "ultra", + "unable", + "unaccountable", + "unadvised", + "unarmed", + "unbecoming", + "unbiased", + "uncovered", + "understood", + "undesirable", + "unequal", + "unequaled", + "uneven", + "unhealthy", + "uninterested", + "unique", + "unkempt", + "unknown", + "unnatural", + "unruly", + "unsightly", + "unsuitable", + "untidy", + "unused", + "unusual", + "unwieldy", + "unwritten", + "upbeat", + "uppity", + "upset", + "uptight", + "used", + "useful", + "useless", + "utopian", + "utter", + "uttermost", + "vacuous", + "vagabond", + "vague", + "valuable", + "various", + "vast", + "vengeful", + "venomous", + "verdant", + "versed", + "victorious", + "vigorous", + "violent", + "violet", + "vivacious", + "voiceless", + "volatile", + "voracious", + "vulgar", + "wacky", + "waggish", + "waiting", + "wakeful", + "wandering", + "wanting", + "warlike", + "warm", + "wary", + "wasteful", + "watery", + "weak", + "wealthy", + "weary", + "well-groomed", + "well-made", + "well-off", + "well-to-do", + "wet", + "whimsical", + "whispering", + "white", + "whole", + "wholesale", + "wicked", + "wide", + "wide-eyed", + "wiggly", + "wild", + "willing", + "windy", + "wiry", + "wise", + "wistful", + "witty", + "woebegone", + "womanly", + "wonderful", + "wooden", + "woozy", + "workable", + "worried", + "worthless", + "wrathful", + "wretched", + "wrong", + "wry", + "xenophobic", + "yellow", + "yielding", + "young", + "youthful", + "yummy", + "zany", + "zealous", + "zesty", + "zippy", + "zonked" + ); + public static final List ANIMALS = Arrays.asList( + "Aardvark", + "Albatross", + "Alligator", + "Alpaca", + "Ant", + "Anteater", + "Antelope", + "Ape", + "Armadillo", + "Donkey", + "Baboon", + "Badger", + "Barracuda", + "Bat", + "Bear", + "Beaver", + "Bee", + "Bison", + "Boar", + "Buffalo", + "Butterfly", + "Camel", + "Capybara", + "Caribou", + "Cassowary", + "Cat", + "Caterpillar", + "Cattle", + "Chamois", + "Cheetah", + "Chicken", + "Chimpanzee", + "Chinchilla", + "Chough", + "Clam", + "Cobra", + "Cockroach", + "Cod", + "Cormorant", + "Coyote", + "Crab", + "Crane", + "Crocodile", + "Crow", + "Curlew", + "Deer", + "Dinosaur", + "Dog", + "Dogfish", + "Dolphin", + "Dotterel", + "Dove", + "Dragonfly", + "Duck", + "Dugong", + "Dunlin", + "Eagle", + "Echidna", + "Eel", + "Eland", + "Elephant", + "Elk", + "Emu", + "Falcon", + "Ferret", + "Finch", + "Fish", + "Flamingo", + "Fly", + "Fox", + "Frog", + "Gaur", + "Gazelle", + "Gerbil", + "Giraffe", + "Gnat", + "Gnu", + "Goat", + "Goldfinch", + "Goldfish", + "Goose", + "Gorilla", + "Goshawk", + "Grasshopper", + "Grouse", + "Guanaco", + "Gull", + "Hamster", + "Hare", + "Hawk", + "Hedgehog", + "Heron", + "Herring", + "Hippopotamus", + "Hornet", + "Horse", + "Human", + "Hummingbird", + "Hyena", + "Ibex", + "Ibis", + "Jackal", + "Jaguar", + "Jay", + "Jellyfish", + "Kangaroo", + "Kingfisher", + "Koala", + "Kookabura", + "Kouprey", + "Kudu", + "Lapwing", + "Lark", + "Lemur", + "Leopard", + "Lion", + "Llama", + "Lobster", + "Locust", + "Loris", + "Louse", + "Lyrebird", + "Magpie", + "Mallard", + "Manatee", + "Mandrill", + "Mantis", + "Marten", + "Meerkat", + "Mink", + "Mole", + "Mongoose", + "Monkey", + "Moose", + "Mosquito", + "Mouse", + "Mule", + "Narwhal", + "Newt", + "Nightingale", + "Octopus", + "Okapi", + "Opossum", + "Oryx", + "Ostrich", + "Otter", + "Owl", + "Oyster", + "Panther", + "Parrot", + "Partridge", + "Peafowl", + "Pelican", + "Penguin", + "Pheasant", + "Pig", + "Pigeon", + "Pony", + "Porcupine", + "Porpoise", + "Quail", + "Quelea", + "Quetzal", + "Rabbit", + "Raccoon", + "Rail", + "Ram", + "Rat", + "Raven", + "Reindeer", + "Rhinoceros", + "Rook", + "Salamander", + "Salmon", + "Sandpiper", + "Sardine", + "Scorpion", + "Seahorse", + "Seal", + "Shark", + "Sheep", + "Shrew", + "Skunk", + "Snail", + "Snake", + "Sparrow", + "Spider", + "Spoonbill", + "Squid", + "Squirrel", + "Starling", + "Stingray", + "Stinkbug", + "Stork", + "Swallow", + "Swan", + "Tapir", + "Tarsier", + "Termite", + "Tiger", + "Toad", + "Trout", + "Turkey", + "Turtle", + "Viper", + "Vulture", + "Wallaby", + "Walrus", + "Wasp", + "Weasel", + "Whale", + "Wildcat", + "Wolf", + "Wolverine", + "Wombat", + "Woodcock", + "Woodpecker", + "Worm", + "Wren", + "Yak", + "Zebra" + ); + + public static final List PUBLISH_LANGUAGES = Arrays.asList( + new Language("en", "English", R.string.english), + new Language("zh", "Chinese", R.string.chinese), + new Language("fr", "French", R.string.french), + new Language("de", "German", R.string.german), + new Language("jp", "Japanese", R.string.japanese), + new Language("ru", "Russian", R.string.russian), + new Language("es", "Spanish", R.string.spanish), + new Language("id", "Indonesian", R.string.indonesian), + new Language("it", "Italian", R.string.italian), + new Language("nl", "Dutch", R.string.dutch), + new Language("tr", "Turkish", R.string.turkish), + new Language("pl", "Polish", R.string.polish), + new Language("ms", "Malay", R.string.malay), + new Language("pt", "Portuguese", R.string.portuguese), + new Language("vi", "Vietnamese", R.string.vietnamese), + new Language("th", "Thai", R.string.thai), + new Language("ar", "Arabic", R.string.arabic), + new Language("cs", "Czech", R.string.czech), + new Language("hr", "Croatian", R.string.croatian), + new Language("km", "Cambodian", R.string.cambodian), + new Language("ko", "Korean", R.string.korean), + new Language("no", "Norwegian", R.string.norwegian), + new Language("ro", "Romanian", R.string.romanian), + new Language("hi", "Hindi", R.string.hindi), + new Language("el", "Greek", R.string.greek) + ); + + public static final String LICENSE_COPYRIGHTED = "Copyrighted"; + public static final String LICENSE_OTHER = "Other"; + public static final List LICENSES = Arrays.asList( + new License("None", R.string.none), + new License("Public Domain", R.string.public_domain), + new License("Creative Commons Attribution 4.0 International", "https://creativecommons.org/licenses/by/4.0/legalcode", R.string.cca_4_0_international), + new License("Creative Commons Attribution-ShareAlike 4.0 International", "https://creativecommons.org/licenses/by-sa/4.0/legalcode", R.string.cca_sa_4_0_international), + new License("Creative Commons Attribution-NoDerivatives 4.0 International", "https://creativecommons.org/licenses/by-nd/4.0/legalcode", R.string.cca_nd_4_0_international), + new License("Creative Commons Attribution-NonCommercial 4.0 International", "https://creativecommons.org/licenses/by-nc/4.0/legalcode", R.string.cca_nc_4_0_international), + new License("Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International", "https://creativecommons.org/licenses/by-nc-sa/4.0/legalcode", R.string.cca_nc_sa_4_0_international), + new License("Creative Commons Attribution-NonCommercial-NoDerivatives 4.0 International", "https://creativecommons.org/licenses/by-nc-nd/4.0/legalcode", R.string.cca_nc_nd_4_0_international), + new License(LICENSE_COPYRIGHTED, R.string.copyrighted), + new License(LICENSE_OTHER, R.string.other) + ); +} 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_add.xml b/app/src/main/res/drawable-anydpi/ic_add.xml new file mode 100644 index 00000000..fa797053 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_add.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_arrow_left.xml b/app/src/main/res/drawable-anydpi/ic_arrow_left.xml new file mode 100644 index 00000000..b94a76b7 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_arrow_left.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_arrow_right.xml b/app/src/main/res/drawable-anydpi/ic_arrow_right.xml new file mode 100644 index 00000000..ac8898d5 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_arrow_right.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_cast.xml b/app/src/main/res/drawable-anydpi/ic_cast.xml new file mode 100644 index 00000000..da0e952f --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_cast.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_cast_connected.xml b/app/src/main/res/drawable-anydpi/ic_cast_connected.xml new file mode 100644 index 00000000..491bf6c8 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_cast_connected.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_check_circle.xml b/app/src/main/res/drawable-anydpi/ic_check_circle.xml new file mode 100644 index 00000000..bbdc3744 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_check_circle.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_delete.xml b/app/src/main/res/drawable-anydpi/ic_delete.xml new file mode 100644 index 00000000..24678bc6 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_delete.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_edit.xml b/app/src/main/res/drawable-anydpi/ic_edit.xml new file mode 100644 index 00000000..f13ba712 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_edit.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_forward_10.xml b/app/src/main/res/drawable-anydpi/ic_forward_10.xml new file mode 100644 index 00000000..55e7a754 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_forward_10.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_fullscreen.xml b/app/src/main/res/drawable-anydpi/ic_fullscreen.xml new file mode 100644 index 00000000..b8f99968 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_fullscreen.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_fullscreen_exit.xml b/app/src/main/res/drawable-anydpi/ic_fullscreen_exit.xml new file mode 100644 index 00000000..f8116a2e --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_fullscreen_exit.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_pause.xml b/app/src/main/res/drawable-anydpi/ic_pause.xml new file mode 100644 index 00000000..06af6a1e --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_pause.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_photo.xml b/app/src/main/res/drawable-anydpi/ic_photo.xml new file mode 100644 index 00000000..e2343761 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_photo.xml @@ -0,0 +1,14 @@ + + + + diff --git a/app/src/main/res/drawable-anydpi/ic_play.xml b/app/src/main/res/drawable-anydpi/ic_play.xml new file mode 100644 index 00000000..420941aa --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_play.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_record.xml b/app/src/main/res/drawable-anydpi/ic_record.xml new file mode 100644 index 00000000..42f62d58 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_record.xml @@ -0,0 +1,11 @@ + + + diff --git a/app/src/main/res/drawable-anydpi/ic_replay_10.xml b/app/src/main/res/drawable-anydpi/ic_replay_10.xml new file mode 100644 index 00000000..a238e2d5 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_replay_10.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_stop.xml b/app/src/main/res/drawable-anydpi/ic_stop.xml new file mode 100644 index 00000000..34bc28f1 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_stop.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_upload.xml b/app/src/main/res/drawable-anydpi/ic_upload.xml new file mode 100644 index 00000000..faebc4a5 --- /dev/null +++ b/app/src/main/res/drawable-anydpi/ic_upload.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_add.png b/app/src/main/res/drawable-hdpi/ic_add.png new file mode 100644 index 00000000..2b125f5f Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_add.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_arrow_left.png b/app/src/main/res/drawable-hdpi/ic_arrow_left.png new file mode 100644 index 00000000..11d1f4b6 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_arrow_left.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_arrow_right.png b/app/src/main/res/drawable-hdpi/ic_arrow_right.png new file mode 100644 index 00000000..76c1c906 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_arrow_right.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_cast.png b/app/src/main/res/drawable-hdpi/ic_cast.png new file mode 100644 index 00000000..5449b8f5 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_cast.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_cast_connected.png b/app/src/main/res/drawable-hdpi/ic_cast_connected.png new file mode 100644 index 00000000..55bddaad Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_cast_connected.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_check_circle.png b/app/src/main/res/drawable-hdpi/ic_check_circle.png new file mode 100644 index 00000000..ff8c711e Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_check_circle.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_delete.png b/app/src/main/res/drawable-hdpi/ic_delete.png new file mode 100644 index 00000000..e366b650 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_delete.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_edit.png b/app/src/main/res/drawable-hdpi/ic_edit.png new file mode 100644 index 00000000..a3226fb9 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_edit.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_forward_10.png b/app/src/main/res/drawable-hdpi/ic_forward_10.png new file mode 100644 index 00000000..d8763537 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_forward_10.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_fullscreen.png b/app/src/main/res/drawable-hdpi/ic_fullscreen.png new file mode 100644 index 00000000..10decc15 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_fullscreen.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_fullscreen_exit.png b/app/src/main/res/drawable-hdpi/ic_fullscreen_exit.png new file mode 100644 index 00000000..a43ff782 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_fullscreen_exit.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_pause.png b/app/src/main/res/drawable-hdpi/ic_pause.png new file mode 100644 index 00000000..0548a384 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_pause.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_photo.png b/app/src/main/res/drawable-hdpi/ic_photo.png new file mode 100644 index 00000000..b94ccbdb Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_photo.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_play.png b/app/src/main/res/drawable-hdpi/ic_play.png new file mode 100644 index 00000000..556c8729 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_play.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_record.png b/app/src/main/res/drawable-hdpi/ic_record.png new file mode 100644 index 00000000..6b81b67b Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_record.png differ diff --git a/app/src/main/res/drawable-hdpi/ic_replay_10.png b/app/src/main/res/drawable-hdpi/ic_replay_10.png new file mode 100644 index 00000000..7a487849 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_replay_10.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_stop.png b/app/src/main/res/drawable-hdpi/ic_stop.png new file mode 100644 index 00000000..b9164b8b Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_stop.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_upload.png b/app/src/main/res/drawable-hdpi/ic_upload.png new file mode 100644 index 00000000..6edf2fd8 Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_upload.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_add.png b/app/src/main/res/drawable-mdpi/ic_add.png new file mode 100644 index 00000000..a69d739e Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_add.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_arrow_left.png b/app/src/main/res/drawable-mdpi/ic_arrow_left.png new file mode 100644 index 00000000..ceaf25a6 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_arrow_left.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_arrow_right.png b/app/src/main/res/drawable-mdpi/ic_arrow_right.png new file mode 100644 index 00000000..3aec31d3 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_arrow_right.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_cast.png b/app/src/main/res/drawable-mdpi/ic_cast.png new file mode 100644 index 00000000..60bb5f50 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_cast.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_cast_connected.png b/app/src/main/res/drawable-mdpi/ic_cast_connected.png new file mode 100644 index 00000000..b4b72ba3 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_cast_connected.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_check_circle.png b/app/src/main/res/drawable-mdpi/ic_check_circle.png new file mode 100644 index 00000000..9545c7f0 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_check_circle.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_delete.png b/app/src/main/res/drawable-mdpi/ic_delete.png new file mode 100644 index 00000000..554abc4c Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_delete.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_edit.png b/app/src/main/res/drawable-mdpi/ic_edit.png new file mode 100644 index 00000000..f0c1fafa Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_edit.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_forward_10.png b/app/src/main/res/drawable-mdpi/ic_forward_10.png new file mode 100644 index 00000000..751934c3 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_forward_10.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_fullscreen.png b/app/src/main/res/drawable-mdpi/ic_fullscreen.png new file mode 100644 index 00000000..baed2aa2 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_fullscreen.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_fullscreen_exit.png b/app/src/main/res/drawable-mdpi/ic_fullscreen_exit.png new file mode 100644 index 00000000..b4a402c2 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_fullscreen_exit.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_pause.png b/app/src/main/res/drawable-mdpi/ic_pause.png new file mode 100644 index 00000000..48cf98d2 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_pause.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_photo.png b/app/src/main/res/drawable-mdpi/ic_photo.png new file mode 100644 index 00000000..668c633d Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_photo.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_play.png b/app/src/main/res/drawable-mdpi/ic_play.png new file mode 100644 index 00000000..2b425776 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_play.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_record.png b/app/src/main/res/drawable-mdpi/ic_record.png new file mode 100644 index 00000000..eb6412b3 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_record.png differ diff --git a/app/src/main/res/drawable-mdpi/ic_replay_10.png b/app/src/main/res/drawable-mdpi/ic_replay_10.png new file mode 100644 index 00000000..d1a44f69 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_replay_10.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_stop.png b/app/src/main/res/drawable-mdpi/ic_stop.png new file mode 100644 index 00000000..307fe3af Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_stop.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_upload.png b/app/src/main/res/drawable-mdpi/ic_upload.png new file mode 100644 index 00000000..c914b144 Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_upload.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_add.png b/app/src/main/res/drawable-xhdpi/ic_add.png new file mode 100644 index 00000000..4423088d Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_add.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_arrow_left.png b/app/src/main/res/drawable-xhdpi/ic_arrow_left.png new file mode 100644 index 00000000..7217af6e Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_arrow_left.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_arrow_right.png b/app/src/main/res/drawable-xhdpi/ic_arrow_right.png new file mode 100644 index 00000000..714857e2 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_arrow_right.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_cast.png b/app/src/main/res/drawable-xhdpi/ic_cast.png new file mode 100644 index 00000000..fdde81a4 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_cast.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_cast_connected.png b/app/src/main/res/drawable-xhdpi/ic_cast_connected.png new file mode 100644 index 00000000..bcc195dc Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_cast_connected.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_check_circle.png b/app/src/main/res/drawable-xhdpi/ic_check_circle.png new file mode 100644 index 00000000..1b7f7109 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_check_circle.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_delete.png b/app/src/main/res/drawable-xhdpi/ic_delete.png new file mode 100644 index 00000000..5eede98d Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_delete.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_edit.png b/app/src/main/res/drawable-xhdpi/ic_edit.png new file mode 100644 index 00000000..b508285a Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_edit.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_forward_10.png b/app/src/main/res/drawable-xhdpi/ic_forward_10.png new file mode 100644 index 00000000..c5acd34f Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_forward_10.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_fullscreen.png b/app/src/main/res/drawable-xhdpi/ic_fullscreen.png new file mode 100644 index 00000000..7f3b8676 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_fullscreen.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_fullscreen_exit.png b/app/src/main/res/drawable-xhdpi/ic_fullscreen_exit.png new file mode 100644 index 00000000..9c052625 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_fullscreen_exit.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_pause.png b/app/src/main/res/drawable-xhdpi/ic_pause.png new file mode 100644 index 00000000..192399d1 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_pause.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_photo.png b/app/src/main/res/drawable-xhdpi/ic_photo.png new file mode 100644 index 00000000..5165e580 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_photo.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_play.png b/app/src/main/res/drawable-xhdpi/ic_play.png new file mode 100644 index 00000000..7770ab21 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_play.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_record.png b/app/src/main/res/drawable-xhdpi/ic_record.png new file mode 100644 index 00000000..f8c501cd Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_record.png differ diff --git a/app/src/main/res/drawable-xhdpi/ic_replay_10.png b/app/src/main/res/drawable-xhdpi/ic_replay_10.png new file mode 100644 index 00000000..14819ffa Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_replay_10.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_stop.png b/app/src/main/res/drawable-xhdpi/ic_stop.png new file mode 100644 index 00000000..3db50c54 Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_stop.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_upload.png b/app/src/main/res/drawable-xhdpi/ic_upload.png new file mode 100644 index 00000000..904b34ea Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_upload.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_add.png b/app/src/main/res/drawable-xxhdpi/ic_add.png new file mode 100644 index 00000000..94b04c8f Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_add.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_arrow_left.png b/app/src/main/res/drawable-xxhdpi/ic_arrow_left.png new file mode 100644 index 00000000..6220d1ea Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_arrow_left.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_arrow_right.png b/app/src/main/res/drawable-xxhdpi/ic_arrow_right.png new file mode 100644 index 00000000..b6b817dd Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_arrow_right.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_cast.png b/app/src/main/res/drawable-xxhdpi/ic_cast.png new file mode 100644 index 00000000..62fc53f7 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_cast.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_cast_connected.png b/app/src/main/res/drawable-xxhdpi/ic_cast_connected.png new file mode 100644 index 00000000..a28df6de Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_cast_connected.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_check_circle.png b/app/src/main/res/drawable-xxhdpi/ic_check_circle.png new file mode 100644 index 00000000..eaf7a66e Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_check_circle.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_delete.png b/app/src/main/res/drawable-xxhdpi/ic_delete.png new file mode 100644 index 00000000..df20cf5d Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_delete.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_edit.png b/app/src/main/res/drawable-xxhdpi/ic_edit.png new file mode 100644 index 00000000..40ef06e3 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_edit.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_forward_10.png b/app/src/main/res/drawable-xxhdpi/ic_forward_10.png new file mode 100644 index 00000000..ac59544c Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_forward_10.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_fullscreen.png b/app/src/main/res/drawable-xxhdpi/ic_fullscreen.png new file mode 100644 index 00000000..ff166dd0 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_fullscreen.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_fullscreen_exit.png b/app/src/main/res/drawable-xxhdpi/ic_fullscreen_exit.png new file mode 100644 index 00000000..60c521bb Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_fullscreen_exit.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_pause.png b/app/src/main/res/drawable-xxhdpi/ic_pause.png new file mode 100644 index 00000000..3073536d Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_pause.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_photo.png b/app/src/main/res/drawable-xxhdpi/ic_photo.png new file mode 100644 index 00000000..9a0ecd70 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_photo.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_play.png b/app/src/main/res/drawable-xxhdpi/ic_play.png new file mode 100644 index 00000000..0e5abef3 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_play.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_record.png b/app/src/main/res/drawable-xxhdpi/ic_record.png new file mode 100644 index 00000000..256cc26b Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_record.png differ diff --git a/app/src/main/res/drawable-xxhdpi/ic_replay_10.png b/app/src/main/res/drawable-xxhdpi/ic_replay_10.png new file mode 100644 index 00000000..7a43cc57 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_replay_10.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_stop.png b/app/src/main/res/drawable-xxhdpi/ic_stop.png new file mode 100644 index 00000000..98e9f13d Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_stop.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_upload.png b/app/src/main/res/drawable-xxhdpi/ic_upload.png new file mode 100644 index 00000000..44167620 Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_upload.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_all_icon.xml b/app/src/main/res/drawable/bg_all_icon.xml new file mode 100644 index 00000000..e6e37658 --- /dev/null +++ b/app/src/main/res/drawable/bg_all_icon.xml @@ -0,0 +1,4 @@ + + + + \ No newline at end of file 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_channel_overlay_icon.xml b/app/src/main/res/drawable/bg_channel_overlay_icon.xml new file mode 100644 index 00000000..554923d7 --- /dev/null +++ b/app/src/main/res/drawable/bg_channel_overlay_icon.xml @@ -0,0 +1,4 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_copyable_text.xml b/app/src/main/res/drawable/bg_copyable_text.xml new file mode 100644 index 00000000..baac4c1f --- /dev/null +++ b/app/src/main/res/drawable/bg_copyable_text.xml @@ -0,0 +1,13 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_floating_balance.xml b/app/src/main/res/drawable/bg_floating_balance.xml new file mode 100644 index 00000000..56b4ed14 --- /dev/null +++ b/app/src/main/res/drawable/bg_floating_balance.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_floating_reward.xml b/app/src/main/res/drawable/bg_floating_reward.xml new file mode 100644 index 00000000..524e0a45 --- /dev/null +++ b/app/src/main/res/drawable/bg_floating_reward.xml @@ -0,0 +1,5 @@ + + + + + \ 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_list_item.xml b/app/src/main/res/drawable/bg_selected_list_item.xml new file mode 100644 index 00000000..9f32fbbe --- /dev/null +++ b/app/src/main/res/drawable/bg_selected_list_item.xml @@ -0,0 +1,9 @@ + + + + + + + + + \ No newline at end of file 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_small_icon.xml b/app/src/main/res/drawable/bg_small_icon.xml new file mode 100644 index 00000000..c7ac9951 --- /dev/null +++ b/app/src/main/res/drawable/bg_small_icon.xml @@ -0,0 +1,6 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_stream_cost.xml b/app/src/main/res/drawable/bg_stream_cost.xml new file mode 100644 index 00000000..2d0dfb26 --- /dev/null +++ b/app/src/main/res/drawable/bg_stream_cost.xml @@ -0,0 +1,6 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/bg_stream_overlay_icon.xml b/app/src/main/res/drawable/bg_stream_overlay_icon.xml new file mode 100644 index 00000000..4f311895 --- /dev/null +++ b/app/src/main/res/drawable/bg_stream_overlay_icon.xml @@ -0,0 +1,4 @@ + + + + \ No newline at end of file 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/determinate_progress_circle.xml b/app/src/main/res/drawable/determinate_progress_circle.xml new file mode 100644 index 00000000..6a3b2303 --- /dev/null +++ b/app/src/main/res/drawable/determinate_progress_circle.xml @@ -0,0 +1,26 @@ + + + + + + + + + + + + + + + \ 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/drawable/thin_divider.xml b/app/src/main/res/drawable/thin_divider.xml new file mode 100644 index 00000000..ea24b286 --- /dev/null +++ b/app/src/main/res/drawable/thin_divider.xml @@ -0,0 +1,5 @@ + + + + + \ 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_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..800b76a1 --- /dev/null +++ b/app/src/main/res/layout/activity_main.xml @@ -0,0 +1,33 @@ + + + + + + + + + + + + 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..04fddc26 --- /dev/null +++ b/app/src/main/res/layout/activity_verification.xml @@ -0,0 +1,46 @@ + + + + + + + + + + + + + \ 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..212efa4c --- /dev/null +++ b/app/src/main/res/layout/app_bar_main.xml @@ -0,0 +1,293 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/card_invites_by_email.xml b/app/src/main/res/layout/card_invites_by_email.xml new file mode 100644 index 00000000..e11fe386 --- /dev/null +++ b/app/src/main/res/layout/card_invites_by_email.xml @@ -0,0 +1,62 @@ + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/card_invites_by_link.xml b/app/src/main/res/layout/card_invites_by_link.xml new file mode 100644 index 00000000..82853634 --- /dev/null +++ b/app/src/main/res/layout/card_invites_by_link.xml @@ -0,0 +1,101 @@ + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/card_invites_history.xml b/app/src/main/res/layout/card_invites_history.xml new file mode 100644 index 00000000..7965d4e6 --- /dev/null +++ b/app/src/main/res/layout/card_invites_history.xml @@ -0,0 +1,45 @@ + + + + + + + + + + + + \ 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..3468bdad --- /dev/null +++ b/app/src/main/res/layout/card_wallet_balance.xml @@ -0,0 +1,295 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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..dc340eaf --- /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..3a67e1d5 --- /dev/null +++ b/app/src/main/res/layout/card_wallet_recent_transactions.xml @@ -0,0 +1,91 @@ + + + + + + + + + + + + + + + + + + + + + + + \ 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..9ba7e51d --- /dev/null +++ b/app/src/main/res/layout/card_wallet_sync.xml @@ -0,0 +1,130 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/container_inline_channel_form.xml b/app/src/main/res/layout/container_inline_channel_form.xml new file mode 100644 index 00000000..adad2a95 --- /dev/null +++ b/app/src/main/res/layout/container_inline_channel_form.xml @@ -0,0 +1,120 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/layout/container_inline_tag_form.xml b/app/src/main/res/layout/container_inline_tag_form.xml new file mode 100644 index 00000000..bdd32a3c --- /dev/null +++ b/app/src/main/res/layout/container_inline_tag_form.xml @@ -0,0 +1,78 @@ + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/container_nothing_at_location.xml b/app/src/main/res/layout/container_nothing_at_location.xml new file mode 100644 index 00000000..fda6b591 --- /dev/null +++ b/app/src/main/res/layout/container_nothing_at_location.xml @@ -0,0 +1,41 @@ + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/container_sdk_initializing.xml b/app/src/main/res/layout/container_sdk_initializing.xml new file mode 100644 index 00000000..63f32959 --- /dev/null +++ b/app/src/main/res/layout/container_sdk_initializing.xml @@ -0,0 +1,31 @@ + + + + + + + \ 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..2e823833 --- /dev/null +++ b/app/src/main/res/layout/content_main.xml @@ -0,0 +1,139 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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_scope.xml b/app/src/main/res/layout/dialog_content_scope.xml new file mode 100644 index 00000000..cffafa04 --- /dev/null +++ b/app/src/main/res/layout/dialog_content_scope.xml @@ -0,0 +1,94 @@ + + + + + + + + + + + + + + + + + \ 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/dialog_customize_tags.xml b/app/src/main/res/layout/dialog_customize_tags.xml new file mode 100644 index 00000000..b19a9ff9 --- /dev/null +++ b/app/src/main/res/layout/dialog_customize_tags.xml @@ -0,0 +1,106 @@ + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/dialog_discover.xml b/app/src/main/res/layout/dialog_discover.xml new file mode 100644 index 00000000..e9a32daf --- /dev/null +++ b/app/src/main/res/layout/dialog_discover.xml @@ -0,0 +1,54 @@ + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/dialog_repost_claim.xml b/app/src/main/res/layout/dialog_repost_claim.xml new file mode 100644 index 00000000..7cbed97f --- /dev/null +++ b/app/src/main/res/layout/dialog_repost_claim.xml @@ -0,0 +1,190 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/layout/dialog_send_tip.xml b/app/src/main/res/layout/dialog_send_tip.xml new file mode 100644 index 00000000..6912778a --- /dev/null +++ b/app/src/main/res/layout/dialog_send_tip.xml @@ -0,0 +1,125 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/exo_playback_control_view.xml b/app/src/main/res/layout/exo_playback_control_view.xml new file mode 100644 index 00000000..235ae4b1 --- /dev/null +++ b/app/src/main/res/layout/exo_playback_control_view.xml @@ -0,0 +1,181 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/floating_wallet_balance.xml b/app/src/main/res/layout/floating_wallet_balance.xml new file mode 100644 index 00000000..2ccb21c2 --- /dev/null +++ b/app/src/main/res/layout/floating_wallet_balance.xml @@ -0,0 +1,94 @@ + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_about.xml b/app/src/main/res/layout/fragment_about.xml new file mode 100644 index 00000000..5f5e3fb9 --- /dev/null +++ b/app/src/main/res/layout/fragment_about.xml @@ -0,0 +1,408 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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..b1e8ef79 --- /dev/null +++ b/app/src/main/res/layout/fragment_all_content.xml @@ -0,0 +1,201 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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..bfac6090 --- /dev/null +++ b/app/src/main/res/layout/fragment_channel.xml @@ -0,0 +1,223 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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..a572155e --- /dev/null +++ b/app/src/main/res/layout/fragment_channel_about.xml @@ -0,0 +1,97 @@ + + + + + + + + + + + + + + + + + + + + + + + + + \ 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..9c30d04f --- /dev/null +++ b/app/src/main/res/layout/fragment_channel_content.xml @@ -0,0 +1,119 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_channel_form.xml b/app/src/main/res/layout/fragment_channel_form.xml new file mode 100644 index 00000000..006a1724 --- /dev/null +++ b/app/src/main/res/layout/fragment_channel_form.xml @@ -0,0 +1,348 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_channel_manager.xml b/app/src/main/res/layout/fragment_channel_manager.xml new file mode 100644 index 00000000..f53e3429 --- /dev/null +++ b/app/src/main/res/layout/fragment_channel_manager.xml @@ -0,0 +1,75 @@ + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_editors_choice.xml b/app/src/main/res/layout/fragment_editors_choice.xml new file mode 100644 index 00000000..42e03e9a --- /dev/null +++ b/app/src/main/res/layout/fragment_editors_choice.xml @@ -0,0 +1,20 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_file_view.xml b/app/src/main/res/layout/fragment_file_view.xml new file mode 100644 index 00000000..9a61879a --- /dev/null +++ b/app/src/main/res/layout/fragment_file_view.xml @@ -0,0 +1,641 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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..48fd73fe --- /dev/null +++ b/app/src/main/res/layout/fragment_following.xml @@ -0,0 +1,203 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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_invites.xml b/app/src/main/res/layout/fragment_invites.xml new file mode 100644 index 00000000..67efda37 --- /dev/null +++ b/app/src/main/res/layout/fragment_invites.xml @@ -0,0 +1,85 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_library.xml b/app/src/main/res/layout/fragment_library.xml new file mode 100644 index 00000000..52df464d --- /dev/null +++ b/app/src/main/res/layout/fragment_library.xml @@ -0,0 +1,356 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_publish.xml b/app/src/main/res/layout/fragment_publish.xml new file mode 100644 index 00000000..fcc19830 --- /dev/null +++ b/app/src/main/res/layout/fragment_publish.xml @@ -0,0 +1,179 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_publish_form.xml b/app/src/main/res/layout/fragment_publish_form.xml new file mode 100644 index 00000000..7d24d34f --- /dev/null +++ b/app/src/main/res/layout/fragment_publish_form.xml @@ -0,0 +1,579 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_publishes.xml b/app/src/main/res/layout/fragment_publishes.xml new file mode 100644 index 00000000..e3af5bb5 --- /dev/null +++ b/app/src/main/res/layout/fragment_publishes.xml @@ -0,0 +1,75 @@ + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_rewards.xml b/app/src/main/res/layout/fragment_rewards.xml new file mode 100644 index 00000000..46838086 --- /dev/null +++ b/app/src/main/res/layout/fragment_rewards.xml @@ -0,0 +1,154 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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_transaction_history.xml b/app/src/main/res/layout/fragment_transaction_history.xml new file mode 100644 index 00000000..2762311d --- /dev/null +++ b/app/src/main/res/layout/fragment_transaction_history.xml @@ -0,0 +1,31 @@ + + + + + + + + + + \ 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_manual.xml b/app/src/main/res/layout/fragment_verification_manual.xml new file mode 100644 index 00000000..5e43ceae --- /dev/null +++ b/app/src/main/res/layout/fragment_verification_manual.xml @@ -0,0 +1,59 @@ + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/fragment_verification_phone.xml b/app/src/main/res/layout/fragment_verification_phone.xml new file mode 100644 index 00000000..6048ef33 --- /dev/null +++ b/app/src/main/res/layout/fragment_verification_phone.xml @@ -0,0 +1,157 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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..61e5e0ae --- /dev/null +++ b/app/src/main/res/layout/fragment_wallet.xml @@ -0,0 +1,100 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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..0469c640 --- /dev/null +++ b/app/src/main/res/layout/list_item_channel.xml @@ -0,0 +1,174 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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..63970ca2 --- /dev/null +++ b/app/src/main/res/layout/list_item_channel_filter.xml @@ -0,0 +1,60 @@ + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_editors_choice.xml b/app/src/main/res/layout/list_item_editors_choice.xml new file mode 100644 index 00000000..c616dea1 --- /dev/null +++ b/app/src/main/res/layout/list_item_editors_choice.xml @@ -0,0 +1,66 @@ + + + + + + + + + + + + + + + + + \ 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..0804a85d --- /dev/null +++ b/app/src/main/res/layout/list_item_featured_search_result.xml @@ -0,0 +1,194 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_gallery.xml b/app/src/main/res/layout/list_item_gallery.xml new file mode 100644 index 00000000..d18d75ed --- /dev/null +++ b/app/src/main/res/layout/list_item_gallery.xml @@ -0,0 +1,28 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/list_item_invitee.xml b/app/src/main/res/layout/list_item_invitee.xml new file mode 100644 index 00000000..57c3f597 --- /dev/null +++ b/app/src/main/res/layout/list_item_invitee.xml @@ -0,0 +1,37 @@ + + + + + + + + + \ 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_reward.xml b/app/src/main/res/layout/list_item_reward.xml new file mode 100644 index 00000000..fec233be --- /dev/null +++ b/app/src/main/res/layout/list_item_reward.xml @@ -0,0 +1,139 @@ + + + + + + + + + + + + + + + + + + + + + + \ 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..04f785a8 --- /dev/null +++ b/app/src/main/res/layout/list_item_stream.xml @@ -0,0 +1,224 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ 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..9c461b53 --- /dev/null +++ b/app/src/main/res/layout/list_item_suggested_channel.xml @@ -0,0 +1,69 @@ + + + + + + + + + + + \ 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..9881babe --- /dev/null +++ b/app/src/main/res/layout/list_item_tag.xml @@ -0,0 +1,31 @@ + + + + + \ 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..c30ccbf5 --- /dev/null +++ b/app/src/main/res/layout/nav_header_main.xml @@ -0,0 +1,71 @@ + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/layout/spinner_item_channel.xml b/app/src/main/res/layout/spinner_item_channel.xml new file mode 100644 index 00000000..bcc2f763 --- /dev/null +++ b/app/src/main/res/layout/spinner_item_channel.xml @@ -0,0 +1,14 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/spinner_item_generic.xml b/app/src/main/res/layout/spinner_item_generic.xml new file mode 100644 index 00000000..fc84a1ee --- /dev/null +++ b/app/src/main/res/layout/spinner_item_generic.xml @@ -0,0 +1,16 @@ + + + + \ No newline at end of file diff --git a/app/src/main/res/menu/menu_claim_list.xml b/app/src/main/res/menu/menu_claim_list.xml new file mode 100644 index 00000000..968f7dc6 --- /dev/null +++ b/app/src/main/res/menu/menu_claim_list.xml @@ -0,0 +1,16 @@ + + + + + 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-af/strings.xml b/app/src/main/res/values-af/strings.xml new file mode 100644 index 00000000..8f38c803 --- /dev/null +++ b/app/src/main/res/values-af/strings.xml @@ -0,0 +1,565 @@ + + LBRY + Maak navigasie-laai oop + Maak navigasielade toe + + + Soek content + U content + Beursie + Volgende + Redakteur se keuse + U etikette + Alle Content + Kanale + Biblioteek + Gepubliseer + Nuut Gepubliseer + Belonings + Uitnodigings + Instellings + Oor + Meld aan + Programstart het misluk. Kontroleer u dataverbinding en probeer weer. Stuur \'n e-pos na hello@lbry.com as hierdie probleem voortduur + Geen content om op die oomblik te vertoon nie. Herdefinieer u keuse of kyk later weer. + + + Welkom tot LBRY. + LBRY is \'n gemeenskapsgebaseerde contentsplatform waar je video\'s, musiek, boeke en meer kan vind en publiseer + Deur voort te gaan, stem ek in tot die <a href=\"https://lbry.com/termsofservice\">TDiensvoorwaardes</a> en bevestig dat ek ouer as 13 is. + Wag asseblief terwyl ons \'n paar dinge gereed maak... + Gebruik LBRY » + + + Soek films, musiek en meer + + + Vind Kanale om te volg + Kanale wat u volg + Vind + LBRY werk beter as u ten minste 5 skeppers volg waarvan u hou. Meld aan om skeppers wat u volg te wys as u reeds \'n account het. + Kies tot 5 skeppers om voort te gaan. + %1$doorblywende... + Gedoen + Almal + Ontdek nuwe kanale + + + Anoniem + + + Etikette + Deel + Publiseer + Fooi + Wysig + Verwyder + Aflaai + Oop + Rapporteer + Laai gedesentraliseerde data... + Verwante Content + Deel LBRY-inhoud + Beskou + Speel + Ongesteunde content + Jammer, ons kan nie die inhoud in die app vertoon nie. U kan die lêer %1$sin u aflaai-lêergids vind. + Daar is niks op hierdie plek nie. + Publiseer iets hier + Daar is tans geen toegang tot hierdie inhoud nie. Probeer asseblief weer later. + 0:00 + Die lêer by \"%1$s\" bestaan nie. + Bevestig die aankoop + Vee lêer uit + Is u seker dat u hierdie lêer van u toestel wil verwyder? + Kon nie laai nie %1$s. Probeer asseblief weer later. + Daar is tans geen rolverdeling beskikbaar nie. + Vee inhoud uit? + Is u seker dat u hierdie inhoud wil publiseer? Geen lêers sal van u toestel verwyder word nie. + Die inhoud is suksesvol uit die blockchain verwyder. + Die inhoud kon tans nie uitgevee word nie. Probeer asseblief weer later. + + %1$s uitsig + %1$s uitsig + + + Dit sal \"%1$s\" vir %2$s krediete koop + Dit sal \"%1$s\" vir %2$s krediete koop + + + + Daar is nog niks hier nie.\nProbeer asseblief later weer. + Content + Webwerf + gepubliseer + + %1$s volgeling + %1$s volgelinge + + + + Rekord + Neem \'n Foto + Laai \'n lêer op + Ons kon geen video\'s op u toestel vind nie. Neem \'n foto of neem \'n video op om aan die gang te kom. + Wag asseblief terwyl ons jou video\'s laai ... + LBRY benodig toegang om u video\'s, beelde en ander lêers vanaf u toestel te kan vertoon en publiseer. + LBRY benodig toegang tot u kamera om video\'s op te neem. + LBRY benodig toegang tot u kamera om foto\'s te neem. + Redigeer inhoud + Volwasse etikette + Prys + U content sal gratis wees. Druk op die skakelaar om \'n prys in te stel. + Contentadres + Dit enige + Adres + Die adres waar mense u content kan vind (bv. lbry://myvideo) + Lisensie + Lisensie beskrywing + Bykomende Opsies + Toon ekstra velde + Steek ekstra velde weg + Geen lêer gevind om te publiseer nie. + Videooptimalisering + \'N Kleinkiekie kan nie outomaties vanuit u inhoudslêer geskep word nie. + U video word geoptimaliseer vir beter ondersteuning op \'n wye verskeidenheid toestelle. U kan die oorblywende velde hieronder invul terwyl dit aan die gang is. + U video is suksesvol geoptimaliseer vir beter afspeel op soveel toestelle as moontlik. Gaan asseblief voort om u inhoud te publiseer. + U video kon nie geoptimaliseer word nie. Die lêer sal sonder veranderinge opgelaai word. + Voltooide Videoduur: %1$s + U kan nie op die oomblik inhoud publiseer nie, omdat die agtergronddiens steeds initialiseer. + U inhoud is suksesvol gepubliseer. Dit kan \'n paar oomblikke neem om op die blockchain te verskyn. + Videooptimalisering is aan die gang. Druk Annuleer onderaan die bladsy as u wil kanselleer. + Videooptimalisering is aan die gang. + Daar is geen kamera-app beskikbaar om video\'s op hierdie toestel op te neem nie. + Daar is geen kamera-app beskikbaar om foto\'s op hierdie toestel te neem nie. + + Verskaf \'n titel. + Spesifiseer \'n adres waar mense u inhoud kan vind. + U inhoudadres bevat ongeldige karakters. + U het reeds na die gespesifiseerde inhoudsadres gepubliseer. Voer \'n nuwe adres in. + Geen lêer gekies nie. Kies \'n video of neem \'n foto, of kies \'n lêer voordat u dit publiseer. + Voer \'n prys in of skakel die skakelaar af om u inhoud gratis te maak. + Kies \'n kleinkiekie om op te laai voordat u dit publiseer. + Wag tot die miniatuur klaar is met die oplaai voordat u dit publiseer. + + Taal + Engels + Sjinees + Frans + Duits + Japannees + Russiese + Spaans + Indonesiese + Italiaanse + Nederlands + Turks + Pools + Maleis + Portugees + Viëtnamees + Thai + Arabies + Tsjeggiese + Kroaties + Kambodja + Koreaanse + Noorse + Roemeens + Hindi + Griekse + + Geen + Publieke Domein + Kopiereg + Creative Commons Attribution 4.0 International + Creative Commons Attribution-ShareAlike 4.0 International + Creative Commons Attribution-NoDerivatives 4.0 International + Creative Commons Attribution-NonCommercial 4.0 International + Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International + Creative Commons Attribution-NonCommercial-NoDerivatives 4.0 International + + + LBC + USD + + + + Een of meer inhouditems kon tans nie uitgevee word nie. Probeer asseblief weer later. + + Is u seker dat u die geselekteerde inhouditems wil uitvee? + Is u seker dat u die geselekteerde inhouditems wil uitvee? + + + Die inhouditems is suksesvol uitgevee. + Die inhouditems is suksesvol uitgevee. + + + + Oeps! Iets het verkeerd geloop. + Laai installasie-ID. + Laai plaaslike bekende en volg merkers. + Laai LBC/USD wisselkoers. + Gebruiker geverifieer. + Installasie geregistreer. + Gelaaide intekeninge. + Ontbinde intekeninge. + + + Inhoud & Gebruikerskoppelvlak + Ander + Aktiveer donker tema + Wys volwasse inhoud + Wys URL-voorstelle + Kennisgewings + Abonnementen + Contentsbelangstellings + + Hou die LBRY-diens op die agtergrond vir verbeterde beursie- en netwerkprestasie + Neem deel aan die datanetwerk (benodig weer app en agtergronddiens) + + + %1$s - Soek + %1$s - Etikette + Soek vir \'%1$s\' + Verken content \'%1$s\' etikette + Kyk na inhoud by %1$s + Kyk na die %1$s kanaal + + + Geen resultate gevind vir \'%1$s\' nie. Voer \'n ander soekterm in. + U kan alles soek, insluitend films, musiek, e-boeke, sagteware en meer. + Geen verwante inhoud op die oomblik nie. + + + Balans + U het tans + U kan u krediete in USD omskakel en die omgeskakelde bedrag met \'n uitruil onttrek. <a href=\"https://lbry.com/faq/exchanges\">Learn more</a>. + <a href=\"https://bittrex.com/Account/Register?referralCode=4M1-P30-BON\">Skakel krediete na USD op Bittrex</a> + Jy het ook + Jy het vasgehou + in wenke + in u publikasies + in u steun + Verdien meer wenke deur cool video\'s op te laai + Die agtergronddiens initialiseer... + Die agtergronddiens word steeds geïnitialiseer. U kan die inhoud intussen verken en kyk. + U kan dit nie nou doen nie, want die agtergronddiens is nog besig om te initialiseer. + + \'N Rugsteun van u beursie word met lbry.tv gesinkroniseer + U beursie is tans nie met lbry.tv gesinkroniseer nie. U is verantwoordelik vir die rugsteun van u beursie. + <a href=\"https://lbry.com/faq/account-sync\">Wat beteken dit?</a> + <a href=\"https://lbry.com/faq/how-to-backup-wallet#android\">Wat beteken dit?</a></a> + + Krediete Ontvang + Gebruik hierdie beursie-adres om krediete te ontvang wat deur \'n ander gebruiker (of u self) gestuur is. + Kry nuwe adres + U kan enige tyd \'n nuwe adres genereer, en alle vorige adresse sal aanhou werk. Die gebruik van verskeie adresse kan nuttig wees om inkomende betalings uit verskeie bronne dop te hou. + + Stuur Krediete + Ontvanger adres + bbFxRyXXXXXXXXXXXZD8nE7XTLUxYnddTs + Bedrag + Stuur + + Onlangse Transaksies + Sien alles + Ontvang + Uitgereik + Publiseer + Ondersteuning + Verlaat + Kanaal + Kanaalopdatering + Publiseer Opdatering + + Wallet-sinkronisering + Sinkroniseer status + Op + Af + E-pos + Gekoppelde e-posadres + <a href=\"https://lbry.com/faq/how-to-backup-wallet#android\">Handmatige rugsteun</a> + <a href=\"https://lbry.com/faq/how-to-backup-wallet#sync\">Sinkroniseer FAQ</a> + 0 + LBC + + Account Aanbeveel + Met \'n lbry.tv-account kan je belonings verdien, je beursie rugsteun en alles sinchroniseer. + Sonder u account aanvaar u alle verantwoordelikheid om u beursie- en LBRY-data te beveilig. + Slaan Account Oor + Teken Aan + Adres gekopieër + + Voer \'n geldige adres in om krediete te stuur + Onvoldoende balans + Voer \'n geldige bedrag in + U krediete kon tans nie gestuur word nie. Probeer asseblief weer later. + Laai tans transaksies... + Daar is geen onlangse transaksies wat vertoon kan word nie. + Daar is tans geen transaksies wat vertoon kan word nie. + belasting %1$s + Transaksiegeskiedenis + + + U het %1$s krediete gestuur + U het %1$s krediete gestuur + + + + Pasmaak u etikette + Sorteer content volgens + Content van + Gewilde content + Nuwe content + Top content + Gewild + Top + Nuwe + Die afgelope 24 uur + Die afgelope week + Die afgelope maand + Verlede jaar + Van alle tye + van + vir + Filter vir + Almal + Etikette wat u volg + Aanpassen + Die geselekteerde aansig is nog nie beskikbaar nie. + Dit lyk asof u nog geen merkers gevolg het nie. + Soek vir meer etikette + U het nog geen merkers gevolg nie. Begin deur merkers by te voeg waarin u belangstel! + Ons kon nie nuwe merkers vind wat u nie volg nie. + Die \'%1$s\' tag is reeds bygevoeg. + U kan nie meer as 5 merkers byvoeg nie. + Stuur \'n wenk + Stuur \'n wenk aan %1$s + Dit sal verskyn as \'n wenk vir %1$s, wat die vermoë verhoog om ontdek te word terwyl dit aktief is. <a href=\"https://lbry.com/faq/tipping\">Learn more</a>. + Dit sal verskyn as \'n wenk vir %1$s, wat die vermoë verhoog om ontdek te word terwyl dit aktief is. <a href=\"https://lbry.com/faq/tipping\">Learn more</a>. + Kanselleer + Publiseer %1$s + Herpos u gunsteling inhoud om meer mense te help om dit te ontdek! + Kanaal om op te plaas + Wys gevorderd + Versteek gevorderd + Naam + 0.001 + Die inhoud is suksesvol weer gepos! + Die herposnaam bevat ongeldige karakters. + + U het %1$s krediete as \'n fooi gestuur, Mahalo! + U het %1$s krediete as \'n fooi gestuur, Mahalo! + + + + Verskaf \'n e-pos adres. + you@example.com + \'N E-pos is gestuur na + Klik op die skakel in die boodskap om aan te meld. + Herstuur + Gaan aan + Voer \'n geldige e-pos adres in + Volg die aanwysings in die e-pos wat na u adres gestuur word om voort te gaan. + U het suksesvol by lbry.tv aangemeld + Haal rekeninginligting op... + Pas beursie-data toe... + Voer die wagwoord in wat u gebruik het om u beursie te beveilig. + Voer \'n wagwoord in om u beursie te beveilig. + Opmerking: vir beursie-beveiligingsdoeleindes kan LBRY nie u wagwoord terugstel nie. + Wagwoord + Aktiveer sinkronisering + Die beursie-sinkronisering kon tans nie voltooi word nie. Probeer asseblief weer later. As hierdie probleem voortduur, stuur \'n e-pos na hello@lbry.com. + Telefoon Nommer + Voer u telefoonnommer in. + Stel nie belang nie + Handmatige Beloningverifikasie + Hierdie rekening moet hersien word voordat u aan die beloningsprogram kan deelneem. Dit kan vanaf \'n paar minute tot \'n paar dae duur. + As u aanhou om hierdie boodskap te sien, versoek u dat dit op die <a href=\"https://discordapp.com/invite/Z3bERWA\">LBRY Discord server</a>. + Geniet intussen gratis inhoud! + Verifieer Telefoonnommer + Voer die verifikasiekode in wat na %1$s gestuur is + 0000 + Verifieer + Voer \'n geldige telefoonnommer in. + Voer die verifikasiekode in wat na u telefoonnommer gestuur is. + + + U het nog geen merkers bygevoeg nie. Voeg etikette by om die ontdekking te verbeter. + Ons kon nie nuwe merkers vind wat nog nie bygevoeg is nie. + + + U het nie \'n kanaal geskep nie.\nBegin nou deur \'n nuwe kanaal te skep! + Skep \'n kanaal + Skep \'n kanaal... + Wysig kanaal + Vee seleksie uit? + Vee kanaal uit? + Is u seker dat u hierdie kanaal wil uitvee? + Die kanaal is suksesvol uitgevee. + Die kanaal kon tans nie uitgevee word nie. Probeer asseblief weer later. + Beskrywing + Ja + Nee + Toon opsionele velde + Steek opsionele velde weg + Save + Kanaalnaam + Titel + \@ + Deposito + Hierdie LBC bly joune. Dit is \'n deposito om die naam te bespreek en kan te eniger tyd ongedaan gemaak word. + LBRY benodig toegang om inhoud op u toestel af te laai. + LBRY benodig toegang om prente vanaf u toestelberging te laai. + Kies kleinkiekie + Kies voorblad + Die lêerpad kon nie vir die geselekteerde prent bepaal word nie. Kies \'n prent op \'n ander plek. + Wag tot die huidige oplaai voltooi is. + Die beeldoplaai-versoek het misluk. Probeer asseblief weer. + Oplaai... + Voer \'n kanaalnaam in. + Jou kanaalnaam bevat ongeldige karakters. + U het reeds \'n kanaal met dieselfde naam geskep. + Voer \'n geldige depositobedrag in. + Deposito kan nie hoër wees as u saldo nie. + Die kanaalstoorversoek het misluk. Probeer asseblief weer. + Die kanaal is suksesvol gestoor. + Die eis is hangende gepubliseer op die blockchain. U kan binne enkele oomblikke toegang tot die eis verkry of dit wysig. + Hangende + Skep + Een of meer kanale kon tans nie uitgevee word nie. Probeer asseblief weer later. + + \'N Minimum deposito van %1$s krediete word vereis. + \'N Minimum deposito van %1$s krediete word vereis. + + + Is u seker dat u die geselekteerde kanale wil uitvee? + Is u seker dat u die geselekteerde kanale wil uitvee? + + + Die kanale is suksesvol uitgevee. + Die kanale is suksesvol uitgevee. + + + + Met LBRY-krediete kan u inhoud publiseer of koop. + U kan gratis krediete ter waarde van $%1$s verwerf nadat u \'n e-posadres verstrek het. + <a href=\"https://lbry.com/faq/earn-credits\">Leer meer</a>. + Begin + abc123 + Eis + Voer \'n pasgemaakte beloningskode in om te eis. + Onopgeëiste + Pasgemaakte Kode + tot + Is u \'n supermodel of rockster wat \'n persoonlike beloningskode ontvang het? Eis dit hier op. + + U het %1$s krediete as beloning geëis. + U het %1$s krediete as beloning geëis. + + + %1$s beskikbare krediete + %1$s beskikbare krediete + + + + LBRY Uitnodigingsprogram + U kan ekstra krediete verdien vir elke persoon wat u uitnooi om LBRY te gebruik. + <a href=\"https://lbry.com/faq/invites\">Leer meer</a>. + Nooi Skakel Uit + Deel hierdie skakel met vriende (of vyande) en kry krediete wanneer hulle by lbry.tv aansluit. + U nooi skakel + Pasmaak nooi skakel + Nooi per e-pos uit + Nooi iemand wat u ken per e-pos en verdien krediete wanneer hulle by lbry.tv aansluit. + imaginary@friend.com + Uitnodiging + Uitnodigingsgeskiedenis + Verdien krediete vir die uitnodiging van \'n vriend, \'n vyand, \'n frenemy of \'n vriend. Almal het vryheid van inhoud nodig. + Beloning + Beweer + Geëis + Unclaimable + Nooi skakel gekopieër. + Uitnodiging gestuur na %1$s + + + Downloads + Aankope + Geskiedenis + U het geen inhoud op hierdie toestel afgelaai nie. + U het geen inhoud op hierdie toestel gesien nie. + U het geen inhoud op u rekening gekoop nie. + Verberg + Statistiek + Video + Audio + Beelde + MB + KB + GB + 0MB + + Is u seker dat u die geselekteerde lêers van u toestel wil verwyder? + Is u seker dat u die geselekteerde lêers van u toestel wil verwyder? + + + Die lêer is suksesvol uitgevee. + Die lêers is suksesvol uitgevee. + + + + Oor LBRY + Contentvryheid + LBRY is \'n gratis, oop en gemeenskapsbeheerde digitale mark. Dit is \'n gedesentraliseerde peer-to-peer-contentverspreidingsplatform vir skeppers om content op te laai en te deel, en verdien LBRY-krediete vir hul moeite. Gebruikers kan \'n wye verskeidenheid video\'s, musiek, e-boeke en ander digitale content waarin hulle belangstel vind. + Word Sosiaal + U kan interaksie met die LBRY-span en lede van die gemeenskap op Discord, Facebook, Instagram, Twitter of Reddit. + Programinligting + Laai... + <a href=\"https://lbry.com/faq/what-is-lbry\">Wat is LBRY?</a> + <a href=\"https://lbry.com/faq/android-basics\">Android Basics</a> + <a href=\"https://lbry.com/faq\">FAQ</a> + <a href=\"https://discordapp.com/invite/Z3bERWA\">Discord</a> + <a href=\"https://www.facebook.com/LBRYio\">Facebook</a> + <a href=\"https://www.instagram.com/LBRYio\">Instagram</a> + <a href=\"https://reddit.com/r/lbry\">Reddit</a> + <a href=\"https://t.me/lbryofficial\">Telegram</a> + <a href=\"https://twitter.com/LBRYio\">Twitter</a> + Dateer posvoorkeure op + Programweergawe + LBRY SDK + Platform + Installasie ID + Firebase Token + Logs + Stuur log + Gekoppelde e-posadres + Onbekend + Die lbrynet.log-lêer kon nie gevind word nie. + Die lbrynet.log-lêer kan nie gedeel word nie weens toestemmingsbeperkings. + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/values-es/strings.xml b/app/src/main/res/values-es/strings.xml new file mode 100644 index 00000000..72bfc7d8 --- /dev/null +++ b/app/src/main/res/values-es/strings.xml @@ -0,0 +1,578 @@ + + LBRY + Abrir menú de navegación + Cerrar menú de navegación + + + Explorar Contenido + Tu Contenido + Cartera + Siguiendo + Elección del Editor + Tus Etiquetas + Todo el Contenido + Canales + Biblioteca + Publicaciones + Nueva Publicación + Recompensas + Invitaciones + Ajustes + Acerca de + Registrarse +   +Error al iniciar la aplicación. Verifique su conexión de datos e inténtelo nuevamente. Si este problema persiste, por favor envíe un correo electrónico a hello@lbry.com + No hay contenido para mostrar en este momento. Por favor, refine su selección o realice la consulta más tarde. + + + Bienvenido a LBRY. + LBRY es una plataforma de contenido controlada por la comunidad donde puedes encontrar y publicar videos, música, libros y más. + Al continuar, acepto los <a href=\"https://lbry.com/termsofservice\"> Términos de servicio </a> y confirmo que tengo más de 13 años. + Por favor espere mientras preparamos algunas cosas... + Usa LBRY » + + + Busca vídeos, música y más + + + Busca Canales para seguir + Canales que sigues + Descubre + LBRY funciona mejor si sigues al menos a 5 creadores que te gustan. Inicie sesión para mostrar los creadores que sigue si ya tiene una cuenta. + Por favor selecciona hasta 5 creadores para continuar. + %1$drestante... + Hecho + Todo + Descubre nuevos canales + + + Anónimo + + + Etiquetas + Compartir + Republicar + Propina + Editar + Eliminar + Descargar + Abrir + Reportar + Cargando datos descentralizados... + Contenido Relacionado + Compartir contenido LBRY + Ver + Reproducir + Contenido no admitido + Lo sentimos, no podemos mostrar este contenido en la aplicación.Puedes encontrar el archivo%1$sen tu carpeta de descargas. +   +No hay nada en este lugar. + Publica algo aquí + No se puede acceder a este contenido en este momento. Por favor, inténtelo de nuevo más tarde. + 0:00 + El archivo en \"%1$s\" no existe. + Confirmar compra + Eliminar archivo + ¿Seguro que quieres eliminar este archivo de tu dispositivo? + Falló al cargar%1$s. Por favor, inténtelo de nuevo más tarde. + No hay sesión de reparto disponible en este momento. + ¿Eliminar contenido? +   +¿Estás seguro de que deseas publicar este contenido? No se eliminarán archivos de su dispositivo. +   +El contenido se eliminó con éxito de la cadena de bloques. +   +El contenido no se pudo eliminar en este momento. Por favor, inténtelo de nuevo más tarde. +  + + %1$svista + %1$s vistas + + + Esto comprará \"%1$s\" para %2$s crédito + Esto comprará \"%1$s\" para %2$s créditos + + + + Todavía no hay nada aquí. \ Por favor, vuelva más tarde. + Contenido + Página web + repostear + + %1$s seguidor + %1$s seguidores + + + +   +Grabar + Toma una foto + Cargar un archivo +   +No pudimos encontrar ningún video en su dispositivo. Tome una foto o grabe un video para comenzar. + Espera mientras cargamos tus videos ... + LBRY requiere acceso para poder mostrar y publicar sus videos, imágenes y otros archivos desde su dispositivo. + LBRY requiere acceso a su cámara para grabar videos. + LBRY requiere acceso a su cámara para tomar fotos. + Contenido editado + Etiquetas maduras + Precio + Tu contenido será gratis. Presiona el botón para establecer un precio. + Dirección de contenido + Aleatorizar + Dirección + La dirección donde las personas pueden encontrar su contenido (ex. lbry://myvideo) + Licencia + Descripción de la licencia + Opciones adicionales + Mostrar campos extras + Ocultar campos extras + No se ha encontrado ningún archivo para publicar. + Optimización de video + No se pudo crear una miniatura automáticamente desde su archivo de contenido. + Su video está siendo optimizado para un mejor soporte en una amplia gama de dispositivos. Puede completar los campos restantes a continuación mientras está en progreso. +   +Su video fue optimizado con éxito para una mejor reproducción en tantos dispositivos como sea posible. Por favor, proceda a publicar su contenido. +   +Tu video no pudo ser optimizado. El archivo se cargará sin cambios. + Duración completa del video: %1$s +   +No puede publicar contenido en este momento porque el servicio en segundo plano aún se está inicializando. +   +Su contenido fue publicado con éxito. Puede tardar unos minutos en aparecer en la cadena de bloques. + La optimización de video está en progreso. Si desea cancelar, presione Cancelar en la parte inferior de la página. + La optimización de video está en progreso. + No hay una aplicación de cámara disponible para grabar videos en este dispositivo. + No hay una aplicación de cámara disponible para tomar fotos en este dispositivo. + + Por favor proporcione un título. + Especifique una dirección donde las personas puedan encontrar su contenido. + Su dirección de contenido contiene caracteres no válidos. + Ya ha publicado en la dirección de contenido especificada. Por favor ingrese una nueva dirección. + Ningún archivo seleccionado. Por favor, elija un video, tome una foto, o seleccione un archivo antes de publicarlo. + Ingrese un precio o desactive la palanca para liberar su contenido. + Seleccione una miniatura para cargar antes de publicar. + Espere a que la miniatura finalice la carga antes de publicar. + + idioma + Inglés + Chino + Francés + Alemán + Japonés + Ruso + Español + Indonesio + Italiano + Holandés + Turco + Polaco + Malayo + Portugués + vietnamita + Tailandés + Arabe + Checo + Croata + Camboyano + Coreano + Noruego + Rumano + Hindi + Griego + + Ninguno + Dominio publico + Material con derechos de autor + Creative Commons Atribución 4.0 Internacional + Atribución-CompartirIgual 4.0 Internacional + Atribución-SinDerivadas 4.0 Internacional + Atribución-NoComercial 4.0 Internacional + Atribución-NoComercial-CompartirIgual 4.0 Internacional + Atribución-NoComercial-SinDerivadas 4.0 Internacional + + + LBC + USD + + + + Uno o más elementos de contenido no se pudieron eliminar en este momento. Por favor, inténtelo de nuevo más tarde. + + ¿Está seguro de que desea eliminar los elemento de contenido seleccionados? + ¿Está seguro de que desea eliminar los elementos de contenido seleccionados? + + + El elemento de contenido se eliminó correctamente. + Los elementos de contenido se eliminaron correctamente. + + + + ¡Uy! Algo salió mal. + ID de instalación cargada. + Etiquetas locales cargadas conocidas y seguidas. + Tipo de cambio LBC/USD cargado. + Usuario autenticado. + Instalación registrada. + Suscripciones cargadas. + Suscripciones resueltas. + + + Contenido & interfaz de usuario + Otro + Habilitar tema oscuro + Mostrar contenido para adultos + Mostrar sugerencias de URL + Notificaciones + Suscripciones + Intereses de contenido + + Mantenga el servicio LBRY ejecutándose en segundo plano para mejorar el rendimiento de la billetera y la red + Participar en la red de datos (requiere reiniciar la aplicación y el servicio en segundo plano) + + + %1$s- Buscar + %1$s- Etiqueta + Buscar \'%1$s\' + Explora la etiqueta \'%1$s\' + Ver contenido en%1$s + Ver el%1$sCanal + + + No se encontraron resultados para \'%1$s\'. Por favor, introduzca un término de búsqueda diferente. + Puede buscar cualquier cosa, incluyendo películas, música, libros electrónicos, software y más. + No hay contenido relacionado para mostrar en este momento. + + + Saldo + Tu Actualmente tienes + Puede convertir sus créditos a USD y retirar el monto convertido mediante un intercambio. <a href=\"https://lbry.com/faq/exchanges\">Learn more</a>. + <a href=\"https://bittrex.com/Account/Register?referralCode=4M1-P30-BON\">Convierta créditos a USD en Bittrex</a> + También tienes + En stake + en propinas + en tus publicaciones + en tus apoyos + Obtenga más propinas al subir videos geniales + El servicio en segundo plano se está inicializando ... + El servicio en segundo plano aún se está iniciando. Puede explorar y ver contenido mientras tanto. + No puede hacer esto ahora porque el servicio en segundo plano todavía no está iniciado. + + Una copia de seguridad de su billetera se sincronizara con lbry.tv + Su billetera no está sincronizada actualmente con lbry.tv. Usted es responsable de respaldar su billetera. + <a href=\"https://lbry.com/faq/account-sync\">¿Qué significa esto?</a> + <a href=\"https://lbry.com/faq/how-to-backup-wallet#android\">¿Qué significa esto?</a> + + Recibir créditos + Use esta dirección de billetera para recibir créditos enviados por otro usuario (o usted mismo). + Obtener nueva dirección + Puede generar una nueva dirección en cualquier momento, y cualquier dirección anterior continuará funcionando. El uso de varias direcciones puede ser útil para realizar un seguimiento de los pagos entrantes de múltiples fuentes. + + Enviar créditos + Dirección del destinatario + bbFxRyXXXXXXXXXXXZD8nE7XTLUxYnddTs + Monto + Enviar + + Transacciones recientes + Ver todo + Recibir + Gastar + Publicar + Apoyo + Abandonar + Canal + Actualización de canal + Publicar actualización + + sincronización de la billetera + estado de sincronización + Encendido + Apagado + Email + No hay correo electrónico conectado + <a href=\"https://lbry.com/faq/how-to-backup-wallet#android\">Copia de seguridad manual</a> + <a href=\"https://lbry.com/faq/how-to-backup-wallet#sync\">Preguntas frecuentes sobre sincronización</a> + 0 + LBC + + Cuenta recomendada + Una cuenta de lbry.tv te permite el ganar recompensas, realizar un respaldo de tu billetera y mantener todo sincronizado. + Sin una cuenta, tendrás que asumir la responsabilidad de asegurar tu billetera y datos de LBRY. + Omitir cuenta + Registrarse + Dirección copiada + + Ingrese una dirección válida para enviar créditos a + Saldo insuficiente + Por favor ingrese una cantidad válida + Sus créditos no se pudieron enviar en este momento. Por favor, inténtelo de nuevo más tarde. + Cargando transacciones... + No hay transacciones recientes para mostrar. + No hay transacciones para mostrar en este momento. + cuota %1$s + Historial de transacciones + + + Has enviado%1$s crédito + Has enviado %1$s créditos + + + + Personaliza tus etiquetas + Ordenar contenido por + Contenido de + Contenido destacado + Nuevo contenido + Mejor contenido + Destacado + Destacado + Nuevo + Últimas 24 horas + Semana pasada + Mes pasado + Año pasado + Todo el tiempo + para + por + Filtrar por + Todos + Etiquetas que sigues + Personalizar + La vista seleccionada aún no está disponible. + Parece que todavía no has seguido ninguna etiqueta. + Buscar mas etiquetas + Aún no has seguido ninguna etiqueta. ¡Comience agregando etiquetas que le interesen! + No pudimos encontrar las nuevas etiquetas que no estás siguiendo. + La etiqueta \'%1$s\' ya se ha agregado. + No puede agregar más de 5 etiquetas. + Envía una propina + Enviar una propina a %1$s + Esto aparecerá como una propina para %1$s,lo que aumentará su capacidad de ser descubierto mientras está activo. <a href=\"https://lbry.com/faq/tipping\">Aprender más</a>. + Esto aparecerá como una propina para %1$s, lo que aumentará la capacidad del canal para ser descubierto mientras está activo. <a href=\"https://lbry.com/faq/tipping\">Aprender más</a>. + Cancelar + Republicar %1$s + ¡Vuelva a publicar su contenido favorito para ayudar a que más personas lo descubran! + Canal para publicar en + Mostrar avanzado + Ocultar avanzado + Nombre + 0.001 + ¡El contenido se volvió a publicar correctamente! + El nombre de reenvío contiene caracteres no válidos. + + ¡has enviado %1$s LBC como propina, Mahalo! + ¡has enviado %1$s LBC como propina, Mahalo! + + + + Por favor introduce una dirección de email. + you@example.com + Un email ha sido enviado a + Por favor da clic en el link del mensaje para completar el inicio de sesión + Reenviar + Continuar + Por favor introduce una dirección valida de email + Por favor sigue las instrucciones en el email que te enviamos a tu dirección para continuar. + Has iniciado sesión con éxito en lbry.tv + Recuperando la información de tu cuenta... + Aplicando datos de billetera... + Por favor introduce la contraseña que utilizaste para asegurar tu billetera. + Por favor introduce una contraseña para asegurar tu billetera. + Nota: por cuestiones de seguridad de tu billetera LBRY no puede restablecer tu contraseña. + Contraseña + Habilitar la sincronización + La operación de sincronización de billetera no se pudo completar en este momento. Por favor, inténtelo de nuevo más tarde. Si este problema persiste, envíe un correo electrónico a hello@lbry.com. + Número de teléfono + Por favor, introduzca su número de teléfono. + No estoy interesado + Verificación manual de recompensas + Esta cuenta debe someterse a revisión antes de poder participar en el programa de recompensas. Esto puede llevar desde varios minutos hasta varios días. + Si continúa viendo este mensaje, solicite ser verificado en el <a href=\"https://discordapp.com/invite/Z3bERWA\">LBRY Discord server</a>. + ¡Mientras tanto, disfruta de contenido gratuito! + Verificar número de teléfono + Ingrese el código de verificación enviado a%1$s + 0000 + Verificar + Por favor ingrese un número de teléfono válido. + Ingrese el código de verificación enviado a su número de teléfono. + + + Aún no ha agregado ninguna etiqueta. Agregue etiquetas para mejorar el alcance. + No pudimos encontrar nuevas etiquetas que aún no se hayan agregado. + + + No has creado ningún canal +¡Empieza ahora creando un nuevo canal! + Crear un canal + Crear un canal... + Editar canal + ¿Eliminar selección? + ¿Eliminar canal? + ¿Estás seguro de que deseas eliminar este canal? + El canal se eliminó correctamente. + El canal no se pudo eliminar en este momento. Por favor, inténtelo de nuevo más tarde. + Descripción + + No + Mostrar campos adicionales + Ocultar campos opcionales + Guardar + Nombre del canal + Titulo + \@ + Depositar + Estos LBC siguen siendo tuyos, Este es un deposito para reservar el nombre y puede ser deshecho en cualquier momento. + LBRY requiere acceso para descargar contenido a su dispositivo. + LBRY requiere acceso para cargar imágenes desde el almacenamiento de su dispositivo. + Seleccionar miniatura + Seleccionar imagen de portada + No se pudo determinar la ruta del archivo para la imagen seleccionada. Seleccione una imagen en una ubicación diferente. + Espera a que finalice la carga actual. + La solicitud de carga de la imagen ha fallado. Inténtalo de nuevo. + Subiendo... + Por favor ingrese un nombre de canal. + El nombre de tu canal contiene caracteres no válidos. + Ya has creado un canal con el mismo nombre. + Por favor ingrese un monto de depósito válido. + El deposito no puede ser mas alto que tu balance. + La solicitud de guardar el canal falló. Inténtalo de nuevo. + El canal se guardó correctamente. + El reclamo está pendiente de publicación en blockchain. Podrá acceder o editar el reclamo en unos momentos. + Pendiente + Crear + Uno o más canales no se pudieron eliminar en este momento. Por favor, inténtelo de nuevo más tarde. + + Se requiere un depósito mínimo de %1$s crédito. + Se requiere un depósito mínimo de %1$s créditos. + + + ¿Está seguro de que desea eliminar el canal seleccionado? + ¿Está seguro de que desea eliminar los canales seleccionados? + + + El canal se eliminó correctamente. + Los canales fueron eliminados con éxito. + + + + Los créditos LBRY le permiten publicar o comprar contenido. + Puede obtener créditos gratuitos por valor de $ %1$s después de proporcionar una dirección de correo electrónico. + <a href=\"https://lbry.com/faq/earn-credits\">Aprende más</a>. + Empezar + abc123 + Reclamación + Ingrese un código de recompensa personalizado para reclamar. + No reclamado + Código personalizado + hasta + ¿Eres una supermodelo o estrella de rock que recibió un código de recompensa personalizado? Reclámalo aquí. + + Has reclamado %1$s crédito como recompensa. + Has reclamado %1$s créditos como recompensa. + + + %1$s créditos disponibles + %1$s créditos disponibles + + + + Programa de Invitación LBRY + Puede obtener créditos adicionales por cada persona que invite a usar LBRY. + <a href=\"https://lbry.com/faq/invites\">Aprende más</a>. + Enlace de invitación + Comparta este enlace con amigos (o enemigos) y obtenga créditos cuando se unan a lbry.tv. + Tu enlace de invitación + Personalizar enlace de invitación + Invitar por Email + Invita a alguien que conozcas por correo electrónico y gana créditos cuando se una a lbry.tv. + amigo@imaginario.es + Invitar + Historial de Invitaciones + Gana créditos por invitar a un amigo, un enemigo, un amigo-enemigo o un enemigo-amigo. Todos necesitan libertad de contenido. + Recompensa + Reclamado + Reclamable + No Reclamable + Enlace de invitación copiado. + Invitación enviada a %1$s + + + Descargas + Compras + Historial + No tienes ningún contenido descargado en este dispositivo. + No has visto ningún contenido en este dispositivo. + No ha comprado ningún contenido en su cuenta. + Mantener + Estadísticas + Vídeo + Audio + Imágenes + MB + KB + GB + 0MB + + ¿Está seguro de que desea eliminar el archivo seleccionado de su dispositivo? + ¿Está seguro de que desea eliminar los archivos seleccionados de su dispositivo? + + + El archivo fue con éxito suprimido. + Los archivos fueron con éxito suprimidos. + + + + Sobre LBRY + Libertad de contenido + LBRY es un mercado digital gratuito, abierto y gestionado por la comunidad. Es una plataforma descentralizada de distribución de contenido entre pares para que los creadores carguen y compartan contenido y obtengan créditos LBRY por su esfuerzo. Los usuarios podrán encontrar una amplia selección de videos, música, libros electrónicos y otro contenido digital que les interese. + Sé social + Puede interactuar con el equipo LBRY y los miembros de la comunidad en Discord, Facebook, Instagram, Twitter o Reddit. + Info de la App + Cargando... + <a href=\"https://lbry.com/faq/what-is-lbry\">¿Qué es LBRY?</a> + <a href=\"https://lbry.com/faq/android-basics\">Android Básicos</a> + <a href=\"https://lbry.com/faq\">FAQ</a> + <a href=\"https://discordapp.com/invite/Z3bERWA\">Discord</a> + <a href=\"https://www.facebook.com/LBRYio\">Facebook</a> + <a href=\"https://www.instagram.com/LBRYio\">Instagram</a> + <a href=\"https://reddit.com/r/lbry\">Reddit</a> + <a href=\"https://t.me/lbryofficial\">Telegram</a> + <a href=\"https://twitter.com/LBRYio\">Twitter</a> + Actualizar preferencias de correo + Versión de la app + LBRY SDK + Plataforma + ID de instalación + Token de Firebase + Registros + Enviar registros + Correo electrónico conectado + Desconocido + No se pudo encontrar el archivo lbrynet.log. + El archivo lbrynet.log no se puede compartir debido a restricciones de permisos. + + + + + + + + + + + + + + + + + + + + + + + + + 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..59546685 --- /dev/null +++ b/app/src/main/res/values-night/colors.xml @@ -0,0 +1,61 @@ + + + #40B887 + #2F9176 + #38D9A9 + + #101010 + #000000 + #000000 + #CCCCCC + #000000 + #AAAAAA + #0E0E0E + #CC000000 + #55000000 + #F4E866 + + #33000000 + @color/nextLbryGreen + @color/nextLbryGreen + @color/nextLbryGreenSemiTransparent + + + #EEEEEE + #757575 + #999999 + #333333 + #5F5F5F + #333333 + + #40B887 + #2F9176 + #38D9A9 + #3338D9A9 + #329A7E + #77D510B8 + + #292929 + #0A0A0A + + #CCCCCC + #AAAAAA + #E5E5E5 + #F1F1F1 + #FFBB00 + #FF0000 + #E35454 + #FFFFFF + #3971DB + #2196f3 + + @color/nextLbryGreen + #F6A637 + #FF4A7D + #26BCF7 + + #252525 + + #CAEDB9 + #CC333333 + \ 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..d35fef96 100644 --- a/app/src/main/res/values/colors.xml +++ b/app/src/main/res/values/colors.xml @@ -1,12 +1,61 @@ - #40B89A - #303F9F - #FFFFFF + #40B887 + #2F9176 + #38D9A9 - #FF0000 - #00C000 - #FFFFFF + #EFEFEF + #FFFFFF + #FFFFFF + #333333 + #FFFFFF + #444444 + #F1F1F1 + #CC000000 + #55000000 + #F4E866 + + #333333 + #33000000 + @color/nextLbryGreen + @color/nextLbryGreen + @color/nextLbryGreenSemiTransparent + + #222222 + #AAAAAA + #333333 + #333333 + #777777 + #F1F1F1 + + #40B887 #2F9176 #38D9A9 + #3338D9A9 + #E3F6F1 + #77F255DA + + #DFDFDF + #F1F1F1 + + #CCCCCC + #AAAAAA + #E5E5E5 + #F1F1F1 + #FFBB00 + #FF0000 + #E35454 + #FFFFFF + #3971DB + #2196f3 + + @color/nextLbryGreen + #F6A637 + #FF4A7D + #26BCF7 + + #D5D5D5 + + #CAEDB9 + #CC333333 \ 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..141c0d66 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -1,14 +1,565 @@ - LBRY - io.lbry.browser.LBRY_ENGAGEMENT_CHANNEL + Open navigation drawer + Close navigation drawer - Running - Service Status - Stopped - START - STOP + + 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 + No content to display at this time. Please refine your selection or check back later. - Unit tests - RUN + + 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 + Discover new channels + + + Anonymous + + + Tags + Share + Repost + Tip + Edit + Delete + Download + Open + Report + Loading decentralized data... + Related Content + Share LBRY content + View + Play + Unsupported Content + Sorry, we are unable to display this content in the app. You can find the file %1$sin your downloads folder. + There\'s nothing at this location. + Publish something here + This content cannot be accessed at this time. Please try again later. + 0:00 + The file at "%1$s" does not exist. + Confirm Purchase + Delete file + Are you sure you want to remove this file from your device? + Failed to load %1$s. Please try again later. + There is no cast session available at this time. + Delete content? + Are you sure you want to unpublish this content? No files will be removed from your device. + The content was successfully deleted from the blockchain. + The content could not be deleted at this time. Please try again later. + + %1$s view + %1$s views + + + This will purchase "%1$s" for %2$s credit + This will purchase "%1$s" for %2$s credits + + + + There\'s nothing here yet.\nPlease check back later. + Content + Website + reposted + + %1$s follower + %1$s followers + + + + Record + Take a Photo + Upload a file + We could not find any videos on your device. Take a photo or record a video to get started. + Please wait while we load your videos... + LBRY requires access to be able to display and publish your videos, images and other files from your device. + LBRY requires access to your camera to record videos. + LBRY requires access to your camera to take photos. + Edit content + Mature tags + Price + Your content will be free. Press the toggle to set a price. + Content address + Randomize + Address + The address where people can find your content (ex. lbry://myvideo) + License + License description + Additional Options + Show extra fields + Hide extra fields + No file found to publish. + Video optimization + A thumbnail could not be automatically created from your content file. + Your video is being optimized for better support on a wide range of devices. You can fill out the remaining fields below while this is in progress. + Your video was successfully optimized for better playback across as many devices as possible. Please proceed to publish your content. + Your video could not be optimized. The file will be uploaded with no changes. + Completed Video Duration: %1$s + You cannot publish content right now because the background service is still initializing. + Your content was successfully published. It may take a few moments to appear on the blockchain. + Video optimization is in progress. If you wish to cancel, press Cancel at the bottom of the page. + Video optimization is in progress. + There is no camera app available to record videos on this device. + There is no camera app available to take photos on this device. + + Please provide a title. + Please specify an address where people can find your content. + Your content address contains invalid characters. + You have already published to the specified content address. Please enter a new address. + No file selected. Please choose a video or take a photo, or select a file before publishing. + Please enter a price or turn off the toggle to make your content free. + Please select a thumbnail to upload before publishing. + Please wait for the thumbnail to finish uploading before publishing. + + Language + English + Chinese + French + German + Japanese + Russian + Spanish + Indonesian + Italian + Dutch + Turkish + Polish + Malay + Portuguese + Vietnamese + Thai + Arabic + Czech + Croatian + Cambodian + Korean + Norwegian + Romanian + Hindi + Greek + + None + Public Domain + Copyrighted + Creative Commons Attribution 4.0 International + Creative Commons Attribution-ShareAlike 4.0 International + Creative Commons Attribution-NoDerivatives 4.0 International + Creative Commons Attribution-NonCommercial 4.0 International + Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International + Creative Commons Attribution-NonCommercial-NoDerivatives 4.0 International + + + LBC + USD + + + + One or more content items could not be deleted at this time. Please try again later. + + Are you sure you want to delete the selected content item? + Are you sure you want to delete the selected content items? + + + The content item was successfully deleted. + The content items were successfully deleted. + + + + Oops! Something went wrong. + Loaded Installation ID. + Loaded local known and followed tags. + Loaded LBC/USD exchange rate. + User authenticated. + Installation registered. + Loaded subscriptions. + Resolved subscriptions. + + + Content & User interface + Other + Enable dark theme + Show mature content + Show URL suggestions + Notifications + Subscriptions + Content Interests + + Keep the LBRY service running in the background for improved wallet and network performance + Participate in the data network (requires app and background service restart) + + + %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. + No related content to display at this time. + + + 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 initializing... + The background service is still initializing. You can explore and watch content in the mean time. + You cannot do this right now because the background service is still initializing. + + 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 continue 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 + Channel Update + Publish Update + + 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. + There are no transactions to display at this time. + fee %1$s + Transaction History + + + You sent %1$s credit + You sent %1$s credits + + + + Customize your tags + 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 + Filter for + Everyone + Tags you follow + Customize + The selected view is not yet available. + It looks like you have not followed any tags yet. + Search for more tags + You have not followed any tags yet. Get started by adding tags that you are interested in! + We could not find new tags that you\'re not following. + The \'%1$s\' tag has already been added. + You cannot add more than 5 tags. + Send a tip + Send a tip to %1$s + This will appear as a tip for %1$s, which will boost its ability to be discovered while active. <a href="https://lbry.com/faq/tipping">Learn more</a>. + This will appear as a tip for %1$s, which will boost the channel\'s ability to be discovered while active. <a href="https://lbry.com/faq/tipping">Learn more</a>. + Cancel + Repost %1$s + Repost your favorite content to help more people discover them! + Channel to post on + Show advanced + Hide advanced + Name + 0.001 + The content was successfully reposted! + The repost name contains invalid characters. + + You sent %1$s credit as a tip, Mahalo! + You sent %1$s credits as a tip, Mahalo! + + + + 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. + Phone Number + Please enter your phone number. + Not interested + Manual Reward Verification + This account must undergo review before you can participate in the rewards program. This can take anywhere from several minutes to several days. + If you continue to see this message, please request to be verified on the <a href="https://discordapp.com/invite/Z3bERWA">LBRY Discord server</a>. + Please enjoy free content in the meantime! + Verify Phone Number + Please enter the verification code sent to %1$s + 0000 + Verify + Please enter a valid phone number. + Please enter the verification code sent to your phone number. + + + You have not added any tags yet. Add tags to improve discovery. + We could not find new tags that have not been added yet. + + + You have not created a channel.\nStart now by creating a new channel! + Create a channel + Create a channel... + Edit channel + Delete selection? + Delete channel? + Are you sure you want to delete this channel? + The channel was successfully deleted. + The channel could not be deleted at this time. Please try again later. + Description + Yes + No + Show optional fields + Hide optional fields + Save + Channel name + Title + \@ + Deposit + This LBC remains yours. It is a deposit to reserve the name and can be undone at any time. + LBRY requires access to download content to your device. + LBRY requires access to load images from your device storage. + Select thumbnail + Select cover image + The file path could not be determined for the selected image. Please select an image in a different location. + Please wait for the current upload to finish. + The image upload request failed. Please try again. + Uploading... + Please enter a channel name. + Your channel name contains invalid characters. + You have already created a channel with the same name. + Please enter a valid deposit amount. + Deposit cannot be higher than your balance. + The channel save request failed. Please try again. + The channel was successfully saved. + The claim is pending publish on the blockchain. You will be able to access or edit the claim in a few moments. + Pending + Create + One or more channels could not be deleted at this time. Please try again later. + + A minimum deposit of %1$s credit is required. + A minimum deposit of %1$s credits is required. + + + Are you sure you want to delete the selected channel? + Are you sure you want to delete the selected channels? + + + The channel was successfully deleted. + The channels were successfully deleted. + + + + LBRY credits allow you to publish or purchase content. + You can obtain free credits worth $%1$s after you provide an email address. + <a href="https://lbry.com/faq/earn-credits">Learn more</a>. + Get started + abc123 + Claim + Please enter a custom reward code to claim. + Unclaimed + Custom Code + up to + Are you a supermodel or rockstar that received a custom reward code? Claim it here. + + You have claimed %1$s credit as a reward. + You have claimed %1$s credits as a reward. + + + %1$s available credit + %1$s available credits + + + + LBRY Invite Program + You can earn extra credits for each person you invite to use LBRY. + <a href="https://lbry.com/faq/invites">Learn more</a>. + Invite Link + Share this link with friends (or enemies) and get credits when they join lbry.tv. + Your invite link + Customize invite link + Invite by Email + Invite someone you know by email and earn credits when they join lbry.tv. + imaginary@friend.com + Invite + Invite History + Earn credits for invite a friend, an enemy, a frenemy, or an enefriend. Everyone needs content freedom. + Reward + Claimed + Claimable + Unclaimable + Invite link copied. + Invite sent to %1$s + + + Downloads + Purchases + History + You have not downloaded any content to this device. + You have not viewed any content on this device. + You have not purchased any content on your account. + Hide + Stats + Video + Audio + Images + MB + KB + GB + 0MB + + Are you sure you want to remove the selected file from your device? + Are you sure you want to remove the selected files from your device? + + + The file was successfully deleted. + The files were successfully deleted. + + + + About LBRY + Content Freedom + LBRY is a free, open, and community-run digital marketplace. It is a decentralized peer-to-peer content distribution platform for creators to upload and share content, and earn LBRY credits for their effort. Users will be able to find a wide selection of videos, music, ebooks and other digital content they are interested in. + Get Social + You can interact with the LBRY team and members of the community on Discord, Facebook, Instagram, Twitter or Reddit. + App info + Loading... + <a href="https://lbry.com/faq/what-is-lbry">What is LBRY?</a> + <a href="https://lbry.com/faq/android-basics">Android Basics</a> + <a href="https://lbry.com/faq">FAQ</a> + <a href="https://discordapp.com/invite/Z3bERWA">Discord</a> + <a href="https://www.facebook.com/LBRYio">Facebook</a> + <a href="https://www.instagram.com/LBRYio">Instagram</a> + <a href="https://reddit.com/r/lbry">Reddit</a> + <a href="https://t.me/lbryofficial">Telegram</a> + <a href="https://twitter.com/LBRYio">Twitter</a> + Update mailing preferences + App version + LBRY SDK + Platform + Installation ID + Firebase Token + Logs + Send log + Connected email + Unknown + The lbrynet.log file could not be found. + The lbrynet.log file cannot be shared due to permission restrictions. + + + + + + + + + + + + + + + + + + + + + + + + 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 index 6c65d1c7..72bd94c2 100644 --- a/app/src/main/res/xml/filepaths.xml +++ b/app/src/main/res/xml/filepaths.xml @@ -1,3 +1,6 @@ + + + \ 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..9999c440 --- /dev/null +++ b/app/src/main/res/xml/settings.xml @@ -0,0 +1,58 @@ + + + + + + + + + + + + + + + + + + + \ 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..e14841db 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.3' 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' - } + maven { url "https://jitpack.io" } } } + +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'