using Android.App; using Android.Content.PM; using Android.Runtime; using Android.OS; using Bit.Core; using System.Linq; using Bit.App.Abstractions; using Bit.Core.Utilities; using Bit.Core.Abstractions; using System.IO; using System; using Android.Content; using Bit.Droid.Utilities; using Bit.Droid.Receivers; using Bit.App.Models; using Bit.Core.Enums; using Android.Nfc; using Bit.App.Utilities; using System.Threading.Tasks; using AndroidX.Core.Content; using ZXing.Net.Mobile.Android; namespace Bit.Droid { [Activity( Label = "Bitwarden", Icon = "@mipmap/ic_launcher", Theme = "@style/LaunchTheme", MainLauncher = true, LaunchMode = LaunchMode.SingleTask, ConfigurationChanges = ConfigChanges.ScreenSize | ConfigChanges.Orientation | ConfigChanges.Keyboard | ConfigChanges.KeyboardHidden | ConfigChanges.Navigation)] [IntentFilter( new[] { Intent.ActionSend }, Categories = new[] { Intent.CategoryDefault }, DataMimeTypes = new[] { @"application/*", @"image/*", @"video/*", @"text/*" })] [Register("com.x8bit.bitwarden.MainActivity")] public class MainActivity : Xamarin.Forms.Platform.Android.FormsAppCompatActivity { private IDeviceActionService _deviceActionService; private IMessagingService _messagingService; private IBroadcasterService _broadcasterService; private IUserService _userService; private IAppIdService _appIdService; private IStorageService _storageService; private IEventService _eventService; private PendingIntent _clearClipboardPendingIntent; private PendingIntent _eventUploadPendingIntent; private AppOptions _appOptions; private string _activityKey = $"{nameof(MainActivity)}_{Java.Lang.JavaSystem.CurrentTimeMillis().ToString()}"; private Java.Util.Regex.Pattern _otpPattern = Java.Util.Regex.Pattern.Compile("^.*?([cbdefghijklnrtuv]{32,64})$"); protected override void OnCreate(Bundle savedInstanceState) { var eventUploadIntent = new Intent(this, typeof(EventUploadReceiver)); _eventUploadPendingIntent = PendingIntent.GetBroadcast(this, 0, eventUploadIntent, PendingIntentFlags.UpdateCurrent); var clearClipboardIntent = new Intent(this, typeof(ClearClipboardAlarmReceiver)); _clearClipboardPendingIntent = PendingIntent.GetBroadcast(this, 0, clearClipboardIntent, PendingIntentFlags.UpdateCurrent); var policy = new StrictMode.ThreadPolicy.Builder().PermitAll().Build(); StrictMode.SetThreadPolicy(policy); _deviceActionService = ServiceContainer.Resolve("deviceActionService"); _messagingService = ServiceContainer.Resolve("messagingService"); _broadcasterService = ServiceContainer.Resolve("broadcasterService"); _userService = ServiceContainer.Resolve("userService"); _appIdService = ServiceContainer.Resolve("appIdService"); _storageService = ServiceContainer.Resolve("storageService"); _eventService = ServiceContainer.Resolve("eventService"); TabLayoutResource = Resource.Layout.Tabbar; ToolbarResource = Resource.Layout.Toolbar; UpdateTheme(ThemeManager.GetTheme(true)); base.OnCreate(savedInstanceState); if (!CoreHelpers.InDebugMode()) { Window.AddFlags(Android.Views.WindowManagerFlags.Secure); } #if !FDROID var appCenterHelper = new AppCenterHelper(_appIdService, _userService); var appCenterTask = appCenterHelper.InitAsync(); #endif Xamarin.Essentials.Platform.Init(this, savedInstanceState); Xamarin.Forms.Forms.Init(this, savedInstanceState); _appOptions = GetOptions(); LoadApplication(new App.App(_appOptions)); _broadcasterService.Subscribe(_activityKey, (message) => { if (message.Command == "startEventTimer") { StartEventAlarm(); } else if (message.Command == "stopEventTimer") { var task = StopEventAlarmAsync(); } else if (message.Command == "finishMainActivity") { Xamarin.Forms.Device.BeginInvokeOnMainThread(() => Finish()); } else if (message.Command == "listenYubiKeyOTP") { ListenYubiKey((bool)message.Data); } else if (message.Command == "updatedTheme") { RestartApp(); } else if (message.Command == "exit") { ExitApp(); } else if (message.Command == "copiedToClipboard") { var task = ClearClipboardAlarmAsync(message.Data as Tuple); } }); } protected override void OnPause() { base.OnPause(); ListenYubiKey(false); } protected override void OnResume() { base.OnResume(); Xamarin.Essentials.Platform.OnResume(); if (_deviceActionService.SupportsNfc()) { try { _messagingService.Send("resumeYubiKey"); } catch { } } var setRestrictions = AndroidHelpers.SetPreconfiguredRestrictionSettingsAsync(this); } protected override void OnNewIntent(Intent intent) { base.OnNewIntent(intent); if (intent.GetBooleanExtra("generatorTile", false)) { _messagingService.Send("popAllAndGoToTabGenerator"); if (_appOptions != null) { _appOptions.GeneratorTile = true; } } else if (intent.GetBooleanExtra("myVaultTile", false)) { _messagingService.Send("popAllAndGoToTabMyVault"); if (_appOptions != null) { _appOptions.MyVaultTile = true; } } else if (intent.Action == Intent.ActionSend && intent.Type != null) { if (_appOptions != null) { _appOptions.CreateSend = GetCreateSendRequest(intent); } _messagingService.Send("popAllAndGoToTabSend"); } else { ParseYubiKey(intent.DataString); } } public async override void OnRequestPermissionsResult(int requestCode, string[] permissions, [GeneratedEnum] Permission[] grantResults) { if (requestCode == Constants.SelectFilePermissionRequestCode) { if (grantResults.Any(r => r != Permission.Granted)) { _messagingService.Send("selectFileCameraPermissionDenied"); } await _deviceActionService.SelectFileAsync(); } else { Xamarin.Essentials.Platform.OnRequestPermissionsResult(requestCode, permissions, grantResults); PermissionsHandler.OnRequestPermissionsResult(requestCode, permissions, grantResults); } base.OnRequestPermissionsResult(requestCode, permissions, grantResults); } protected override void OnActivityResult(int requestCode, Result resultCode, Intent data) { if (resultCode == Result.Ok && (requestCode == Constants.SelectFileRequestCode || requestCode == Constants.SaveFileRequestCode)) { Android.Net.Uri uri = null; string fileName = null; if (data != null && data.Data != null) { uri = data.Data; fileName = AndroidHelpers.GetFileName(ApplicationContext, uri); } else { // camera var file = new Java.IO.File(FilesDir, "temp_camera_photo.jpg"); uri = FileProvider.GetUriForFile(this, "com.x8bit.bitwarden.fileprovider", file); fileName = $"photo_{DateTime.UtcNow.ToString("yyyyMMddHHmmss")}.jpg"; } if (uri == null) { return; } if (requestCode == Constants.SaveFileRequestCode) { _messagingService.Send("selectSaveFileResult", new Tuple(uri.ToString(), fileName)); return; } try { using (var stream = ContentResolver.OpenInputStream(uri)) using (var memoryStream = new MemoryStream()) { stream.CopyTo(memoryStream); _messagingService.Send("selectFileResult", new Tuple(memoryStream.ToArray(), fileName ?? "unknown_file_name")); } } catch (Java.IO.FileNotFoundException) { return; } } } protected override void OnDestroy() { base.OnDestroy(); _broadcasterService.Unsubscribe(_activityKey); } private void ListenYubiKey(bool listen) { if (!_deviceActionService.SupportsNfc()) { return; } var adapter = NfcAdapter.GetDefaultAdapter(this); if (listen) { var intent = new Intent(this, Class); intent.AddFlags(ActivityFlags.SingleTop); var pendingIntent = PendingIntent.GetActivity(this, 0, intent, 0); // register for all NDEF tags starting with http och https var ndef = new IntentFilter(NfcAdapter.ActionNdefDiscovered); ndef.AddDataScheme("http"); ndef.AddDataScheme("https"); var filters = new IntentFilter[] { ndef }; try { // register for foreground dispatch so we'll receive tags according to our intent filters adapter.EnableForegroundDispatch(this, pendingIntent, filters, null); } catch { } } else { try { adapter.DisableForegroundDispatch(this); } catch { } } } private AppOptions GetOptions() { var options = new AppOptions { Uri = Intent.GetStringExtra("uri") ?? Intent.GetStringExtra("autofillFrameworkUri"), MyVaultTile = Intent.GetBooleanExtra("myVaultTile", false), GeneratorTile = Intent.GetBooleanExtra("generatorTile", false), FromAutofillFramework = Intent.GetBooleanExtra("autofillFramework", false), CreateSend = GetCreateSendRequest(Intent) }; var fillType = Intent.GetIntExtra("autofillFrameworkFillType", 0); if (fillType > 0) { options.FillType = (CipherType)fillType; } if (Intent.GetBooleanExtra("autofillFrameworkSave", false)) { options.SaveType = (CipherType)Intent.GetIntExtra("autofillFrameworkType", 0); options.SaveName = Intent.GetStringExtra("autofillFrameworkName"); options.SaveUsername = Intent.GetStringExtra("autofillFrameworkUsername"); options.SavePassword = Intent.GetStringExtra("autofillFrameworkPassword"); options.SaveCardName = Intent.GetStringExtra("autofillFrameworkCardName"); options.SaveCardNumber = Intent.GetStringExtra("autofillFrameworkCardNumber"); options.SaveCardExpMonth = Intent.GetStringExtra("autofillFrameworkCardExpMonth"); options.SaveCardExpYear = Intent.GetStringExtra("autofillFrameworkCardExpYear"); options.SaveCardCode = Intent.GetStringExtra("autofillFrameworkCardCode"); } return options; } private Tuple GetCreateSendRequest(Intent intent) { if (intent.Action == Intent.ActionSend && intent.Type != null) { var type = intent.Type; if (type.Contains("text/")) { var subject = intent.GetStringExtra(Intent.ExtraSubject); var text = intent.GetStringExtra(Intent.ExtraText); return new Tuple(SendType.Text, subject, null, text); } else { var data = intent.ClipData?.GetItemAt(0); var uri = data?.Uri; var filename = AndroidHelpers.GetFileName(ApplicationContext, uri); try { using (var stream = ContentResolver.OpenInputStream(uri)) using (var memoryStream = new MemoryStream()) { stream.CopyTo(memoryStream); return new Tuple(SendType.File, filename, memoryStream.ToArray(), null); } } catch (Java.IO.FileNotFoundException) { } } } return null; } private void ParseYubiKey(string data) { if (data == null) { return; } var otpMatch = _otpPattern.Matcher(data); if (otpMatch.Matches()) { var otp = otpMatch.Group(1); _messagingService.Send("gotYubiKeyOTP", otp); } } private void UpdateTheme(string theme) { if (theme == "light") { SetTheme(Resource.Style.LightTheme); } else if (theme == "dark") { SetTheme(Resource.Style.DarkTheme); } else if (theme == "black") { SetTheme(Resource.Style.BlackTheme); } else if (theme == "nord") { SetTheme(Resource.Style.NordTheme); } else { if (_deviceActionService.UsingDarkTheme()) { SetTheme(Resource.Style.DarkTheme); } else { SetTheme(Resource.Style.LightTheme); } } } private void RestartApp() { var intent = new Intent(this, typeof(MainActivity)); var pendingIntent = PendingIntent.GetActivity(this, 5923650, intent, PendingIntentFlags.CancelCurrent); var alarmManager = GetSystemService(AlarmService) as AlarmManager; var triggerMs = Java.Lang.JavaSystem.CurrentTimeMillis() + 500; alarmManager.Set(AlarmType.Rtc, triggerMs, pendingIntent); Java.Lang.JavaSystem.Exit(0); } private void ExitApp() { FinishAffinity(); Java.Lang.JavaSystem.Exit(0); } private async Task ClearClipboardAlarmAsync(Tuple data) { if (data.Item3) { return; } var clearMs = data.Item2; if (clearMs == null) { var clearSeconds = await _storageService.GetAsync(Constants.ClearClipboardKey); if (clearSeconds != null) { clearMs = clearSeconds.Value * 1000; } } if (clearMs == null) { return; } var triggerMs = Java.Lang.JavaSystem.CurrentTimeMillis() + clearMs.Value; var alarmManager = GetSystemService(AlarmService) as AlarmManager; alarmManager.Set(AlarmType.Rtc, triggerMs, _clearClipboardPendingIntent); } private void StartEventAlarm() { var alarmManager = GetSystemService(AlarmService) as AlarmManager; alarmManager.SetInexactRepeating(AlarmType.ElapsedRealtime, 120000, 300000, _eventUploadPendingIntent); } private async Task StopEventAlarmAsync() { var alarmManager = GetSystemService(AlarmService) as AlarmManager; alarmManager.Cancel(_eventUploadPendingIntent); await _eventService.UploadEventsAsync(); } } }