mirror of
https://github.com/bitwarden/mobile
synced 2025-12-28 14:13:25 +00:00
* Account Switching (#1720) * Account switching * WIP * wip * wip * updates to send test logic * fixed Send tests * fixes for theme handling on account switching and re-adding existing account * switch fixes * fixes * fixes * cleanup * vault timeout fixes * account list status enhancements * logout fixes and token handling improvements * merge latest (#1727) * remove duplicate dependency * fix for initial login token storage paradox (#1730) * Fix avatar color update toolbar item issue on iOS for account switching (#1735) * Updated account switching menu UI (#1733) * updated account switching menu UI * additional changes * add key suffix to constant * GetFirstLetters method tweaks * Fix crash on account switching when logging out when having more than user at a time (#1740) * single account migration to multi-account on app update (#1741) * Account Switching Tap to dismiss (#1743) * Added tap to dismiss on the Account switching overlay and improved a bit the code * Fix account switching overlay background transparent on the proper place * Fixed transparent background and the shadow on the account switching overlay * Fix iOS top space on Account switching list overlay after modal (#1746) * Fix top space added to Account switching list overlay after closing modal * Fix top space added to Account switching list overlay after closing modal on lock, login and home views just in case we add modals in the future there as well * Usability: dismiss account list on certain events (#1748) * dismiss account list on certain events * use new FireAndForget method for back button logic * Create and use Account Switching overlay control (#1753) * Added Account switching overlay control and its own ViewModel and refactored accordingly * Fix account switching Accounts list binding update * Implemented dismiss account switching overlay when changing tabs and when selecting the same tab. Also updated the deprecated listener on CustomTabbedRenderer on Android (#1755) * Overriden Equals on AvatarImageSource so it doesn't get set multiple times when it's the same image thus producing blinking on tab chaged (#1756) * Usability improvements for logout on vault timeout (#1781) * accountswitching fixes (#1784) * Fix for invalid PIN lock state when switching accounts (#1792) * fix for pin lock flow * named tuple values and updated async * clear send service cache on account switch (#1796) * Global theme and account removal (#1793) * Global theme and account removal * remove redundant call to hide account list overlay * cleanup and additional tweaks * add try/catch to remove account dialog flow Co-authored-by: Federico Maccaroni <fedemkr@gmail.com>
170 lines
6.5 KiB
C#
170 lines
6.5 KiB
C#
using Android;
|
|
using Android.App;
|
|
using Android.Content;
|
|
using Android.OS;
|
|
using Android.Runtime;
|
|
using Android.Service.Autofill;
|
|
using Android.Widget;
|
|
using Bit.Core;
|
|
using Bit.Core.Abstractions;
|
|
using Bit.Core.Enums;
|
|
using Bit.Core.Utilities;
|
|
#if !FDROID
|
|
using Microsoft.AppCenter.Crashes;
|
|
#endif
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
|
|
namespace Bit.Droid.Autofill
|
|
{
|
|
[Service(Permission = Manifest.Permission.BindAutofillService, Label = "Bitwarden")]
|
|
[IntentFilter(new string[] { "android.service.autofill.AutofillService" })]
|
|
[MetaData("android.autofill", Resource = "@xml/autofillservice")]
|
|
[Register("com.x8bit.bitwarden.Autofill.AutofillService")]
|
|
public class AutofillService : Android.Service.Autofill.AutofillService
|
|
{
|
|
private ICipherService _cipherService;
|
|
private IVaultTimeoutService _vaultTimeoutService;
|
|
private IPolicyService _policyService;
|
|
private IStateService _stateService;
|
|
|
|
public async override void OnFillRequest(FillRequest request, CancellationSignal cancellationSignal,
|
|
FillCallback callback)
|
|
{
|
|
try
|
|
{
|
|
var structure = request.FillContexts?.LastOrDefault()?.Structure;
|
|
if (structure == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
var parser = new Parser(structure, ApplicationContext);
|
|
parser.Parse();
|
|
|
|
if (_stateService == null)
|
|
{
|
|
_stateService = ServiceContainer.Resolve<IStateService>("stateService");
|
|
}
|
|
|
|
var shouldAutofill = await parser.ShouldAutofillAsync(_stateService);
|
|
if (!shouldAutofill)
|
|
{
|
|
return;
|
|
}
|
|
|
|
var inlineAutofillEnabled = await _stateService.GetInlineAutofillEnabledAsync() ?? true;
|
|
|
|
if (_vaultTimeoutService == null)
|
|
{
|
|
_vaultTimeoutService = ServiceContainer.Resolve<IVaultTimeoutService>("vaultTimeoutService");
|
|
}
|
|
|
|
List<FilledItem> items = null;
|
|
await _vaultTimeoutService.CheckVaultTimeoutAsync();
|
|
var locked = await _vaultTimeoutService.IsLockedAsync();
|
|
if (!locked)
|
|
{
|
|
if (_cipherService == null)
|
|
{
|
|
_cipherService = ServiceContainer.Resolve<ICipherService>("cipherService");
|
|
}
|
|
items = await AutofillHelpers.GetFillItemsAsync(parser, _cipherService);
|
|
}
|
|
|
|
// build response
|
|
var response = AutofillHelpers.CreateFillResponse(parser, items, locked, inlineAutofillEnabled, request);
|
|
var disableSavePrompt = await _stateService.GetAutofillDisableSavePromptAsync();
|
|
if (!disableSavePrompt.GetValueOrDefault())
|
|
{
|
|
AutofillHelpers.AddSaveInfo(parser, request, response, parser.FieldCollection);
|
|
}
|
|
callback.OnSuccess(response.Build());
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
#if !FDROID
|
|
Crashes.TrackError(e);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
public async override void OnSaveRequest(SaveRequest request, SaveCallback callback)
|
|
{
|
|
try
|
|
{
|
|
var structure = request.FillContexts?.LastOrDefault()?.Structure;
|
|
if (structure == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (_stateService == null)
|
|
{
|
|
_stateService = ServiceContainer.Resolve<IStateService>("stateService");
|
|
}
|
|
|
|
var disableSavePrompt = await _stateService.GetAutofillDisableSavePromptAsync();
|
|
if (disableSavePrompt.GetValueOrDefault())
|
|
{
|
|
return;
|
|
}
|
|
|
|
_policyService ??= ServiceContainer.Resolve<IPolicyService>("policyService");
|
|
|
|
var personalOwnershipPolicyApplies = await _policyService.PolicyAppliesToUser(PolicyType.PersonalOwnership);
|
|
if (personalOwnershipPolicyApplies)
|
|
{
|
|
return;
|
|
}
|
|
|
|
var parser = new Parser(structure, ApplicationContext);
|
|
parser.Parse();
|
|
|
|
var savedItem = parser.FieldCollection.GetSavedItem();
|
|
if (savedItem == null)
|
|
{
|
|
Toast.MakeText(this, "Unable to save this form.", ToastLength.Short).Show();
|
|
return;
|
|
}
|
|
|
|
var intent = new Intent(this, typeof(MainActivity));
|
|
intent.SetFlags(ActivityFlags.NewTask | ActivityFlags.ClearTop);
|
|
intent.PutExtra("autofillFramework", true);
|
|
intent.PutExtra("autofillFrameworkSave", true);
|
|
intent.PutExtra("autofillFrameworkType", (int)savedItem.Type);
|
|
switch (savedItem.Type)
|
|
{
|
|
case CipherType.Login:
|
|
intent.PutExtra("autofillFrameworkName", parser.Uri
|
|
.Replace(Constants.AndroidAppProtocol, string.Empty)
|
|
.Replace("https://", string.Empty)
|
|
.Replace("http://", string.Empty));
|
|
intent.PutExtra("autofillFrameworkUri", parser.Uri);
|
|
intent.PutExtra("autofillFrameworkUsername", savedItem.Login.Username);
|
|
intent.PutExtra("autofillFrameworkPassword", savedItem.Login.Password);
|
|
break;
|
|
case CipherType.Card:
|
|
intent.PutExtra("autofillFrameworkCardName", savedItem.Card.Name);
|
|
intent.PutExtra("autofillFrameworkCardNumber", savedItem.Card.Number);
|
|
intent.PutExtra("autofillFrameworkCardExpMonth", savedItem.Card.ExpMonth);
|
|
intent.PutExtra("autofillFrameworkCardExpYear", savedItem.Card.ExpYear);
|
|
intent.PutExtra("autofillFrameworkCardCode", savedItem.Card.Code);
|
|
break;
|
|
default:
|
|
Toast.MakeText(this, "Unable to save this type of form.", ToastLength.Short).Show();
|
|
return;
|
|
}
|
|
StartActivity(intent);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
#if !FDROID
|
|
Crashes.TrackError(e);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
}
|