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 android.support.v4.content.FileProvider; import android.support.v4.app.NotificationCompat; import android.support.v4.app.NotificationManagerCompat; import android.support.v4.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.ReactContextBaseJavaModule; import com.facebook.react.bridge.ReactMethod; import com.squareup.picasso.Picasso; import java.io.File; import java.io.IOException; 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.LbrynetService; import io.lbry.browser.R; import io.lbry.browser.Utils; public class UtilityModule extends ReactContextBaseJavaModule { private static final Map<String, Integer> activeNotifications = new HashMap<String, Integer>(); 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"; 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 } } @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 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<Void, Void, Bitmap>() { 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); } } }