using System; using System.Collections.Generic; using System.Threading.Tasks; using System.Windows.Input; using Bit.App.Resources; using Bit.App.Utilities; using Bit.Core; using Bit.Core.Abstractions; using Bit.Core.Enums; using Bit.Core.Models.Domain; using Bit.Core.Utilities; using Xamarin.CommunityToolkit.ObjectModel; using Xamarin.Forms; namespace Bit.App.Pages { public class GeneratorPageViewModel : BaseViewModel { private readonly IPasswordGenerationService _passwordGenerationService; private readonly IPlatformUtilsService _platformUtilsService; private readonly IClipboardService _clipboardService; private readonly IUsernameGenerationService _usernameGenerationService; readonly LazyResolve _logger = new LazyResolve("logger"); private PasswordGenerationOptions _options; private UsernameGenerationOptions _usernameOptions; private PasswordGeneratorPolicyOptions _enforcedPolicyOptions; private string _password; private bool _isPassword; private bool _isUsername; private bool _uppercase; private bool _lowercase; private bool _number; private bool _special; private bool _allowAmbiguousChars; private int _minNumber; private int _minSpecial; private int _length = 5; private int _numWords = 3; private string _wordSeparator; private bool _capitalize; private bool _includeNumber; private string _username; private GeneratorType _generatorTypeSelected; private int _passwordTypeSelectedIndex; private bool _doneIniting; private bool _showTypePicker; private string _emailWebsite; private bool _showFirefoxRelayApiAccessToken; private bool _showAnonAddyApiAccessToken; private bool _showSimpleLoginApiKey; private UsernameEmailType _catchAllEmailTypeSelected; private UsernameEmailType _plusAddressedEmailTypeSelected; private bool _editMode; public GeneratorPageViewModel() { _passwordGenerationService = ServiceContainer.Resolve(); _platformUtilsService = ServiceContainer.Resolve(); _clipboardService = ServiceContainer.Resolve(); _usernameGenerationService = ServiceContainer.Resolve(); PageTitle = AppResources.Generator; GeneratorTypeOptions = new List { GeneratorType.Password, GeneratorType.Username }; PasswordTypeOptions = new List { AppResources.Password, AppResources.Passphrase }; UsernameTypeOptions = new List { UsernameType.PlusAddressedEmail, UsernameType.CatchAllEmail, UsernameType.ForwardedEmailAlias, UsernameType.RandomWord }; ForwardedEmailServiceTypeOptions = new List { ForwardedEmailServiceType.AnonAddy, ForwardedEmailServiceType.FirefoxRelay, ForwardedEmailServiceType.SimpleLogin }; UsernameEmailTypeOptions = new List { UsernameEmailType.Random, UsernameEmailType.Website }; UsernameTypePromptHelpCommand = new Command(UsernameTypePromptHelp); RegenerateCommand = new AsyncCommand(RegenerateAsync, onException: ex => OnSubmitException(ex), allowsMultipleExecutions: false); RegenerateUsernameCommand = new AsyncCommand(RegenerateUsernameAsync, onException: ex => OnSubmitException(ex), allowsMultipleExecutions: false); ToggleForwardedEmailHiddenValueCommand = new AsyncCommand(ToggleForwardedEmailHiddenValueAsync, onException: ex => OnSubmitException(ex), allowsMultipleExecutions: false); CopyCommand = new AsyncCommand(CopyAsync, onException: ex => OnSubmitException(ex), allowsMultipleExecutions: false); } public List GeneratorTypeOptions { get; set; } public List PasswordTypeOptions { get; set; } public List UsernameTypeOptions { get; set; } public List ForwardedEmailServiceTypeOptions { get; set; } public List UsernameEmailTypeOptions { get; set; } public Command UsernameTypePromptHelpCommand { get; set; } public ICommand RegenerateCommand { get; set; } public ICommand RegenerateUsernameCommand { get; set; } public ICommand ToggleForwardedEmailHiddenValueCommand { get; set; } public ICommand CopyCommand { get; set; } public string Password { get => _password; set => SetProperty(ref _password, value, additionalPropertyNames: new string[] { nameof(ColoredPassword) }); } public string Username { get => _username; set => SetProperty(ref _username, value, additionalPropertyNames: new string[] { nameof(ColoredUsername) }); } public string ColoredPassword => GeneratedValueFormatter.Format(Password); public string ColoredUsername => GeneratedValueFormatter.Format(Username); public bool IsPassword { get => _isPassword; set => SetProperty(ref _isPassword, value); } public bool IsUsername { get => _isUsername; set => SetProperty(ref _isUsername, value); } public bool ShowTypePicker { get => _showTypePicker; set => SetProperty(ref _showTypePicker, value); } public bool EditMode { get => _editMode; set => SetProperty(ref _editMode, value, additionalPropertyNames: new string[] { nameof(ShowUsernameEmailType) }); } public bool ShowUsernameEmailType { get => !string.IsNullOrWhiteSpace(EmailWebsite) || EditMode; } public int Length { get => _length; set { if (SetProperty(ref _length, value)) { _options.Length = value; var task = SliderInputAsync(); } } } public bool Uppercase { get => _uppercase; set { if (SetProperty(ref _uppercase, value)) { _options.Uppercase = value; var task = SaveOptionsAsync(); } } } public bool Lowercase { get => _lowercase; set { if (SetProperty(ref _lowercase, value)) { _options.Lowercase = value; var task = SaveOptionsAsync(); } } } public bool Number { get => _number; set { if (SetProperty(ref _number, value)) { _options.Number = value; var task = SaveOptionsAsync(); } } } public bool Special { get => _special; set { if (SetProperty(ref _special, value)) { _options.Special = value; var task = SaveOptionsAsync(); } } } public bool AllowAmbiguousChars { get => _allowAmbiguousChars; set { if (SetProperty(ref _allowAmbiguousChars, value, additionalPropertyNames: new string[] { nameof(AvoidAmbiguousChars) })) { _options.AllowAmbiguousChar = value; var task = SaveOptionsAsync(); } } } public bool AvoidAmbiguousChars { get => !AllowAmbiguousChars; set => AllowAmbiguousChars = !value; } public int MinNumber { get => _minNumber; set { if (SetProperty(ref _minNumber, value)) { _options.MinNumber = value; var task = SaveOptionsAsync(); } } } public int MinSpecial { get => _minSpecial; set { if (SetProperty(ref _minSpecial, value)) { _options.MinSpecial = value; var task = SaveOptionsAsync(); } } } public int NumWords { get => _numWords; set { if (SetProperty(ref _numWords, value)) { _options.NumWords = value; var task = SaveOptionsAsync(); } } } public string WordSeparator { get => _wordSeparator; set { if (value == null) { return; } var val = value.Trim(); if (SetProperty(ref _wordSeparator, val)) { _options.WordSeparator = val; var task = SaveOptionsAsync(); } } } public bool Capitalize { get => _capitalize; set { if (SetProperty(ref _capitalize, value)) { _options.Capitalize = value; var task = SaveOptionsAsync(); } } } public bool IncludeNumber { get => _includeNumber; set { if (SetProperty(ref _includeNumber, value)) { _options.Number = value; var task = SaveOptionsAsync(); } } } public string PlusAddressedEmail { get => _usernameOptions.PlusAddressedEmail; set { if (_usernameOptions.PlusAddressedEmail != value) { _usernameOptions.PlusAddressedEmail = value; TriggerPropertyChanged(nameof(PlusAddressedEmail)); SaveUsernameOptionsAsync(false).FireAndForget(); } } } public PasswordGeneratorPolicyOptions EnforcedPolicyOptions { get => _enforcedPolicyOptions; set => SetProperty(ref _enforcedPolicyOptions, value, additionalPropertyNames: new[] { nameof(IsPolicyInEffect) }); } public bool IsPolicyInEffect => _enforcedPolicyOptions.InEffect(); public GeneratorType GeneratorTypeSelected { get => _generatorTypeSelected; set { if (SetProperty(ref _generatorTypeSelected, value)) { IsUsername = value == GeneratorType.Username; TriggerPropertyChanged(nameof(GeneratorTypeSelected)); SaveOptionsAsync().FireAndForget(); SaveUsernameOptionsAsync(false).FireAndForget(); } } } public int PasswordTypeSelectedIndex { get => _passwordTypeSelectedIndex; set { if (SetProperty(ref _passwordTypeSelectedIndex, value)) { IsPassword = value == 0; TriggerPropertyChanged(nameof(PasswordTypeSelectedIndex)); SaveOptionsAsync().FireAndForget(); } } } public UsernameType UsernameTypeSelected { get => _usernameOptions.Type; set { if (_usernameOptions.Type != value) { _usernameOptions.Type = value; Username = Constants.DefaultUsernameGenerated; TriggerPropertyChanged(nameof(UsernameTypeSelected), new string[] { nameof(UsernameTypeDescriptionLabel) }); SaveUsernameOptionsAsync(false).FireAndForget(); } } } public string UsernameTypeDescriptionLabel => GetUsernameTypeLabelDescription(UsernameTypeSelected); public ForwardedEmailServiceType ForwardedEmailServiceSelected { get => _usernameOptions.ServiceType; set { if (_usernameOptions.ServiceType != value) { _usernameOptions.ServiceType = value; Username = Constants.DefaultUsernameGenerated; TriggerPropertyChanged(nameof(ForwardedEmailServiceSelected)); SaveUsernameOptionsAsync(false).FireAndForget(); } } } public string CatchAllEmailDomain { get => _usernameOptions.CatchAllEmailDomain; set { if (_usernameOptions.CatchAllEmailDomain != value) { _usernameOptions.CatchAllEmailDomain = value; TriggerPropertyChanged(nameof(CatchAllEmailDomain)); SaveUsernameOptionsAsync(false).FireAndForget(); } } } public string AnonAddyApiAccessToken { get => _usernameOptions.AnonAddyApiAccessToken; set { if (_usernameOptions.AnonAddyApiAccessToken != value) { _usernameOptions.AnonAddyApiAccessToken = value; TriggerPropertyChanged(nameof(AnonAddyApiAccessToken)); SaveUsernameOptionsAsync(false).FireAndForget(); } } } public bool ShowAnonAddyApiAccessToken { get { return _showAnonAddyApiAccessToken; } set => SetProperty(ref _showAnonAddyApiAccessToken, value, additionalPropertyNames: new string[] { nameof(ShowAnonAddyHiddenValueIcon) }); } public string ShowAnonAddyHiddenValueIcon => _showAnonAddyApiAccessToken ? BitwardenIcons.EyeSlash : BitwardenIcons.Eye; public string AnonAddyDomainName { get => _usernameOptions.AnonAddyDomainName; set { if (_usernameOptions.AnonAddyDomainName != value) { _usernameOptions.AnonAddyDomainName = value; TriggerPropertyChanged(nameof(AnonAddyDomainName)); SaveUsernameOptionsAsync(false).FireAndForget(); } } } public string FirefoxRelayApiAccessToken { get => _usernameOptions.FirefoxRelayApiAccessToken; set { if (_usernameOptions.FirefoxRelayApiAccessToken != value) { _usernameOptions.FirefoxRelayApiAccessToken = value; TriggerPropertyChanged(nameof(FirefoxRelayApiAccessToken)); SaveUsernameOptionsAsync(false).FireAndForget(); } } } public bool ShowFirefoxRelayApiAccessToken { get { return _showFirefoxRelayApiAccessToken; } set => SetProperty(ref _showFirefoxRelayApiAccessToken, value, additionalPropertyNames: new string[] { nameof(ShowFirefoxRelayHiddenValueIcon) }); } public string ShowFirefoxRelayHiddenValueIcon => _showFirefoxRelayApiAccessToken ? BitwardenIcons.EyeSlash : BitwardenIcons.Eye; public string SimpleLoginApiKey { get => _usernameOptions.SimpleLoginApiKey; set { if (_usernameOptions.SimpleLoginApiKey != value) { _usernameOptions.SimpleLoginApiKey = value; TriggerPropertyChanged(nameof(SimpleLoginApiKey)); SaveUsernameOptionsAsync(false).FireAndForget(); } } } public bool ShowSimpleLoginApiKey { get { return _showSimpleLoginApiKey; } set => SetProperty(ref _showSimpleLoginApiKey, value, additionalPropertyNames: new string[] { nameof(ShowSimpleLoginHiddenValueIcon) }); } public string ShowSimpleLoginHiddenValueIcon => _showSimpleLoginApiKey ? BitwardenIcons.EyeSlash : BitwardenIcons.Eye; public bool CapitalizeRandomWordUsername { get => _usernameOptions.CapitalizeRandomWordUsername; set { if (_usernameOptions.CapitalizeRandomWordUsername != value) { _usernameOptions.CapitalizeRandomWordUsername = value; TriggerPropertyChanged(nameof(CapitalizeRandomWordUsername)); SaveUsernameOptionsAsync().FireAndForget(); } } } public bool IncludeNumberRandomWordUsername { get => _usernameOptions.IncludeNumberRandomWordUsername; set { if (_usernameOptions.IncludeNumberRandomWordUsername != value) { _usernameOptions.IncludeNumberRandomWordUsername = value; TriggerPropertyChanged(nameof(IncludeNumberRandomWordUsername)); SaveUsernameOptionsAsync().FireAndForget(); } } } public UsernameEmailType PlusAddressedEmailTypeSelected { get => _plusAddressedEmailTypeSelected; set { if (SetProperty(ref _plusAddressedEmailTypeSelected, value)) { _usernameOptions.PlusAddressedEmailType = value; SaveUsernameOptionsAsync(false).FireAndForget(); } } } public UsernameEmailType CatchAllEmailTypeSelected { get => _catchAllEmailTypeSelected; set { if (SetProperty(ref _catchAllEmailTypeSelected, value)) { _usernameOptions.CatchAllEmailType = value; SaveUsernameOptionsAsync(false).FireAndForget(); } } } public string EmailWebsite { get => _emailWebsite; set => SetProperty(ref _emailWebsite, value, additionalPropertyNames: new string[] { nameof(ShowUsernameEmailType) }); } public async Task InitAsync() { (_options, EnforcedPolicyOptions) = await _passwordGenerationService.GetOptionsAsync(); LoadFromOptions(); await RegenerateAsync(); _usernameOptions = await _usernameGenerationService.GetOptionsAsync(); if (!EditMode) { _usernameOptions.CatchAllEmailType = _usernameOptions.PlusAddressedEmailType = UsernameEmailType.Random; } TriggerUsernamePropertiesChanged(); Username = Constants.DefaultUsernameGenerated; _doneIniting = true; } public async Task RegenerateAsync() { Password = await _passwordGenerationService.GeneratePasswordAsync(_options); await _passwordGenerationService.AddHistoryAsync(Password); } public async Task RegenerateUsernameAsync() { Username = await _usernameGenerationService.GenerateAsync(_usernameOptions); } public void RedrawPassword() { if (!string.IsNullOrEmpty(_password)) { TriggerPropertyChanged(nameof(ColoredPassword)); } } public void RedrawUsername() { if (!string.IsNullOrEmpty(_username)) { TriggerPropertyChanged(nameof(ColoredUsername)); } } public async Task SaveOptionsAsync(bool regenerate = true) { if (!_doneIniting) { return; } SetOptions(); _passwordGenerationService.NormalizeOptions(_options, _enforcedPolicyOptions); await _passwordGenerationService.SaveOptionsAsync(_options); LoadFromOptions(); if (regenerate) { await RegenerateAsync(); } } public async Task SaveUsernameOptionsAsync(bool regenerate = true) { if (!_doneIniting) { return; } await _usernameGenerationService.SaveOptionsAsync(_usernameOptions); if (regenerate) { await RegenerateUsernameAsync(); } } public async Task SliderChangedAsync() { await SaveOptionsAsync(false); await _passwordGenerationService.AddHistoryAsync(Password); } public async Task SliderInputAsync() { if (!_doneIniting) { return; } SetOptions(); _passwordGenerationService.NormalizeOptions(_options, _enforcedPolicyOptions); Password = await _passwordGenerationService.GeneratePasswordAsync(_options); } public async Task CopyAsync() { await _clipboardService.CopyTextAsync(IsUsername ? Username : Password); _platformUtilsService.ShowToastForCopiedValue(IsUsername ? AppResources.Username : AppResources.Password); } public void UsernameTypePromptHelp() { try { _platformUtilsService.LaunchUri("https://bitwarden.com/help/generator/#username-types"); } catch (Exception ex) { _logger.Value.Exception(ex); Page.DisplayAlert(AppResources.AnErrorHasOccurred, AppResources.GenericErrorMessage, AppResources.Ok); } } private void LoadFromOptions() { AllowAmbiguousChars = _options.AllowAmbiguousChar.GetValueOrDefault(); PasswordTypeSelectedIndex = _options.Type == "passphrase" ? 1 : 0; IsPassword = PasswordTypeSelectedIndex == 0; MinNumber = _options.MinNumber.GetValueOrDefault(); MinSpecial = _options.MinSpecial.GetValueOrDefault(); Special = _options.Special.GetValueOrDefault(); Number = _options.Number.GetValueOrDefault(); NumWords = _options.NumWords.GetValueOrDefault(); WordSeparator = _options.WordSeparator; Uppercase = _options.Uppercase.GetValueOrDefault(); Lowercase = _options.Lowercase.GetValueOrDefault(); Length = _options.Length.GetValueOrDefault(5); Capitalize = _options.Capitalize.GetValueOrDefault(); IncludeNumber = _options.IncludeNumber.GetValueOrDefault(); } private void TriggerUsernamePropertiesChanged() { TriggerPropertyChanged(nameof(CatchAllEmailTypeSelected)); TriggerPropertyChanged(nameof(PlusAddressedEmailTypeSelected)); TriggerPropertyChanged(nameof(IncludeNumberRandomWordUsername)); TriggerPropertyChanged(nameof(CapitalizeRandomWordUsername)); TriggerPropertyChanged(nameof(SimpleLoginApiKey)); TriggerPropertyChanged(nameof(FirefoxRelayApiAccessToken)); TriggerPropertyChanged(nameof(AnonAddyDomainName)); TriggerPropertyChanged(nameof(AnonAddyApiAccessToken)); TriggerPropertyChanged(nameof(CatchAllEmailDomain)); TriggerPropertyChanged(nameof(ForwardedEmailServiceSelected)); TriggerPropertyChanged(nameof(UsernameTypeSelected)); TriggerPropertyChanged(nameof(PasswordTypeSelectedIndex)); TriggerPropertyChanged(nameof(GeneratorTypeSelected)); TriggerPropertyChanged(nameof(PlusAddressedEmail)); TriggerPropertyChanged(nameof(GeneratorTypeSelected)); TriggerPropertyChanged(nameof(UsernameTypeDescriptionLabel)); } private void SetOptions() { _options.AllowAmbiguousChar = AllowAmbiguousChars; _options.Type = PasswordTypeSelectedIndex == 1 ? "passphrase" : "password"; _options.MinNumber = MinNumber; _options.MinSpecial = MinSpecial; _options.Special = Special; _options.NumWords = NumWords; _options.Number = Number; _options.WordSeparator = WordSeparator; _options.Uppercase = Uppercase; _options.Lowercase = Lowercase; _options.Length = Length; _options.Capitalize = Capitalize; _options.IncludeNumber = IncludeNumber; } private async void OnSubmitException(Exception ex) { _logger.Value.Exception(ex); if (IsUsername && UsernameTypeSelected == UsernameType.ForwardedEmailAlias) { await Device.InvokeOnMainThreadAsync(() => Page.DisplayAlert( AppResources.AnErrorHasOccurred, string.Format(AppResources.UnknownXErrorMessage, ForwardedEmailServiceSelected), AppResources.Ok)); } else { await Device.InvokeOnMainThreadAsync(() => Page.DisplayAlert(AppResources.AnErrorHasOccurred, AppResources.GenericErrorMessage, AppResources.Ok)); } } private string GetUsernameTypeLabelDescription(UsernameType value) { switch (value) { case UsernameType.PlusAddressedEmail: return AppResources.PlusAddressedEmailDescription; case UsernameType.CatchAllEmail: return AppResources.CatchAllEmailDescription; case UsernameType.ForwardedEmailAlias: return AppResources.ForwardedEmailDescription; default: return string.Empty; } } private async Task ToggleForwardedEmailHiddenValueAsync() { switch (ForwardedEmailServiceSelected) { case ForwardedEmailServiceType.AnonAddy: ShowAnonAddyApiAccessToken = !ShowAnonAddyApiAccessToken; break; case ForwardedEmailServiceType.FirefoxRelay: ShowFirefoxRelayApiAccessToken = !ShowFirefoxRelayApiAccessToken; break; case ForwardedEmailServiceType.SimpleLogin: ShowSimpleLoginApiKey = !ShowSimpleLoginApiKey; break; } } } }